| author | truebrain | 
| Mon, 31 Mar 2008 16:11:00 +0000 | |
| branch | noai | 
| changeset 9849 | edf90dda4ec4 | 
| parent 9826 | 9707ad4c9b60 | 
| child 10455 | 22c441f5adf9 | 
| permissions | -rw-r--r-- | 
| 2186 | 1 | /* $Id$ */ | 
| 2 | ||
| 9601 
b499fdd106d5
(svn r9713) [NoAI] -Sync with trunk (r9631:9712).
 rubidium parents: 
9574diff
changeset | 3 | /** @file spritecache.cpp */ | 
| 9574 | 4 | |
| 0 | 5 | #include "stdafx.h" | 
| 1891 
92a3b0aa0946
(svn r2397) - CodeChange: rename all "ttd" files to "openttd" files.
 Darkvater parents: 
1844diff
changeset | 6 | #include "openttd.h" | 
| 9628 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 7 | #include "variables.h" | 
| 1299 
0a6510cc889b
(svn r1803) Move debugging stuff into files of it's own
 tron parents: 
1250diff
changeset | 8 | #include "debug.h" | 
| 1349 
07514c2cc6d1
(svn r1853) Move spritecache function declarations into a header of their own and use SpriteID as parameter type where appropriate
 tron parents: 
1348diff
changeset | 9 | #include "spritecache.h" | 
| 0 | 10 | #include "fileio.h" | 
| 9626 | 11 | #include "spriteloader/grf.hpp" | 
| 9723 
eee46cb39750
(svn r11796) [NoAI] -Sync: with trunk r11502:11795.
 rubidium parents: 
9722diff
changeset | 12 | #include "core/alloc_func.hpp" | 
| 
eee46cb39750
(svn r11796) [NoAI] -Sync: with trunk r11502:11795.
 rubidium parents: 
9722diff
changeset | 13 | #include "core/math_func.hpp" | 
| 9628 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 14 | #ifdef WITH_PNG | 
| 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 15 | #include "spriteloader/png.hpp" | 
| 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 16 | #endif /* WITH_PNG */ | 
| 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 17 | #include "blitter/factory.hpp" | 
| 0 | 18 | |
| 9724 
b39bc69bb2f2
(svn r12051) [NoAI] -Sync: with trunk (r11795:12050).
 rubidium parents: 
9723diff
changeset | 19 | #include "table/sprites.h" | 
| 
b39bc69bb2f2
(svn r12051) [NoAI] -Sync: with trunk (r11795:12050).
 rubidium parents: 
9723diff
changeset | 20 | |
| 9626 | 21 | /* Default of 4MB spritecache */ | 
| 22 | uint _sprite_cache_size = 4; | |
| 0 | 23 | |
| 24 | ||
| 6574 
e1d1a12faaf7
(svn r9051) -Codechange: typedef [enum|struct] Y {} X; -> [enum|struct] X {};
 rubidium parents: 
6573diff
changeset | 25 | struct SpriteCache {
 | 
| 9826 
9707ad4c9b60
(svn r12462) [NoAI] -Sync: with trunk r12304:12461.
 rubidium parents: 
9732diff
changeset | 26 | void *ptr; | 
| 9628 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 27 | uint32 id; | 
| 9826 
9707ad4c9b60
(svn r12462) [NoAI] -Sync: with trunk r12304:12461.
 rubidium parents: 
9732diff
changeset | 28 | uint32 file_pos; | 
| 9724 
b39bc69bb2f2
(svn r12051) [NoAI] -Sync: with trunk (r11795:12050).
 rubidium parents: 
9723diff
changeset | 29 | uint16 file_slot; | 
| 9826 
9707ad4c9b60
(svn r12462) [NoAI] -Sync: with trunk r12304:12461.
 rubidium parents: 
9732diff
changeset | 30 | int16 lru; | 
| 6574 
e1d1a12faaf7
(svn r9051) -Codechange: typedef [enum|struct] Y {} X; -> [enum|struct] X {};
 rubidium parents: 
6573diff
changeset | 31 | }; | 
| 5755 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 32 | |
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 33 | |
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 34 | static uint _spritecache_items = 0; | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 35 | static SpriteCache *_spritecache = NULL; | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 36 | |
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 37 | |
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 38 | static inline SpriteCache *GetSpriteCache(uint index) | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 39 | {
 | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 40 | return &_spritecache[index]; | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 41 | } | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 42 | |
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 43 | |
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 44 | static SpriteCache *AllocateSpriteCache(uint index) | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 45 | {
 | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 46 | 	if (index >= _spritecache_items) {
 | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 47 | /* Add another 1024 items to the 'pool' */ | 
| 9722 
ebf0ece7d8f6
(svn r11503) [NoAI] -Sync: with trunk r11308:11502.
 rubidium parents: 
9703diff
changeset | 48 | uint items = Align(index + 1, 1024); | 
| 5755 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 49 | |
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 50 | DEBUG(sprite, 4, "Increasing sprite cache to %d items (%d bytes)", items, items * sizeof(*_spritecache)); | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 51 | |
| 5860 
7fdc9b423ba1
(svn r8066) - Codechange: MallocT(), CallocT(), ReallocT() now return the pointer to allocated memory instead of modifying the pointer given as parameter
 KUDr parents: 
5838diff
changeset | 52 | _spritecache = ReallocT(_spritecache, items); | 
| 5755 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 53 | |
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 54 | /* Reset the new items and update the count */ | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 55 | memset(_spritecache + _spritecache_items, 0, (items - _spritecache_items) * sizeof(*_spritecache)); | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 56 | _spritecache_items = items; | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 57 | } | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 58 | |
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 59 | return GetSpriteCache(index); | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 60 | } | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 61 | |
| 0 | 62 | |
| 6574 
e1d1a12faaf7
(svn r9051) -Codechange: typedef [enum|struct] Y {} X; -> [enum|struct] X {};
 rubidium parents: 
6573diff
changeset | 63 | struct MemBlock {
 | 
| 1353 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 64 | uint32 size; | 
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 65 | byte data[VARARRAY_SIZE]; | 
| 6574 
e1d1a12faaf7
(svn r9051) -Codechange: typedef [enum|struct] Y {} X; -> [enum|struct] X {};
 rubidium parents: 
6573diff
changeset | 66 | }; | 
| 1353 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 67 | |
| 0 | 68 | static uint _sprite_lru_counter; | 
| 1353 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 69 | static MemBlock *_spritecache_ptr; | 
| 0 | 70 | static int _compact_cache_counter; | 
| 71 | ||
| 6573 | 72 | static void CompactSpriteCache(); | 
| 0 | 73 | |
| 6573 | 74 | static bool ReadSpriteHeaderSkipData() | 
| 0 | 75 | {
 | 
| 2329 
f68428464540
(svn r2855) Make ReadSpriteHeaderSkipData() responsible for detecting the end of a grf file instead of its callers - this simplifies the code a bit
 tron parents: 
2321diff
changeset | 76 | uint16 num = FioReadWord(); | 
| 0 | 77 | byte type; | 
| 78 | ||
| 2329 
f68428464540
(svn r2855) Make ReadSpriteHeaderSkipData() responsible for detecting the end of a grf file instead of its callers - this simplifies the code a bit
 tron parents: 
2321diff
changeset | 79 | if (num == 0) return false; | 
| 
f68428464540
(svn r2855) Make ReadSpriteHeaderSkipData() responsible for detecting the end of a grf file instead of its callers - this simplifies the code a bit
 tron parents: 
2321diff
changeset | 80 | |
| 0 | 81 | type = FioReadByte(); | 
| 82 | 	if (type == 0xFF) {
 | |
| 2342 
c19fb4f2df30
(svn r2868) Change the way NewGRFs are loaded: The loading process i no longer bolted onto the normal graphics loading.
 tron parents: 
2340diff
changeset | 83 | FioSkipBytes(num); | 
| 5150 
247343201280
(svn r7242) -Fix: Don't load 1-byte pseudo sprites, as used in some NewGRF sets. If the sprite is ever drawn this will result in a "missing sprite" error instead of undefined misbehaviour leading to segmentation faults...
 peter1138 parents: 
4522diff
changeset | 84 | /* Some NewGRF files have "empty" pseudo-sprites which are 1 | 
| 
247343201280
(svn r7242) -Fix: Don't load 1-byte pseudo sprites, as used in some NewGRF sets. If the sprite is ever drawn this will result in a "missing sprite" error instead of undefined misbehaviour leading to segmentation faults...
 peter1138 parents: 
4522diff
changeset | 85 | * byte long. Catch these so the sprites won't be displayed. */ | 
| 
247343201280
(svn r7242) -Fix: Don't load 1-byte pseudo sprites, as used in some NewGRF sets. If the sprite is ever drawn this will result in a "missing sprite" error instead of undefined misbehaviour leading to segmentation faults...
 peter1138 parents: 
4522diff
changeset | 86 | return num != 1; | 
| 0 | 87 | } | 
| 184 
dbeaaaf8b2bb
(svn r185) -Fix: [1016954] Cached_sprites does now work again
 truelight parents: 
182diff
changeset | 88 | |
| 0 | 89 | FioSkipBytes(7); | 
| 90 | num -= 8; | |
| 2329 
f68428464540
(svn r2855) Make ReadSpriteHeaderSkipData() responsible for detecting the end of a grf file instead of its callers - this simplifies the code a bit
 tron parents: 
2321diff
changeset | 91 | if (num == 0) return true; | 
| 0 | 92 | |
| 93 | 	if (type & 2) {
 | |
| 94 | FioSkipBytes(num); | |
| 1355 | 95 | 	} else {
 | 
| 96 | 		while (num > 0) {
 | |
| 97 | int8 i = FioReadByte(); | |
| 98 | 			if (i >= 0) {
 | |
| 99 | num -= i; | |
| 100 | FioSkipBytes(i); | |
| 101 | 			} else {
 | |
| 102 | i = -(i >> 3); | |
| 103 | num -= i; | |
| 104 | FioReadByte(); | |
| 105 | } | |
| 0 | 106 | } | 
| 107 | } | |
| 2329 
f68428464540
(svn r2855) Make ReadSpriteHeaderSkipData() responsible for detecting the end of a grf file instead of its callers - this simplifies the code a bit
 tron parents: 
2321diff
changeset | 108 | |
| 
f68428464540
(svn r2855) Make ReadSpriteHeaderSkipData() responsible for detecting the end of a grf file instead of its callers - this simplifies the code a bit
 tron parents: 
2321diff
changeset | 109 | return true; | 
| 0 | 110 | } | 
| 111 | ||
| 3565 
03d870cc3dcd
(svn r4446) - Add function to determine if a Sprite ID exists.
 peter1138 parents: 
2548diff
changeset | 112 | /* Check if the given Sprite ID exists */ | 
| 
03d870cc3dcd
(svn r4446) - Add function to determine if a Sprite ID exists.
 peter1138 parents: 
2548diff
changeset | 113 | bool SpriteExists(SpriteID id) | 
| 
03d870cc3dcd
(svn r4446) - Add function to determine if a Sprite ID exists.
 peter1138 parents: 
2548diff
changeset | 114 | {
 | 
| 
03d870cc3dcd
(svn r4446) - Add function to determine if a Sprite ID exists.
 peter1138 parents: 
2548diff
changeset | 115 | /* Special case for Sprite ID zero -- its position is also 0... */ | 
| 5755 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 116 | if (id == 0) return true; | 
| 5953 
7000c5bdd70b
(svn r8166) -Fix (r7797): Protect against out of bounds access to the sprite ptr
 peter1138 parents: 
5860diff
changeset | 117 | if (id >= _spritecache_items) return false; | 
| 9703 
d2a6acdbd665
(svn r11146) [NoAI] -Sync: with trunk r11035:11045.
 rubidium parents: 
9631diff
changeset | 118 | return !(GetSpriteCache(id)->file_pos == 0 && GetSpriteCache(id)->file_slot == 0); | 
| 3565 
03d870cc3dcd
(svn r4446) - Add function to determine if a Sprite ID exists.
 peter1138 parents: 
2548diff
changeset | 119 | } | 
| 
03d870cc3dcd
(svn r4446) - Add function to determine if a Sprite ID exists.
 peter1138 parents: 
2548diff
changeset | 120 | |
| 9626 | 121 | void* AllocSprite(size_t); | 
| 2014 
0230ed9186bc
(svn r2522) Reorganize sprite load and decompression in order to remove a special case from the sprite blitter, which decompressed certain sprites every time when blitting them
 tron parents: 
1891diff
changeset | 122 | |
| 9627 | 123 | static void* ReadSprite(SpriteCache *sc, SpriteID id, bool real_sprite) | 
| 0 | 124 | {
 | 
| 9703 
d2a6acdbd665
(svn r11146) [NoAI] -Sync: with trunk r11035:11045.
 rubidium parents: 
9631diff
changeset | 125 | uint8 file_slot = sc->file_slot; | 
| 9626 | 126 | uint32 file_pos = sc->file_pos; | 
| 2014 
0230ed9186bc
(svn r2522) Reorganize sprite load and decompression in order to remove a special case from the sprite blitter, which decompressed certain sprites every time when blitting them
 tron parents: 
1891diff
changeset | 127 | |
| 5568 
75f13d7bfaed
(svn r7565) -Codechange: Rework DEBUG functionality. Look for appropiate debugging levels to
 Darkvater parents: 
5150diff
changeset | 128 | DEBUG(sprite, 9, "Load sprite %d", id); | 
| 184 
dbeaaaf8b2bb
(svn r185) -Fix: [1016954] Cached_sprites does now work again
 truelight parents: 
182diff
changeset | 129 | |
| 3565 
03d870cc3dcd
(svn r4446) - Add function to determine if a Sprite ID exists.
 peter1138 parents: 
2548diff
changeset | 130 | 	if (!SpriteExists(id)) {
 | 
| 6213 
164d84e1c5b4
(svn r8634) -Codechange: replace missing sprites with a red question mark.
 rubidium parents: 
5971diff
changeset | 131 | DEBUG(sprite, 1, "Tried to load non-existing sprite #%d. Probable cause: Wrong/missing NewGRFs", id); | 
| 
164d84e1c5b4
(svn r8634) -Codechange: replace missing sprites with a red question mark.
 rubidium parents: 
5971diff
changeset | 132 | |
| 
164d84e1c5b4
(svn r8634) -Codechange: replace missing sprites with a red question mark.
 rubidium parents: 
5971diff
changeset | 133 | /* SPR_IMG_QUERY is a BIG FAT RED ? */ | 
| 
164d84e1c5b4
(svn r8634) -Codechange: replace missing sprites with a red question mark.
 rubidium parents: 
5971diff
changeset | 134 | id = SPR_IMG_QUERY; | 
| 9703 
d2a6acdbd665
(svn r11146) [NoAI] -Sync: with trunk r11035:11045.
 rubidium parents: 
9631diff
changeset | 135 | file_slot = GetSpriteCache(SPR_IMG_QUERY)->file_slot; | 
| 
d2a6acdbd665
(svn r11146) [NoAI] -Sync: with trunk r11035:11045.
 rubidium parents: 
9631diff
changeset | 136 | file_pos = GetSpriteCache(SPR_IMG_QUERY)->file_pos; | 
| 1378 
ebb8d52f0352
(svn r1882) Add a basic check if a non-existent sprite gets accessed.
 tron parents: 
1363diff
changeset | 137 | } | 
| 
ebb8d52f0352
(svn r1882) Add a basic check if a non-existent sprite gets accessed.
 tron parents: 
1363diff
changeset | 138 | |
| 9628 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 139 | 	if (BlitterFactoryBase::GetCurrentBlitter()->GetScreenDepth() == 32) {
 | 
| 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 140 | #ifdef WITH_PNG | 
| 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 141 | /* Try loading 32bpp graphics in case we are 32bpp output */ | 
| 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 142 | SpriteLoaderPNG sprite_loader; | 
| 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 143 | SpriteLoader::Sprite sprite; | 
| 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 144 | |
| 9724 
b39bc69bb2f2
(svn r12051) [NoAI] -Sync: with trunk (r11795:12050).
 rubidium parents: 
9723diff
changeset | 145 | 		if (sprite_loader.LoadSprite(&sprite, file_slot, sc->id)) {
 | 
| 9628 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 146 | sc->ptr = BlitterFactoryBase::GetCurrentBlitter()->Encode(&sprite, &AllocSprite); | 
| 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 147 | free(sprite.data); | 
| 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 148 | |
| 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 149 | return sc->ptr; | 
| 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 150 | } | 
| 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 151 | /* If the PNG couldn't be loaded, fall back to 8bpp grfs */ | 
| 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 152 | #else | 
| 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 153 | static bool show_once = true; | 
| 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 154 | 		if (show_once) {
 | 
| 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 155 | DEBUG(misc, 0, "You are running a 32bpp blitter, but this build is without libpng support; falling back to 8bpp graphics"); | 
| 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 156 | show_once = false; | 
| 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 157 | } | 
| 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 158 | #endif /* WITH_PNG */ | 
| 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 159 | } | 
| 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 160 | |
| 9703 
d2a6acdbd665
(svn r11146) [NoAI] -Sync: with trunk r11035:11045.
 rubidium parents: 
9631diff
changeset | 161 | FioSeekToFile(file_slot, file_pos); | 
| 1354 
5e5c89b9b169
(svn r1858) Let ReadSprite() handle the subtleties of loading a sprite, not its caller
 tron parents: 
1353diff
changeset | 162 | |
| 9626 | 163 | /* Read the size and type */ | 
| 164 | int num = FioReadWord(); | |
| 165 | byte type = FioReadByte(); | |
| 166 | /* Type 0xFF indicates either a colormap or some other non-sprite info */ | |
| 2014 
0230ed9186bc
(svn r2522) Reorganize sprite load and decompression in order to remove a special case from the sprite blitter, which decompressed certain sprites every time when blitting them
 tron parents: 
1891diff
changeset | 167 | 	if (type == 0xFF) {
 | 
| 9627 | 168 | 		if (real_sprite) {
 | 
| 169 | static byte warning_level = 0; | |
| 170 | DEBUG(sprite, warning_level, "Tried to load non sprite #%d as a real sprite. Probable cause: NewGRF interference", id); | |
| 171 | warning_level = 6; | |
| 172 | 			if (id == SPR_IMG_QUERY) error("Uhm, would you be so kind not to load a NewGRF that makes the 'query' sprite a non- sprite?");
 | |
| 173 | return (void*)GetSprite(SPR_IMG_QUERY); | |
| 174 | } | |
| 175 | ||
| 9626 | 176 | byte *dest = (byte *)AllocSprite(num); | 
| 2014 
0230ed9186bc
(svn r2522) Reorganize sprite load and decompression in order to remove a special case from the sprite blitter, which decompressed certain sprites every time when blitting them
 tron parents: 
1891diff
changeset | 177 | |
| 5755 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 178 | sc->ptr = dest; | 
| 2014 
0230ed9186bc
(svn r2522) Reorganize sprite load and decompression in order to remove a special case from the sprite blitter, which decompressed certain sprites every time when blitting them
 tron parents: 
1891diff
changeset | 179 | FioReadBlock(dest, num); | 
| 
0230ed9186bc
(svn r2522) Reorganize sprite load and decompression in order to remove a special case from the sprite blitter, which decompressed certain sprites every time when blitting them
 tron parents: 
1891diff
changeset | 180 | |
| 9627 | 181 | return sc->ptr; | 
| 182 | } | |
| 183 | /* Ugly hack to work around the problem that the old landscape | |
| 184 | * generator assumes that those sprites are stored uncompressed in | |
| 185 | * the memory, and they are only read directly by the code, never | |
| 186 | * send to the blitter. So do not send it to the blitter (which will | |
| 187 | * result in a data array in the format the blitter likes most), but | |
| 188 | * read the data directly from disk and store that as sprite. | |
| 189 | * Ugly: yes. Other solution: no. Blame the original author or | |
| 190 | * something ;) The image should really have been a data-stream | |
| 191 | * (so type = 0xFF basicly). */ | |
| 192 | 	if (id >= 4845 && id <= 4881) {
 | |
| 193 | uint height = FioReadByte(); | |
| 194 | uint width = FioReadWord(); | |
| 195 | Sprite *sprite; | |
| 196 | byte *dest; | |
| 197 | ||
| 198 | num = width * height; | |
| 199 | sprite = (Sprite *)AllocSprite(sizeof(*sprite) + num); | |
| 200 | sc->ptr = sprite; | |
| 201 | sprite->height = height; | |
| 202 | sprite->width = width; | |
| 203 | sprite->x_offs = FioReadWord(); | |
| 204 | sprite->y_offs = FioReadWord(); | |
| 205 | ||
| 206 | dest = sprite->data; | |
| 207 | 		while (num > 0) {
 | |
| 208 | int8 i = FioReadByte(); | |
| 209 | 			if (i >= 0) {
 | |
| 210 | num -= i; | |
| 211 | for (; i > 0; --i) *dest++ = FioReadByte(); | |
| 212 | 			} else {
 | |
| 213 | const byte* rel = dest - (((i & 7) << 8) | FioReadByte()); | |
| 214 | i = -(i >> 3); | |
| 215 | num -= i; | |
| 216 | for (; i > 0; --i) *dest++ = *rel++; | |
| 217 | } | |
| 218 | } | |
| 219 | ||
| 220 | return sc->ptr; | |
| 221 | } | |
| 222 | ||
| 223 | 	if (!real_sprite) {
 | |
| 224 | static byte warning_level = 0; | |
| 225 | DEBUG(sprite, warning_level, "Tried to load real sprite #%d as a non sprite. Probable cause: NewGRF interference", id); | |
| 226 | warning_level = 6; | |
| 9626 | 227 | } | 
| 2014 
0230ed9186bc
(svn r2522) Reorganize sprite load and decompression in order to remove a special case from the sprite blitter, which decompressed certain sprites every time when blitting them
 tron parents: 
1891diff
changeset | 228 | |
| 9626 | 229 | SpriteLoaderGrf sprite_loader; | 
| 230 | SpriteLoader::Sprite sprite; | |
| 0 | 231 | |
| 9724 
b39bc69bb2f2
(svn r12051) [NoAI] -Sync: with trunk (r11795:12050).
 rubidium parents: 
9723diff
changeset | 232 | if (!sprite_loader.LoadSprite(&sprite, file_slot, file_pos)) return NULL; | 
| 9626 | 233 | if (id == 142) sprite.height = 10; // Compensate for a TTD bug | 
| 234 | sc->ptr = BlitterFactoryBase::GetCurrentBlitter()->Encode(&sprite, &AllocSprite); | |
| 235 | free(sprite.data); | |
| 184 
dbeaaaf8b2bb
(svn r185) -Fix: [1016954] Cached_sprites does now work again
 truelight parents: 
182diff
changeset | 236 | |
| 9626 | 237 | return sc->ptr; | 
| 0 | 238 | } | 
| 239 | ||
| 240 | ||
| 9703 
d2a6acdbd665
(svn r11146) [NoAI] -Sync: with trunk r11035:11045.
 rubidium parents: 
9631diff
changeset | 241 | bool LoadNextSprite(int load_index, byte file_slot, uint file_sprite_id) | 
| 0 | 242 | {
 | 
| 5755 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 243 | SpriteCache *sc; | 
| 9703 
d2a6acdbd665
(svn r11146) [NoAI] -Sync: with trunk r11035:11045.
 rubidium parents: 
9631diff
changeset | 244 | uint32 file_pos = FioGetPos(); | 
| 0 | 245 | |
| 2342 
c19fb4f2df30
(svn r2868) Change the way NewGRFs are loaded: The loading process i no longer bolted onto the normal graphics loading.
 tron parents: 
2340diff
changeset | 246 | if (!ReadSpriteHeaderSkipData()) return false; | 
| 361 
ad7a042ee0eb
(svn r549) -newgrf: Support for action 0xd (change a parameter (sorta variable for the GRF scripts)). Based on patch by octo, heavy changes by pasky.
 darkvater parents: 
184diff
changeset | 247 | |
| 3591 
05a4b88d875f
(svn r4481) - Fix: Validate the given sprite ID when loading a sprite.
 peter1138 parents: 
3565diff
changeset | 248 | 	if (load_index >= MAX_SPRITES) {
 | 
| 
05a4b88d875f
(svn r4481) - Fix: Validate the given sprite ID when loading a sprite.
 peter1138 parents: 
3565diff
changeset | 249 | 		error("Tried to load too many sprites (#%d; max %d)", load_index, MAX_SPRITES);
 | 
| 
05a4b88d875f
(svn r4481) - Fix: Validate the given sprite ID when loading a sprite.
 peter1138 parents: 
3565diff
changeset | 250 | } | 
| 
05a4b88d875f
(svn r4481) - Fix: Validate the given sprite ID when loading a sprite.
 peter1138 parents: 
3565diff
changeset | 251 | |
| 5755 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 252 | sc = AllocateSpriteCache(load_index); | 
| 9703 
d2a6acdbd665
(svn r11146) [NoAI] -Sync: with trunk r11035:11045.
 rubidium parents: 
9631diff
changeset | 253 | sc->file_slot = file_slot; | 
| 5755 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 254 | sc->file_pos = file_pos; | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 255 | sc->ptr = NULL; | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 256 | sc->lru = 0; | 
| 9628 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 257 | sc->id = file_sprite_id; | 
| 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 258 | |
| 0 | 259 | return true; | 
| 260 | } | |
| 261 | ||
| 2407 | 262 | |
| 5838 
9c3129cb019b
(svn r8038) -Merge: the cpp branch. Effort of KUDr, Celestar, glx, Smoovius, stillunknown and pv2b.
 rubidium parents: 
5835diff
changeset | 263 | void DupSprite(SpriteID old_spr, SpriteID new_spr) | 
| 2407 | 264 | {
 | 
| 9724 
b39bc69bb2f2
(svn r12051) [NoAI] -Sync: with trunk (r11795:12050).
 rubidium parents: 
9723diff
changeset | 265 | SpriteCache *scnew = AllocateSpriteCache(new_spr); // may reallocate: so put it first | 
| 5838 
9c3129cb019b
(svn r8038) -Merge: the cpp branch. Effort of KUDr, Celestar, glx, Smoovius, stillunknown and pv2b.
 rubidium parents: 
5835diff
changeset | 266 | SpriteCache *scold = GetSpriteCache(old_spr); | 
| 5755 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 267 | |
| 9703 
d2a6acdbd665
(svn r11146) [NoAI] -Sync: with trunk r11035:11045.
 rubidium parents: 
9631diff
changeset | 268 | scnew->file_slot = scold->file_slot; | 
| 5755 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 269 | scnew->file_pos = scold->file_pos; | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 270 | scnew->ptr = NULL; | 
| 9628 
b5c2449616b5
(svn r10195) [NoAI] -Sync: with trunk r10119:10194.
 rubidium parents: 
9627diff
changeset | 271 | scnew->id = scold->id; | 
| 37 
61bf1df68d82
(svn r38) Preliminary slopes graphics fix. Neighboring tile check not done yet
 dominik parents: 
0diff
changeset | 272 | } | 
| 
61bf1df68d82
(svn r38) Preliminary slopes graphics fix. Neighboring tile check not done yet
 dominik parents: 
0diff
changeset | 273 | |
| 0 | 274 | |
| 275 | #define S_FREE_MASK 1 | |
| 1353 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 276 | |
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 277 | static inline MemBlock* NextBlock(MemBlock* block) | 
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 278 | {
 | 
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 279 | return (MemBlock*)((byte*)block + (block->size & ~S_FREE_MASK)); | 
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 280 | } | 
| 0 | 281 | |
| 6573 | 282 | static uint32 GetSpriteCacheUsage() | 
| 0 | 283 | {
 | 
| 4321 
b763b7007162
(svn r5974) -Codechange: added casts all around the place to make Windows 64bit happy (michi_cc)
 truelight parents: 
3591diff
changeset | 284 | uint32 tot_size = 0; | 
| 1353 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 285 | MemBlock* s; | 
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 286 | |
| 9732 | 287 | 	for (s = _spritecache_ptr; s->size != 0; s = NextBlock(s)) {
 | 
| 1353 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 288 | if (!(s->size & S_FREE_MASK)) tot_size += s->size; | 
| 9732 | 289 | } | 
| 0 | 290 | |
| 291 | return tot_size; | |
| 292 | } | |
| 293 | ||
| 294 | ||
| 6573 | 295 | void IncreaseSpriteLRU() | 
| 0 | 296 | {
 | 
| 9574 | 297 | /* Increase all LRU values */ | 
| 0 | 298 | 	if (_sprite_lru_counter > 16384) {
 | 
| 5755 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 299 | SpriteID i; | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 300 | |
| 5568 
75f13d7bfaed
(svn r7565) -Codechange: Rework DEBUG functionality. Look for appropiate debugging levels to
 Darkvater parents: 
5150diff
changeset | 301 | DEBUG(sprite, 3, "Fixing lru %d, inuse=%d", _sprite_lru_counter, GetSpriteCacheUsage()); | 
| 0 | 302 | |
| 5755 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 303 | 		for (i = 0; i != _spritecache_items; i++) {
 | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 304 | SpriteCache *sc = GetSpriteCache(i); | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 305 | 			if (sc->ptr != NULL) {
 | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 306 | 				if (sc->lru >= 0) {
 | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 307 | sc->lru = -1; | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 308 | 				} else if (sc->lru != -32768) {
 | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 309 | sc->lru--; | 
| 0 | 310 | } | 
| 311 | } | |
| 5755 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 312 | } | 
| 0 | 313 | _sprite_lru_counter = 0; | 
| 314 | } | |
| 315 | ||
| 9574 | 316 | /* Compact sprite cache every now and then. */ | 
| 0 | 317 | 	if (++_compact_cache_counter >= 740) {
 | 
| 318 | CompactSpriteCache(); | |
| 319 | _compact_cache_counter = 0; | |
| 320 | } | |
| 321 | } | |
| 322 | ||
| 9574 | 323 | /** Called when holes in the sprite cache should be removed. | 
| 324 | * That is accomplished by moving the cached data. */ | |
| 6573 | 325 | static void CompactSpriteCache() | 
| 0 | 326 | {
 | 
| 1353 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 327 | MemBlock *s; | 
| 184 
dbeaaaf8b2bb
(svn r185) -Fix: [1016954] Cached_sprites does now work again
 truelight parents: 
182diff
changeset | 328 | |
| 5568 
75f13d7bfaed
(svn r7565) -Codechange: Rework DEBUG functionality. Look for appropiate debugging levels to
 Darkvater parents: 
5150diff
changeset | 329 | DEBUG(sprite, 3, "Compacting sprite cache, inuse=%d", GetSpriteCacheUsage()); | 
| 184 
dbeaaaf8b2bb
(svn r185) -Fix: [1016954] Cached_sprites does now work again
 truelight parents: 
182diff
changeset | 330 | |
| 1353 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 331 | 	for (s = _spritecache_ptr; s->size != 0;) {
 | 
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 332 | 		if (s->size & S_FREE_MASK) {
 | 
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 333 | MemBlock* next = NextBlock(s); | 
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 334 | MemBlock temp; | 
| 5755 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 335 | SpriteID i; | 
| 1353 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 336 | |
| 9574 | 337 | /* Since free blocks are automatically coalesced, this should hold true. */ | 
| 1353 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 338 | assert(!(next->size & S_FREE_MASK)); | 
| 184 
dbeaaaf8b2bb
(svn r185) -Fix: [1016954] Cached_sprites does now work again
 truelight parents: 
182diff
changeset | 339 | |
| 9574 | 340 | /* If the next block is the sentinel block, we can safely return */ | 
| 9826 
9707ad4c9b60
(svn r12462) [NoAI] -Sync: with trunk r12304:12461.
 rubidium parents: 
9732diff
changeset | 341 | if (next->size == 0) break; | 
| 0 | 342 | |
| 9574 | 343 | /* Locate the sprite belonging to the next pointer. */ | 
| 5755 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 344 | 			for (i = 0; GetSpriteCache(i)->ptr != next->data; i++) {
 | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 345 | assert(i != _spritecache_items); | 
| 1353 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 346 | } | 
| 0 | 347 | |
| 5755 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 348 | GetSpriteCache(i)->ptr = s->data; // Adjust sprite array entry | 
| 9574 | 349 | /* Swap this and the next block */ | 
| 1353 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 350 | temp = *s; | 
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 351 | memmove(s, next, next->size); | 
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 352 | s = NextBlock(s); | 
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 353 | *s = temp; | 
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 354 | |
| 9574 | 355 | /* Coalesce free blocks */ | 
| 1353 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 356 | 			while (NextBlock(s)->size & S_FREE_MASK) {
 | 
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 357 | s->size += NextBlock(s)->size & ~S_FREE_MASK; | 
| 0 | 358 | } | 
| 1353 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 359 | 		} else {
 | 
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 360 | s = NextBlock(s); | 
| 0 | 361 | } | 
| 362 | } | |
| 363 | } | |
| 364 | ||
| 6573 | 365 | static void DeleteEntryFromSpriteCache() | 
| 0 | 366 | {
 | 
| 5755 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 367 | SpriteID i; | 
| 5838 
9c3129cb019b
(svn r8038) -Merge: the cpp branch. Effort of KUDr, Celestar, glx, Smoovius, stillunknown and pv2b.
 rubidium parents: 
5835diff
changeset | 368 | uint best = UINT_MAX; | 
| 1353 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 369 | MemBlock* s; | 
| 0 | 370 | int cur_lru; | 
| 371 | ||
| 5568 
75f13d7bfaed
(svn r7565) -Codechange: Rework DEBUG functionality. Look for appropiate debugging levels to
 Darkvater parents: 
5150diff
changeset | 372 | DEBUG(sprite, 3, "DeleteEntryFromSpriteCache, inuse=%d", GetSpriteCacheUsage()); | 
| 0 | 373 | |
| 374 | cur_lru = 0xffff; | |
| 5755 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 375 | 	for (i = 0; i != _spritecache_items; i++) {
 | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 376 | SpriteCache *sc = GetSpriteCache(i); | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 377 | 		if (sc->ptr != NULL && sc->lru < cur_lru) {
 | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 378 | cur_lru = sc->lru; | 
| 0 | 379 | best = i; | 
| 380 | } | |
| 381 | } | |
| 382 | ||
| 9574 | 383 | /* Display an error message and die, in case we found no sprite at all. | 
| 384 | * This shouldn't really happen, unless all sprites are locked. */ | |
| 9723 
eee46cb39750
(svn r11796) [NoAI] -Sync: with trunk r11502:11795.
 rubidium parents: 
9722diff
changeset | 385 | 	if (best == (uint)-1) error("Out of sprite memory");
 | 
| 0 | 386 | |
| 9574 | 387 | /* Mark the block as free (the block must be in use) */ | 
| 5755 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 388 | s = (MemBlock*)GetSpriteCache(best)->ptr - 1; | 
| 1353 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 389 | assert(!(s->size & S_FREE_MASK)); | 
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 390 | s->size |= S_FREE_MASK; | 
| 5755 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 391 | GetSpriteCache(best)->ptr = NULL; | 
| 0 | 392 | |
| 9574 | 393 | /* And coalesce adjacent free blocks */ | 
| 1353 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 394 | 	for (s = _spritecache_ptr; s->size != 0; s = NextBlock(s)) {
 | 
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 395 | 		if (s->size & S_FREE_MASK) {
 | 
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 396 | 			while (NextBlock(s)->size & S_FREE_MASK) {
 | 
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 397 | s->size += NextBlock(s)->size & ~S_FREE_MASK; | 
| 0 | 398 | } | 
| 399 | } | |
| 400 | } | |
| 401 | } | |
| 402 | ||
| 9626 | 403 | void* AllocSprite(size_t mem_req) | 
| 0 | 404 | {
 | 
| 2014 
0230ed9186bc
(svn r2522) Reorganize sprite load and decompression in order to remove a special case from the sprite blitter, which decompressed certain sprites every time when blitting them
 tron parents: 
1891diff
changeset | 405 | mem_req += sizeof(MemBlock); | 
| 0 | 406 | |
| 1353 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 407 | /* Align this to an uint32 boundary. This also makes sure that the 2 least | 
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 408 | * bits are not used, so we could use those for other things. */ | 
| 9722 
ebf0ece7d8f6
(svn r11503) [NoAI] -Sync: with trunk r11308:11502.
 rubidium parents: 
9703diff
changeset | 409 | mem_req = Align(mem_req, sizeof(uint32)); | 
| 0 | 410 | |
| 1353 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 411 | 	for (;;) {
 | 
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 412 | MemBlock* s; | 
| 0 | 413 | |
| 1353 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 414 | 		for (s = _spritecache_ptr; s->size != 0; s = NextBlock(s)) {
 | 
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 415 | 			if (s->size & S_FREE_MASK) {
 | 
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 416 | size_t cur_size = s->size & ~S_FREE_MASK; | 
| 0 | 417 | |
| 1353 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 418 | /* Is the block exactly the size we need or | 
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 419 | * big enough for an additional free block? */ | 
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 420 | if (cur_size == mem_req || | 
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 421 | 						cur_size >= mem_req + sizeof(MemBlock)) {
 | 
| 9574 | 422 | /* Set size and in use */ | 
| 1353 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 423 | s->size = mem_req; | 
| 184 
dbeaaaf8b2bb
(svn r185) -Fix: [1016954] Cached_sprites does now work again
 truelight parents: 
182diff
changeset | 424 | |
| 9574 | 425 | /* Do we need to inject a free block too? */ | 
| 1353 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 426 | 					if (cur_size != mem_req) {
 | 
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 427 | NextBlock(s)->size = (cur_size - mem_req) | S_FREE_MASK; | 
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 428 | } | 
| 0 | 429 | |
| 1353 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 430 | return s->data; | 
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 431 | } | 
| 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 432 | } | 
| 0 | 433 | } | 
| 434 | ||
| 9574 | 435 | /* Reached sentinel, but no block found yet. Delete some old entry. */ | 
| 1353 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 436 | DeleteEntryFromSpriteCache(); | 
| 0 | 437 | } | 
| 438 | } | |
| 439 | ||
| 440 | ||
| 9627 | 441 | const void *GetRawSprite(SpriteID sprite, bool real_sprite) | 
| 0 | 442 | {
 | 
| 5755 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 443 | SpriteCache *sc; | 
| 2014 
0230ed9186bc
(svn r2522) Reorganize sprite load and decompression in order to remove a special case from the sprite blitter, which decompressed certain sprites every time when blitting them
 tron parents: 
1891diff
changeset | 444 | void* p; | 
| 0 | 445 | |
| 5953 
7000c5bdd70b
(svn r8166) -Fix (r7797): Protect against out of bounds access to the sprite ptr
 peter1138 parents: 
5860diff
changeset | 446 | assert(sprite < _spritecache_items); | 
| 0 | 447 | |
| 5755 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 448 | sc = GetSpriteCache(sprite); | 
| 0 | 449 | |
| 9574 | 450 | /* Update LRU */ | 
| 5755 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 451 | sc->lru = ++_sprite_lru_counter; | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 452 | |
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 453 | p = sc->ptr; | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 454 | |
| 9574 | 455 | /* Load the sprite, if it is not loaded, yet */ | 
| 9627 | 456 | if (p == NULL) p = ReadSprite(sc, sprite, real_sprite); | 
| 9732 | 457 | |
| 184 
dbeaaaf8b2bb
(svn r185) -Fix: [1016954] Cached_sprites does now work again
 truelight parents: 
182diff
changeset | 458 | return p; | 
| 0 | 459 | } | 
| 460 | ||
| 961 | 461 | |
| 6573 | 462 | void GfxInitSpriteMem() | 
| 0 | 463 | {
 | 
| 9574 | 464 | /* initialize sprite cache heap */ | 
| 9723 
eee46cb39750
(svn r11796) [NoAI] -Sync: with trunk r11502:11795.
 rubidium parents: 
9722diff
changeset | 465 | if (_spritecache_ptr == NULL) _spritecache_ptr = (MemBlock*)MallocT<byte>(_sprite_cache_size * 1024 * 1024); | 
| 0 | 466 | |
| 9574 | 467 | /* A big free block */ | 
| 9626 | 468 | _spritecache_ptr->size = ((_sprite_cache_size * 1024 * 1024) - sizeof(MemBlock)) | S_FREE_MASK; | 
| 9574 | 469 | /* Sentinel block (identified by size == 0) */ | 
| 1353 
48b59d472641
(svn r1857) Rewrite parts of the sprite heap. It's functionally equivalent but should be easier to read and maintain.
 tron parents: 
1352diff
changeset | 470 | NextBlock(_spritecache_ptr)->size = 0; | 
| 0 | 471 | |
| 5755 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 472 | /* Reset the spritecache 'pool' */ | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 473 | free(_spritecache); | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 474 | _spritecache_items = 0; | 
| 
5db1cb757991
(svn r7797) -Codechange: Replace static _sprite_ptr and associated arrays with dynamic array to allow variable number of sprites. This does not change the sprite limit.
 peter1138 parents: 
5752diff
changeset | 475 | _spritecache = NULL; | 
| 0 | 476 | |
| 2340 
0a9f3eeccb96
(svn r2866) Move all functions and tables which aren't directly involved in managing the sprite heap to a new file gfxinit.c.
 tron parents: 
2339diff
changeset | 477 | _compact_cache_counter = 0; | 
| 0 | 478 | } |