/* SDL_image: An example image loading library for use with SDL Copyright (C) 1997-2019 Sam Lantinga This software is provided 'as-is', without any express or implied warranty. In no event will the authors be held liable for any damages arising from the use of this software. Permission is granted to anyone to use this software for any purpose, including commercial applications, and to alter it and redistribute it freely, subject to the following restrictions: 1. The origin of this software must not be misrepresented; you must not claim that you wrote the original software. If you use this software in a product, an acknowledgment in the product documentation would be appreciated but is not required. 2. Altered source versions must be plainly marked as such, and must not be misrepresented as being the original software. 3. This notice may not be removed or altered from any source distribution. */ /* This is a GIF image file loading framework */ #include "SDL_image.h" #ifdef LOAD_GIF /* Code from here to end of file has been adapted from XPaint: */ /* +-------------------------------------------------------------------+ */ /* | Copyright 1990, 1991, 1993 David Koblas. | */ /* | Copyright 1996 Torsten Martinsen. | */ /* | Permission to use, copy, modify, and distribute this software | */ /* | and its documentation for any purpose and without fee is hereby | */ /* | granted, provided that the above copyright notice appear in all | */ /* | copies and that both that copyright notice and this permission | */ /* | notice appear in supporting documentation. This software is | */ /* | provided "as is" without express or implied warranty. | */ /* +-------------------------------------------------------------------+ */ /* Adapted for use in SDL by Sam Lantinga -- 7/20/98 */ /* Changes to work with SDL: Include SDL header file Use SDL_Surface rather than xpaint Image structure Define SDL versions of RWSetMsg(), ImageNewCmap() and ImageSetCmap() */ #include "SDL.h" #define Image SDL_Surface #define RWSetMsg IMG_SetError #define ImageNewCmap(w, h, s) SDL_CreateRGBSurface(SDL_SWSURFACE,w,h,8,0,0,0,0) #define ImageSetCmap(s, i, R, G, B) do { \ s->format->palette->colors[i].r = R; \ s->format->palette->colors[i].g = G; \ s->format->palette->colors[i].b = B; \ } while (0) /* * * * * */ #define GIF_DISPOSE_NA 0 /* No disposal specified */ #define GIF_DISPOSE_NONE 1 /* Do not dispose */ #define GIF_DISPOSE_RESTORE_BACKGROUND 2 /* Restore to background */ #define GIF_DISPOSE_RESTORE_PREVIOUS 3 /* Restore to previous */ #define MAXCOLORMAPSIZE 256 #define TRUE 1 #define FALSE 0 #define CM_RED 0 #define CM_GREEN 1 #define CM_BLUE 2 #define MAX_LWZ_BITS 12 #define INTERLACE 0x40 #define LOCALCOLORMAP 0x80 #define BitSet(byte, bit) (((byte) & (bit)) == (bit)) #define ReadOK(file,buffer,len) SDL_RWread(file, buffer, len, 1) #define LM_to_uint(a,b) (((b)<<8)|(a)) typedef struct { struct { unsigned int Width; unsigned int Height; unsigned char ColorMap[3][MAXCOLORMAPSIZE]; unsigned int BitPixel; unsigned int ColorResolution; unsigned int Background; unsigned int AspectRatio; int GrayScale; } GifScreen; struct { int transparent; int delayTime; int inputFlag; int disposal; } Gif89; unsigned char buf[280]; int curbit, lastbit, done, last_byte; int fresh; int code_size, set_code_size; int max_code, max_code_size; int firstcode, oldcode; int clear_code, end_code; int table[2][(1 << MAX_LWZ_BITS)]; int stack[(1 << (MAX_LWZ_BITS)) * 2], *sp; int ZeroDataBlock; } State_t; typedef struct { Image *image; int x, y; int disposal; int delay; } Frame_t; typedef struct { int count; Frame_t *frames; } Anim_t; static int ReadColorMap(SDL_RWops * src, int number, unsigned char buffer[3][MAXCOLORMAPSIZE], int *flag); static int DoExtension(SDL_RWops * src, int label, State_t * state); static int GetDataBlock(SDL_RWops * src, unsigned char *buf, State_t * state); static int GetCode(SDL_RWops * src, int code_size, int flag, State_t * state); static int LWZReadByte(SDL_RWops * src, int flag, int input_code_size, State_t * state); static Image *ReadImage(SDL_RWops * src, int len, int height, int, unsigned char cmap[3][MAXCOLORMAPSIZE], int gray, int interlace, int ignore, State_t * state); static SDL_bool NormalizeFrames(Frame_t *frames, int count) { SDL_Surface *image; int i; int lastDispose = GIF_DISPOSE_RESTORE_BACKGROUND; int iRestore = 0; Uint32 fill; SDL_Rect rect; if (SDL_HasColorKey(frames[0].image)) { image = SDL_ConvertSurfaceFormat(frames[0].image, SDL_PIXELFORMAT_ARGB8888, 0); } else { image = SDL_ConvertSurfaceFormat(frames[0].image, SDL_PIXELFORMAT_RGB888, 0); } if (!image) { return SDL_FALSE; } fill = SDL_MapRGBA(image->format, 0, 0, 0, SDL_ALPHA_TRANSPARENT); rect.x = 0; rect.y = 0; rect.w = image->w; rect.h = image->h; for (i = 0; i < count; ++i) { switch (lastDispose) { case GIF_DISPOSE_RESTORE_BACKGROUND: SDL_FillRect(image, &rect, fill); break; case GIF_DISPOSE_RESTORE_PREVIOUS: SDL_BlitSurface(frames[iRestore].image, &rect, image, &rect); break; default: break; } if (frames[i].disposal != GIF_DISPOSE_RESTORE_PREVIOUS) { iRestore = i; } rect.x = (Sint16)frames[i].x; rect.y = (Sint16)frames[i].y; rect.w = frames[i].image->w; rect.h = frames[i].image->h; SDL_BlitSurface(frames[i].image, NULL, image, &rect); SDL_FreeSurface(frames[i].image); frames[i].image = SDL_DuplicateSurface(image); if (!frames[i].image) { return SDL_FALSE; } lastDispose = frames[i].disposal; } SDL_FreeSurface( image ); return SDL_TRUE; } Anim_t * IMG_LoadGIF_RW_Internal(SDL_RWops *src, SDL_bool load_anim) { Sint64 start; unsigned char buf[16]; unsigned char c; unsigned char localColorMap[3][MAXCOLORMAPSIZE]; int grayScale; int useGlobalColormap; int bitPixel; char version[4]; Image *image = NULL; Anim_t *anim; Frame_t *frames, *frame; State_t state; state.ZeroDataBlock = FALSE; state.fresh = FALSE; state.last_byte = 0; if (src == NULL) { return NULL; } start = SDL_RWtell(src); anim = (Anim_t *)SDL_calloc(1, sizeof(*anim)); if (!anim) { SDL_OutOfMemory(); return NULL; } if (!ReadOK(src, buf, 6)) { RWSetMsg("error reading magic number"); goto done; } if (SDL_strncmp((char *) buf, "GIF", 3) != 0) { RWSetMsg("not a GIF file"); goto done; } SDL_memcpy(version, (char *) buf + 3, 3); version[3] = '\0'; if ((SDL_strcmp(version, "87a") != 0) && (SDL_strcmp(version, "89a") != 0)) { RWSetMsg("bad version number, not '87a' or '89a'"); goto done; } state.Gif89.transparent = -1; state.Gif89.delayTime = -1; state.Gif89.inputFlag = -1; state.Gif89.disposal = GIF_DISPOSE_NA; if (!ReadOK(src, buf, 7)) { RWSetMsg("failed to read screen descriptor"); goto done; } state.GifScreen.Width = LM_to_uint(buf[0], buf[1]); state.GifScreen.Height = LM_to_uint(buf[2], buf[3]); state.GifScreen.BitPixel = 2 << (buf[4] & 0x07); state.GifScreen.ColorResolution = (((buf[4] & 0x70) >> 3) + 1); state.GifScreen.Background = buf[5]; state.GifScreen.AspectRatio = buf[6]; if (BitSet(buf[4], LOCALCOLORMAP)) { /* Global Colormap */ if (ReadColorMap(src, state.GifScreen.BitPixel, state.GifScreen.ColorMap, &state.GifScreen.GrayScale)) { RWSetMsg("error reading global colormap"); goto done; } } for ( ; ; ) { if (!ReadOK(src, &c, 1)) { RWSetMsg("EOF / read error on image data"); goto done; } if (c == ';') { /* GIF terminator */ goto done; } if (c == '!') { /* Extension */ if (!ReadOK(src, &c, 1)) { RWSetMsg("EOF / read error on extention function code"); goto done; } DoExtension(src, c, &state); continue; } if (c != ',') { /* Not a valid start character */ continue; } if (!ReadOK(src, buf, 9)) { RWSetMsg("couldn't read left/top/width/height"); goto done; } useGlobalColormap = !BitSet(buf[8], LOCALCOLORMAP); bitPixel = 1 << ((buf[8] & 0x07) + 1); if (!useGlobalColormap) { if (ReadColorMap(src, bitPixel, localColorMap, &grayScale)) { RWSetMsg("error reading local colormap"); goto done; } image = ReadImage(src, LM_to_uint(buf[4], buf[5]), LM_to_uint(buf[6], buf[7]), bitPixel, localColorMap, grayScale, BitSet(buf[8], INTERLACE), 0, &state); } else { image = ReadImage(src, LM_to_uint(buf[4], buf[5]), LM_to_uint(buf[6], buf[7]), state.GifScreen.BitPixel, state.GifScreen.ColorMap, state.GifScreen.GrayScale, BitSet(buf[8], INTERLACE), 0, &state); } if (image) { if (state.Gif89.transparent >= 0) { SDL_SetColorKey(image, SDL_TRUE, state.Gif89.transparent); } frames = (Frame_t *)SDL_realloc(anim->frames, (anim->count + 1) * sizeof(*anim->frames)); if (!frames) { SDL_OutOfMemory(); goto done; } ++anim->count; anim->frames = frames; frame = &anim->frames[anim->count - 1]; frame->image = image; frame->x = LM_to_uint(buf[0], buf[1]); frame->y = LM_to_uint(buf[2], buf[3]); frame->disposal = state.Gif89.disposal; frame->delay = (state.Gif89.delayTime < 2 ? 10 : state.Gif89.delayTime) * 10; if (!load_anim) { /* We only need one frame, we're done */ goto done; } } } done: if (anim->count > 1) { /* Normalize the frames */ if (!NormalizeFrames(anim->frames, anim->count)) { int i; for (i = 0; i < anim->count; ++i) { SDL_FreeSurface(anim->frames[i].image); } anim->count = 0; } } if (anim->count == 0) { SDL_free(anim->frames); SDL_free(anim); return NULL; } return anim; } static int ReadColorMap(SDL_RWops *src, int number, unsigned char buffer[3][MAXCOLORMAPSIZE], int *gray) { int i; unsigned char rgb[3]; int flag; flag = TRUE; for (i = 0; i < number; ++i) { if (!ReadOK(src, rgb, sizeof(rgb))) { RWSetMsg("bad colormap"); return 1; } buffer[CM_RED][i] = rgb[0]; buffer[CM_GREEN][i] = rgb[1]; buffer[CM_BLUE][i] = rgb[2]; flag &= (rgb[0] == rgb[1] && rgb[1] == rgb[2]); } #if 0 if (flag) *gray = (number == 2) ? PBM_TYPE : PGM_TYPE; else *gray = PPM_TYPE; #else *gray = 0; #endif return FALSE; } static int DoExtension(SDL_RWops *src, int label, State_t * state) { unsigned char buf[256]; char *str; switch (label) { case 0x01: /* Plain Text Extension */ str = "Plain Text Extension"; break; case 0xff: /* Application Extension */ str = "Application Extension"; break; case 0xfe: /* Comment Extension */ str = "Comment Extension"; while (GetDataBlock(src, (unsigned char *) buf, state) > 0) ; return FALSE; case 0xf9: /* Graphic Control Extension */ str = "Graphic Control Extension"; (void) GetDataBlock(src, (unsigned char *) buf, state); state->Gif89.disposal = (buf[0] >> 2) & 0x7; state->Gif89.inputFlag = (buf[0] >> 1) & 0x1; state->Gif89.delayTime = LM_to_uint(buf[1], buf[2]); if ((buf[0] & 0x1) != 0) state->Gif89.transparent = buf[3]; while (GetDataBlock(src, (unsigned char *) buf, state) > 0) ; return FALSE; default: str = (char *)buf; SDL_snprintf(str, 256, "UNKNOWN (0x%02x)", label); break; } while (GetDataBlock(src, (unsigned char *) buf, state) > 0) ; return FALSE; } static int GetDataBlock(SDL_RWops *src, unsigned char *buf, State_t * state) { unsigned char count; if (!ReadOK(src, &count, 1)) { /* pm_message("error in getting DataBlock size" ); */ return -1; } state->ZeroDataBlock = count == 0; if ((count != 0) && (!ReadOK(src, buf, count))) { /* pm_message("error in reading DataBlock" ); */ return -1; } return count; } static int GetCode(SDL_RWops *src, int code_size, int flag, State_t * state) { int i, j, ret; unsigned char count; if (flag) { state->curbit = 0; state->lastbit = 0; state->done = FALSE; return 0; } if ((state->curbit + code_size) >= state->lastbit) { if (state->done) { if (state->curbit >= state->lastbit) RWSetMsg("ran off the end of my bits"); return -1; } state->buf[0] = state->buf[state->last_byte - 2]; state->buf[1] = state->buf[state->last_byte - 1]; if ((ret = GetDataBlock(src, &state->buf[2], state)) > 0) count = (unsigned char) ret; else { count = 0; state->done = TRUE; } state->last_byte = 2 + count; state->curbit = (state->curbit - state->lastbit) + 16; state->lastbit = (2 + count) * 8; } ret = 0; for (i = state->curbit, j = 0; j < code_size; ++i, ++j) ret |= ((state->buf[i / 8] & (1 << (i % 8))) != 0) << j; state->curbit += code_size; return ret; } static int LWZReadByte(SDL_RWops *src, int flag, int input_code_size, State_t * state) { int code, incode; register int i; /* Fixed buffer overflow found by Michael Skladnikiewicz */ if (input_code_size > MAX_LWZ_BITS) return -1; if (flag) { state->set_code_size = input_code_size; state->code_size = state->set_code_size + 1; state->clear_code = 1 << state->set_code_size; state->end_code = state->clear_code + 1; state->max_code_size = 2 * state->clear_code; state->max_code = state->clear_code + 2; GetCode(src, 0, TRUE, state); state->fresh = TRUE; for (i = 0; i < state->clear_code; ++i) { state->table[0][i] = 0; state->table[1][i] = i; } state->table[1][0] = 0; for (; i < (1 << MAX_LWZ_BITS); ++i) state->table[0][i] = 0; state->sp = state->stack; return 0; } else if (state->fresh) { state->fresh = FALSE; do { state->firstcode = state->oldcode = GetCode(src, state->code_size, FALSE, state); } while (state->firstcode == state->clear_code); return state->firstcode; } if (state->sp > state->stack) return *--state->sp; while ((code = GetCode(src, state->code_size, FALSE, state)) >= 0) { if (code == state->clear_code) { for (i = 0; i < state->clear_code; ++i) { state->table[0][i] = 0; state->table[1][i] = i; } for (; i < (1 << MAX_LWZ_BITS); ++i) state->table[0][i] = state->table[1][i] = 0; state->code_size = state->set_code_size + 1; state->max_code_size = 2 * state->clear_code; state->max_code = state->clear_code + 2; state->sp = state->stack; state->firstcode = state->oldcode = GetCode(src, state->code_size, FALSE, state); return state->firstcode; } else if (code == state->end_code) { int count; unsigned char buf[260]; if (state->ZeroDataBlock) return -2; while ((count = GetDataBlock(src, buf, state)) > 0) ; if (count != 0) { /* * pm_message("missing EOD in data stream (common occurence)"); */ } return -2; } incode = code; if (code >= state->max_code) { *state->sp++ = state->firstcode; code = state->oldcode; } while (code >= state->clear_code) { /* Guard against buffer overruns */ if (code < 0 || code >= (1 << MAX_LWZ_BITS)) { RWSetMsg("invalid LWZ data"); return -3; } *state->sp++ = state->table[1][code]; if (code == state->table[0][code]) { RWSetMsg("circular table entry BIG ERROR"); return -3; } code = state->table[0][code]; } /* Guard against buffer overruns */ if (code < 0 || code >= (1 << MAX_LWZ_BITS)) { RWSetMsg("invalid LWZ data"); return -4; } *state->sp++ = state->firstcode = state->table[1][code]; if ((code = state->max_code) < (1 << MAX_LWZ_BITS)) { state->table[0][code] = state->oldcode; state->table[1][code] = state->firstcode; ++state->max_code; if ((state->max_code >= state->max_code_size) && (state->max_code_size < (1 << MAX_LWZ_BITS))) { state->max_code_size *= 2; ++state->code_size; } } state->oldcode = incode; if (state->sp > state->stack) return *--state->sp; } return code; } static Image * ReadImage(SDL_RWops * src, int len, int height, int cmapSize, unsigned char cmap[3][MAXCOLORMAPSIZE], int gray, int interlace, int ignore, State_t * state) { Image *image; unsigned char c; int i, v; int xpos = 0, ypos = 0, pass = 0; (void) gray; /* unused */ /* ** Initialize the compression routines */ if (!ReadOK(src, &c, 1)) { RWSetMsg("EOF / read error on image data"); return NULL; } if (LWZReadByte(src, TRUE, c, state) < 0) { RWSetMsg("error reading image"); return NULL; } /* ** If this is an "uninteresting picture" ignore it. */ if (ignore) { while (LWZReadByte(src, FALSE, c, state) >= 0) ; return NULL; } image = ImageNewCmap(len, height, cmapSize); for (i = 0; i < cmapSize; i++) ImageSetCmap(image, i, cmap[CM_RED][i], cmap[CM_GREEN][i], cmap[CM_BLUE][i]); while ((v = LWZReadByte(src, FALSE, c, state)) >= 0) { ((Uint8 *)image->pixels)[xpos + ypos * image->pitch] = (Uint8)v; ++xpos; if (xpos == len) { xpos = 0; if (interlace) { switch (pass) { case 0: case 1: ypos += 8; break; case 2: ypos += 4; break; case 3: ypos += 2; break; } if (ypos >= height) { ++pass; switch (pass) { case 1: ypos = 4; break; case 2: ypos = 2; break; case 3: ypos = 1; break; default: goto fini; } } } else { ++ypos; } } if (ypos >= height) break; } fini: return image; } /* Load a GIF type animation from an SDL datasource */ IMG_Animation *IMG_LoadGIFAnimation_RW(SDL_RWops *src) { Anim_t *internal = IMG_LoadGIF_RW_Internal(src, SDL_TRUE); if (internal) { IMG_Animation *anim = (IMG_Animation *)SDL_malloc(sizeof(*anim)); if (anim) { anim->w = internal->frames[0].image->w; anim->h = internal->frames[0].image->h; anim->count = internal->count; anim->frames = (SDL_Surface **)SDL_calloc(anim->count, sizeof(*anim->frames)); anim->delays = (int *)SDL_calloc(anim->count, sizeof(*anim->delays)); if (anim->frames && anim->delays) { int i; for (i = 0; i < anim->count; ++i) { anim->frames[i] = internal->frames[i].image; anim->delays[i] = internal->frames[i].delay; } } else { IMG_FreeAnimation(anim); anim = NULL; } } if (!anim) { SDL_OutOfMemory(); } SDL_free(internal->frames); SDL_free(internal); return anim; } return NULL; } #else /* Load a GIF type animation from an SDL datasource */ IMG_Animation *IMG_LoadGIFAnimation_RW(SDL_RWops *src) { return NULL; } #endif /* LOAD_GIF */ #if !defined(__APPLE__) || defined(SDL_IMAGE_USE_COMMON_BACKEND) #ifdef LOAD_GIF /* See if an image is contained in a data source */ int IMG_isGIF(SDL_RWops *src) { Sint64 start; int is_GIF; char magic[6]; if ( !src ) return 0; start = SDL_RWtell(src); is_GIF = 0; if ( SDL_RWread(src, magic, sizeof(magic), 1) ) { if ( (SDL_strncmp(magic, "GIF", 3) == 0) && ((SDL_memcmp(magic + 3, "87a", 3) == 0) || (SDL_memcmp(magic + 3, "89a", 3) == 0)) ) { is_GIF = 1; } } SDL_RWseek(src, start, RW_SEEK_SET); return(is_GIF); } /* Load a GIF type image from an SDL datasource */ SDL_Surface *IMG_LoadGIF_RW(SDL_RWops *src) { SDL_Surface *image = NULL; Anim_t *internal = IMG_LoadGIF_RW_Internal(src, SDL_FALSE); if (internal) { image = internal->frames[0].image; SDL_free(internal->frames); SDL_free(internal); } return image; } #else #if _MSC_VER >= 1300 #pragma warning(disable : 4100) /* warning C4100: 'op' : unreferenced formal parameter */ #endif /* See if an image is contained in a data source */ int IMG_isGIF(SDL_RWops *src) { return(0); } /* Load a GIF type image from an SDL datasource */ SDL_Surface *IMG_LoadGIF_RW(SDL_RWops *src) { return(NULL); } #endif /* LOAD_GIF */ #endif /* !defined(__APPLE__) || defined(SDL_IMAGE_USE_COMMON_BACKEND) */