10 #include "../stdafx.h"
11 #include "../gfx_func.h"
12 #include "../fileio_func.h"
14 #include "../strings_func.h"
15 #include "table/strings.h"
17 #include "../core/math_func.hpp"
18 #include "../core/alloc_type.hpp"
19 #include "../core/bitmath_func.hpp"
22 #include "../safeguards.h"
45 static byte warning_level = 0;
46 if (warning_level == 0) {
50 DEBUG(sprite, warning_level,
"[%i] Loading corrupted sprite from %s at position %i", line,
FioGetFilename(file_slot), (
int)file_pos);
70 std::unique_ptr<byte[]> dest_orig(
new byte[num]);
71 byte *dest = dest_orig.get();
72 const int64 dest_size = num;
80 int size = (code == 0) ? 0x80 : code;
83 for (; size > 0; size--) {
89 const uint data_offset = ((code & 7) << 8) |
FioReadByte();
90 if (dest - data_offset < dest_orig.get())
return WarnCorruptSprite(file_slot, file_pos, __LINE__);
91 int size = -(code >> 3);
94 for (; size > 0; size--) {
95 *dest = *(dest - data_offset);
107 if (colour_fmt &
SCC_RGB) bpp += 3;
109 if (colour_fmt &
SCC_PAL) bpp++;
113 for (
int y = 0; y < sprite->
height; y++) {
114 bool last_item =
false;
117 if (container_format >= 2 && dest_size > UINT16_MAX) {
118 offset = (dest_orig[y * 4 + 3] << 24) | (dest_orig[y * 4 + 2] << 16) | (dest_orig[y * 4 + 1] << 8) | dest_orig[y * 4];
120 offset = (dest_orig[y * 2 + 1] << 8) | dest_orig[y * 2];
124 dest = dest_orig.get() + offset;
127 if (dest + (container_format >= 2 && sprite->
width > 256 ? 4 : 2) > dest_orig.get() + dest_size) {
134 if (container_format >= 2 && sprite->
width > 256) {
138 last_item = (dest[1] & 0x80) != 0;
139 length = ((dest[1] & 0x7F) << 8) | dest[0];
140 skip = (dest[3] << 8) | dest[2];
146 last_item = ((*dest) & 0x80) != 0;
147 length = (*dest++) & 0x7F;
151 data = &sprite->
data[y * sprite->
width + skip];
153 if (skip + length > sprite->
width || dest + length * bpp > dest_orig.get() + dest_size) {
157 for (
int x = 0; x < length; x++) {
163 data->
a = (colour_fmt &
SCC_ALPHA) ? *dest++ : 0xFF;
165 switch (sprite_type) {
167 case ST_FONT: data->
m = std::min<uint>(*dest, 2u);
break;
168 default: data->
m = *dest;
break;
171 if (colour_fmt ==
SCC_PAL && *dest == 0) data->
a = 0x00;
176 }
while (!last_item);
179 if (dest_size < sprite->
width * sprite->
height * bpp) {
183 if (dest_size > sprite->
width * sprite->
height * bpp) {
184 static byte warning_level = 0;
185 DEBUG(sprite, warning_level,
"Ignoring " OTTD_PRINTF64
" unused extra bytes from the sprite from %s at position %i", dest_size - sprite->
width * sprite->
height * bpp,
FioGetFilename(file_slot), (
int)file_pos);
189 dest = dest_orig.get();
191 for (
int i = 0; i < sprite->
width * sprite->
height; i++) {
192 byte *pixel = &dest[i * bpp];
195 sprite->
data[i].
r = *pixel++;
196 sprite->
data[i].
g = *pixel++;
197 sprite->
data[i].
b = *pixel++;
201 switch (sprite_type) {
203 case ST_FONT: sprite->
data[i].
m = std::min<uint>(*pixel, 2u);
break;
204 default: sprite->
data[i].
m = *pixel;
break;
207 if (colour_fmt ==
SCC_PAL && *pixel == 0) sprite->
data[i].
a = 0x00;
219 if (load_32bpp)
return 0;
229 if (type == 0xFF)
return 0;
238 if (sprite[zoom_lvl].
width > INT16_MAX) {
245 num = (type & 0x02) ? sprite[zoom_lvl].
width * sprite[zoom_lvl].
height : num - 8;
247 if (
DecodeSingleSprite(&sprite[zoom_lvl], file_slot, file_pos, sprite_type, num, type, zoom_lvl,
SCC_PAL, 1))
return 1 << zoom_lvl;
257 if (file_pos == SIZE_MAX)
return 0;
264 uint8 loaded_sprites = 0;
271 if (type == 0xFF)
return 0;
279 if (
HasBit(loaded_sprites, zoom_lvl)) {
281 DEBUG(sprite, 1,
"Ignoring duplicate zoom level sprite %u from %s",
id,
FioGetFilename(file_slot));
291 if (sprite[zoom_lvl].
width > INT16_MAX || sprite[zoom_lvl].
height > INT16_MAX) {
301 if (colour &
SCC_RGB) bpp += 3;
309 bool valid =
DecodeSingleSprite(&sprite[zoom_lvl], file_slot, file_pos, sprite_type, decomp_size, type, zoom_lvl, colour, 2);
323 return loaded_sprites;
328 if (this->container_ver >= 2) {
329 return LoadSpriteV2(sprite, file_slot, file_pos, sprite_type, load_32bpp);
331 return LoadSpriteV1(sprite, file_slot, file_pos, sprite_type, load_32bpp);