]> granicus.if.org Git - nethack/commitdiff
*** empty log message ***
authorjwalz <jwalz>
Sat, 5 Jan 2002 21:06:02 +0000 (21:06 +0000)
committerjwalz <jwalz>
Sat, 5 Jan 2002 21:06:02 +0000 (21:06 +0000)
win/share/gifread.c [new file with mode: 0644]

diff --git a/win/share/gifread.c b/win/share/gifread.c
new file mode 100644 (file)
index 0000000..9757227
--- /dev/null
@@ -0,0 +1,706 @@
+/* GIF reading routines based on those in pbmplus:ppm/giftoppm.c, bearing
+ * following copyright notice:
+ */
+
+/* +-------------------------------------------------------------------+ */
+/* | Copyright 1990, David Koblas.                                     | */
+/* |   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.           | */
+/* +-------------------------------------------------------------------+ */
+
+
+#include "config.h"
+#include "tile.h"
+
+#ifndef MONITOR_HEAP
+extern long *FDECL(alloc, (unsigned int));
+#endif
+
+#define PPM_ASSIGN(p,red,grn,blu) do { (p).r = (red); (p).g = (grn); (p).b = (blu); } while ( 0 )
+
+#define        MAX_LWZ_BITS            12
+
+#define INTERLACE              0x40
+#define LOCALCOLORMAP  0x80
+#define BitSet(byte, bit)      (((byte) & (bit)) == (bit))
+
+#define        ReadOK(file,buffer,len) (fread((genericptr_t)buffer, (int)len, 1, file) != 0)
+
+#define LM_to_uint(a,b)                        (((b)<<8)|(a))
+
+struct gifscreen {
+       int     Width;
+       int     Height;
+       int     Colors;
+       int     ColorResolution;
+       int     Background;
+       int     AspectRatio;
+       int     Interlace;
+} GifScreen;
+
+struct {
+       int     transparent;
+       int     delayTime;
+       int     inputFlag;
+       int     disposal;
+} Gif89 = { -1, -1, -1, 0 };
+
+int    ZeroDataBlock = FALSE;
+
+static FILE *gif_file;
+static int tiles_across, tiles_down, curr_tiles_across, curr_tiles_down;
+static pixel **image;
+static unsigned char input_code_size;
+
+static int FDECL(GetDataBlock, (FILE *fd, unsigned char *buf));
+static void FDECL(DoExtension, (FILE *fd, int label));
+static boolean FDECL(ReadColorMap, (FILE *fd, int number));
+static void FDECL(read_header, (FILE *fd));
+static int FDECL(GetCode, (FILE *fd, int code_size, int flag));
+static int FDECL(LWZReadByte, (FILE *fd, int flag, int input_code_size));
+static void FDECL(ReadInterleavedImage, (FILE *fd, int len, int height));
+static void FDECL(ReadTileStrip, (FILE *fd, int len));
+
+/* These should be in gif.h, but there isn't one. */
+boolean FDECL(fopen_gif_file, (const char *, const char *));
+boolean FDECL(read_gif_tile, (pixel(*)[]));
+int NDECL(fclose_gif_file);
+
+static int
+GetDataBlock(fd, buf)
+FILE           *fd;
+unsigned char  *buf;
+{
+       unsigned char   count;
+
+       if (!ReadOK(fd,&count,1)) {
+               Fprintf(stderr, "error in getting DataBlock size\n");
+               return -1;
+       }
+
+       ZeroDataBlock = (count == 0);
+
+       if ((count != 0) && (!ReadOK(fd, buf, count))) {
+               Fprintf(stderr, "error in reading DataBlock\n");
+               return -1;
+       }
+
+       return count;
+}
+
+static void
+DoExtension(fd, label)
+FILE   *fd;
+int    label;
+{
+       static char     buf[256];
+       char            *str;
+
+       switch (label) {
+       case 0x01:              /* Plain Text Extension */
+               str = "Plain Text Extension";
+#ifdef notdef
+               if (GetDataBlock(fd, (unsigned char*) buf) == 0)
+                       ;
+
+               lpos   = LM_to_uint(buf[0], buf[1]);
+               tpos   = LM_to_uint(buf[2], buf[3]);
+               width  = LM_to_uint(buf[4], buf[5]);
+               height = LM_to_uint(buf[6], buf[7]);
+               cellw  = buf[8];
+               cellh  = buf[9];
+               foreground = buf[10];
+               background = buf[11];
+
+               while (GetDataBlock(fd, (unsigned char*) buf) != 0) {
+                       PPM_ASSIGN(image[ypos][xpos],
+                                       cmap[CM_RED][v],
+                                       cmap[CM_GREEN][v],
+                                       cmap[CM_BLUE][v]);
+                       ++index;
+               }
+
+               return;
+#else
+               break;
+#endif
+       case 0xff:              /* Application Extension */
+               str = "Application Extension";
+               break;
+       case 0xfe:              /* Comment Extension */
+               str = "Comment Extension";
+               while (GetDataBlock(fd, (unsigned char*) buf) != 0) {
+                       Fprintf(stderr, "gif comment: %s\n", buf );
+               }
+               return;
+       case 0xf9:              /* Graphic Control Extension */
+               str = "Graphic Control Extension";
+               (void) GetDataBlock(fd, (unsigned char*) buf);
+               Gif89.disposal    = (buf[0] >> 2) & 0x7;
+               Gif89.inputFlag   = (buf[0] >> 1) & 0x1;
+               Gif89.delayTime   = LM_to_uint(buf[1],buf[2]);
+               if ((buf[0] & 0x1) != 0)
+                       Gif89.transparent = buf[3];
+
+               while (GetDataBlock(fd, (unsigned char*) buf) != 0)
+                       ;
+               return;
+       default:
+               str = buf;
+               Sprintf(buf, "UNKNOWN (0x%02x)", label);
+               break;
+       }
+
+       Fprintf(stderr, "got a '%s' extension\n", str);
+
+       while (GetDataBlock(fd, (unsigned char*) buf) != 0)
+               ;
+}
+
+static
+boolean
+ReadColorMap(fd,number)
+FILE           *fd;
+int            number;
+{
+       int             i;
+       unsigned char   rgb[3];
+
+       for (i = 0; i < number; ++i) {
+               if (!ReadOK(fd, rgb, sizeof(rgb))) {
+                       return(FALSE);
+               }
+
+               ColorMap[CM_RED][i] = rgb[0] ;
+               ColorMap[CM_GREEN][i] = rgb[1] ;
+               ColorMap[CM_BLUE][i] = rgb[2] ;
+       }
+       colorsinmap = number;
+       return TRUE;
+}
+
+/*
+ * Read gif header, including colormaps.  We expect only one image per
+ * file, so if that image has a local colormap, overwrite the global one.
+ */
+static void
+read_header(fd)
+FILE   *fd;
+{
+       unsigned char   buf[16];
+       unsigned char   c;
+       char            version[4];
+
+       if (!ReadOK(fd,buf,6)) {
+               Fprintf(stderr, "error reading magic number\n");
+               exit(EXIT_FAILURE);
+       }
+
+       if (strncmp((genericptr_t)buf,"GIF",3) != 0) {
+               Fprintf(stderr, "not a GIF file\n");
+               exit(EXIT_FAILURE);
+       }
+
+       (void) strncpy(version, (char *)buf + 3, 3);
+       version[3] = '\0';
+
+       if ((strcmp(version, "87a") != 0) && (strcmp(version, "89a") != 0)) {
+               Fprintf(stderr, "bad version number, not '87a' or '89a'\n");
+               exit(EXIT_FAILURE);
+       }
+
+       if (!ReadOK(fd,buf,7)) {
+               Fprintf(stderr, "failed to read screen descriptor\n");
+               exit(EXIT_FAILURE);
+       }
+
+       GifScreen.Width           = LM_to_uint(buf[0],buf[1]);
+       GifScreen.Height          = LM_to_uint(buf[2],buf[3]);
+       GifScreen.Colors          = 2<<(buf[4]&0x07);
+       GifScreen.ColorResolution = (((buf[4]&0x70)>>3)+1);
+       GifScreen.Background      = buf[5];
+       GifScreen.AspectRatio     = buf[6];
+
+       if (BitSet(buf[4], LOCALCOLORMAP)) {    /* Global Colormap */
+               if (!ReadColorMap(fd, GifScreen.Colors)) {
+                       Fprintf(stderr, "error reading global colormap\n");
+                       exit(EXIT_FAILURE);
+               }
+       }
+
+       if (GifScreen.AspectRatio != 0 && GifScreen.AspectRatio != 49) {
+               Fprintf(stderr, "warning - non-square pixels\n");
+       }
+
+       for (;;) {
+               if (!ReadOK(fd,&c,1)) {
+                       Fprintf(stderr, "EOF / read error on image data\n");
+                       exit(EXIT_FAILURE);
+               }
+
+               if (c == ';') {         /* GIF terminator */
+                       return;
+               }
+
+               if (c == '!') {         /* Extension */
+                   if (!ReadOK(fd,&c,1)) {
+                       Fprintf(stderr,
+                           "EOF / read error on extension function code\n");
+                       exit(EXIT_FAILURE);
+                   }
+                   DoExtension(fd, (int)c);
+                   continue;
+               }
+
+               if (c != ',') {         /* Not a valid start character */
+                       Fprintf(stderr,
+                               "bogus character 0x%02x, ignoring\n", (int) c);
+                       continue;
+               }
+
+               if (!ReadOK(fd,buf,9)) {
+                   Fprintf(stderr, "couldn't read left/top/width/height\n");
+                   exit(EXIT_FAILURE);
+               }
+
+               if (BitSet(buf[8], LOCALCOLORMAP)) {
+                       /* replace global color map with local */
+                       GifScreen.Colors = 1<<((buf[8]&0x07)+1);
+                       if (!ReadColorMap(fd, GifScreen.Colors)) {
+                           Fprintf(stderr, "error reading local colormap\n");
+                           exit(EXIT_FAILURE);
+                       }
+
+               }
+               if (GifScreen.Width != LM_to_uint(buf[4],buf[5])) {
+                       Fprintf(stderr, "warning: widths don't match\n");
+                       GifScreen.Width = LM_to_uint(buf[4],buf[5]);
+               }
+               if (GifScreen.Height != LM_to_uint(buf[6],buf[7])) {
+                       Fprintf(stderr, "warning: heights don't match\n");
+                       GifScreen.Height = LM_to_uint(buf[6],buf[7]);
+               }
+               GifScreen.Interlace = BitSet(buf[8], INTERLACE);
+               return;
+       }
+}
+
+static int
+GetCode(fd, code_size, flag)
+FILE   *fd;
+int    code_size;
+int    flag;
+{
+       static unsigned char    buf[280];
+       static int              curbit, lastbit, done, last_byte;
+       int                     i, j, ret;
+       unsigned char           count;
+
+       if (flag) {
+               curbit = 0;
+               lastbit = 0;
+               done = FALSE;
+               return 0;
+       }
+
+       if ((curbit+code_size) >= lastbit) {
+               if (done) {
+                       if (curbit >= lastbit)
+                               Fprintf(stderr, "ran off the end of my bits\n");
+                       return -1;
+               }
+               buf[0] = buf[last_byte-2];
+               buf[1] = buf[last_byte-1];
+
+               if ((count = GetDataBlock(fd, &buf[2])) == 0)
+                       done = TRUE;
+
+               last_byte = 2 + count;
+               curbit = (curbit - lastbit) + 16;
+               lastbit = (2+count)*8 ;
+       }
+
+       ret = 0;
+       for (i = curbit, j = 0; j < code_size; ++i, ++j)
+               ret |= ((buf[ i / 8 ] & (1 << (i % 8))) != 0) << j;
+
+       curbit += code_size;
+
+       return ret;
+}
+
+static int
+LWZReadByte(fd, flag, input_code_size)
+FILE   *fd;
+int    flag;
+int    input_code_size;
+{
+       static int      fresh = FALSE;
+       int             code, incode;
+       static int      code_size, set_code_size;
+       static int      max_code, max_code_size;
+       static int      firstcode, oldcode;
+       static int      clear_code, end_code;
+       static int      table[2][(1<< MAX_LWZ_BITS)];
+       static int      stack[(1<<(MAX_LWZ_BITS))*2], *sp;
+       register int    i;
+
+       if (flag) {
+               set_code_size = input_code_size;
+               code_size = set_code_size+1;
+               clear_code = 1 << set_code_size ;
+               end_code = clear_code + 1;
+               max_code_size = 2*clear_code;
+               max_code = clear_code+2;
+
+               (void) GetCode(fd, 0, TRUE);
+
+               fresh = TRUE;
+
+               for (i = 0; i < clear_code; ++i) {
+                       table[0][i] = 0;
+                       table[1][i] = i;
+               }
+               for (; i < (1<<MAX_LWZ_BITS); ++i)
+                       table[0][i] = table[1][0] = 0;
+
+               sp = stack;
+
+               return 0;
+       } else if (fresh) {
+               fresh = FALSE;
+               do {
+                       firstcode = oldcode = GetCode(fd, code_size, FALSE);
+               } while (firstcode == clear_code);
+               return firstcode;
+       }
+
+       if (sp > stack)
+               return *--sp;
+
+       while ((code = GetCode(fd, code_size, FALSE)) >= 0) {
+               if (code == clear_code) {
+                       for (i = 0; i < clear_code; ++i) {
+                               table[0][i] = 0;
+                               table[1][i] = i;
+                       }
+                       for (; i < (1<<MAX_LWZ_BITS); ++i)
+                               table[0][i] = table[1][i] = 0;
+                       code_size = set_code_size+1;
+                       max_code_size = 2*clear_code;
+                       max_code = clear_code+2;
+                       sp = stack;
+                       firstcode = oldcode = GetCode(fd, code_size, FALSE);
+                       return firstcode;
+               } else if (code == end_code) {
+                       int             count;
+                       unsigned char   buf[260];
+
+                       if (ZeroDataBlock)
+                               return -2;
+
+                       while ((count = GetDataBlock(fd, buf)) > 0)
+                               ;
+
+                       if (count != 0)
+                           Fprintf(stderr,
+                           "missing EOD in data stream (common occurrence)\n");
+                       return -2;
+               }
+
+               incode = code;
+
+               if (code >= max_code) {
+                       *sp++ = firstcode;
+                       code = oldcode;
+               }
+
+               while (code >= clear_code) {
+                       *sp++ = table[1][code];
+                       if (code == table[0][code]) {
+                           Fprintf(stderr, "circular table entry BIG ERROR\n");
+                           exit(EXIT_FAILURE);
+                       }
+                       code = table[0][code];
+               }
+
+               *sp++ = firstcode = table[1][code];
+
+               if ((code = max_code) <(1<<MAX_LWZ_BITS)) {
+                       table[0][code] = oldcode;
+                       table[1][code] = firstcode;
+                       ++max_code;
+                       if ((max_code >= max_code_size) &&
+                               (max_code_size < (1<<MAX_LWZ_BITS))) {
+                               max_code_size *= 2;
+                               ++code_size;
+                       }
+               }
+
+               oldcode = incode;
+
+               if (sp > stack)
+                       return *--sp;
+       }
+       return code;
+}
+
+
+static void
+ReadInterleavedImage(fd, len, height)
+FILE   *fd;
+int    len, height;
+{
+       int             v;
+       int             xpos = 0, ypos = 0, pass = 0;
+
+       while ((v = LWZReadByte(fd,FALSE,(int)input_code_size)) >= 0 ) {
+               PPM_ASSIGN(image[ypos][xpos], ColorMap[CM_RED][v],
+                               ColorMap[CM_GREEN][v], ColorMap[CM_BLUE][v]);
+
+               ++xpos;
+               if (xpos == len) {
+                       xpos = 0;
+                       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;
+                               }
+                       }
+               }
+               if (ypos >= height)
+                       break;
+       }
+
+fini:
+       if (LWZReadByte(fd,FALSE,(int)input_code_size)>=0)
+               Fprintf(stderr, "too much input data, ignoring extra...\n");
+}
+
+static
+void
+ReadTileStrip(fd,len)
+FILE *fd;
+int len;
+{
+       int     v;
+       int     xpos = 0, ypos = 0;
+
+       while ((v = LWZReadByte(fd,FALSE,(int)input_code_size)) >= 0 ) {
+               PPM_ASSIGN(image[ypos][xpos], ColorMap[CM_RED][v],
+                               ColorMap[CM_GREEN][v], ColorMap[CM_BLUE][v]);
+
+               ++xpos;
+               if (xpos == len) {
+                       xpos = 0;
+                       ++ypos;
+               }
+               if (ypos >= TILE_Y)
+                       break;
+       }
+}
+
+
+
+
+boolean
+fopen_gif_file(filename, type)
+const char *filename;
+const char *type;
+{
+       int i;
+
+       if (strcmp(type, RDBMODE)) {
+               Fprintf(stderr, "using reading routine for non-reading?\n");
+               return FALSE;
+       }
+       gif_file = fopen(filename, type);
+       if (gif_file == (FILE *)0) {
+               Fprintf(stderr, "cannot open gif file %s\n", filename);
+               return FALSE;
+       }
+
+       read_header(gif_file);
+       if (GifScreen.Width % TILE_X) {
+               Fprintf(stderr, "error: width %d not divisible by %d\n",
+                               GifScreen.Width, TILE_X);
+               exit(EXIT_FAILURE);
+       }
+       tiles_across = GifScreen.Width / TILE_X;
+       curr_tiles_across = 0;
+       if (GifScreen.Height % TILE_Y) {
+               Fprintf(stderr, "error: height %d not divisible by %d\n",
+                               GifScreen.Height, TILE_Y);
+               /* exit(EXIT_FAILURE) */;
+       }
+       tiles_down = GifScreen.Height / TILE_Y;
+       curr_tiles_down = 0;
+
+       if (GifScreen.Interlace) {
+           /* sigh -- hope this doesn't happen on micros */
+           image = (pixel **)alloc(GifScreen.Height * sizeof(pixel *));
+           for (i = 0; i < GifScreen.Height; i++) {
+               image[i] = (pixel *) alloc(GifScreen.Width * sizeof(pixel));
+           }
+       } else {
+           image = (pixel **)alloc(TILE_Y * sizeof(pixel *));
+           for (i = 0; i < TILE_Y; i++) {
+               image[i] = (pixel *) alloc(GifScreen.Width * sizeof(pixel));
+           }
+       }
+
+       /*
+       **  Initialize the Compression routines
+       */
+       if (!ReadOK(gif_file,&input_code_size,1)) {
+               Fprintf(stderr, "EOF / read error on image data\n");
+               exit(EXIT_FAILURE);
+       }
+
+       if (LWZReadByte(gif_file, TRUE, (int)input_code_size) < 0) {
+               Fprintf(stderr, "error reading image\n");
+               exit(EXIT_FAILURE);
+       }
+
+       /* read first section */
+       if (GifScreen.Interlace) {
+               ReadInterleavedImage(gif_file,
+                                       GifScreen.Width,GifScreen.Height);
+       } else {
+               ReadTileStrip(gif_file,GifScreen.Width);
+       }
+       return TRUE;
+}
+
+/* Read a tile.  Returns FALSE when there are no more tiles */
+boolean
+read_gif_tile(pixels)
+pixel (*pixels)[TILE_X];
+{
+       int i, j;
+
+       if (curr_tiles_down >= tiles_down) return FALSE;
+       if (curr_tiles_across == tiles_across) {
+               curr_tiles_across = 0;
+               curr_tiles_down++;
+               if (curr_tiles_down >= tiles_down) return FALSE;
+               if (!GifScreen.Interlace)
+                       ReadTileStrip(gif_file,GifScreen.Width);
+       }
+       if (GifScreen.Interlace) {
+               for (j = 0; j < TILE_Y; j++) {
+                   for (i = 0; i < TILE_X; i++) {
+                       pixels[j][i] = image[curr_tiles_down*TILE_Y + j]
+                                           [curr_tiles_across*TILE_X + i];
+                   }
+               }
+       } else {
+               for (j = 0; j < TILE_Y; j++) {
+                   for (i = 0; i < TILE_X; i++) {
+                       pixels[j][i] = image[j][curr_tiles_across*TILE_X + i];
+                   }
+               }
+       }
+       curr_tiles_across++;
+
+       /* check for "filler" tile */
+       for (j = 0; j < TILE_Y; j++) {
+               for (i = 0; i < TILE_X && i < 4; i += 2) {
+                       if (pixels[j][i].r != ColorMap[CM_RED][0] ||
+                           pixels[j][i].g != ColorMap[CM_GREEN][0] ||
+                           pixels[j][i].b != ColorMap[CM_BLUE][0] ||
+                           pixels[j][i+1].r != ColorMap[CM_RED][1] ||
+                           pixels[j][i+1].g != ColorMap[CM_GREEN][1] ||
+                           pixels[j][i+1].b != ColorMap[CM_BLUE][1])
+                               return TRUE;
+               }
+       }
+       return FALSE;
+}
+
+int
+fclose_gif_file()
+{
+       int i;
+
+       if (GifScreen.Interlace) {
+               for (i = 0; i < GifScreen.Height; i++) {
+                       free((genericptr_t)image[i]);
+               }
+               free((genericptr_t)image);
+       } else {
+               for (i = 0; i < TILE_Y; i++) {
+                       free((genericptr_t)image[i]);
+               }
+               free((genericptr_t)image);
+       }
+       return(fclose(gif_file));
+}
+
+#ifndef AMIGA
+static char *std_args[] = { "tilemap", /* dummy argv[0] */
+                       "monsters.gif", "monsters.txt",
+                       "objects.gif",  "objects.txt",
+                       "other.gif",    "other.txt" };
+
+int
+main(argc, argv)
+int argc;
+char *argv[];
+{
+       pixel pixels[TILE_Y][TILE_X];
+
+       if (argc == 1) {
+               argc = SIZE(std_args);
+               argv = std_args;
+       } else if (argc != 3) {
+               Fprintf(stderr, "usage: gif2txt giffile txtfile\n");
+               exit(EXIT_FAILURE);
+       }
+
+       while (argc > 1) {
+               if (!fopen_gif_file(argv[1], RDBMODE))
+                       exit(EXIT_FAILURE);
+
+               init_colormap();
+
+               if (!fopen_text_file(argv[2], WRTMODE)) {
+                       (void) fclose_gif_file();
+                       exit(EXIT_FAILURE);
+               }
+
+               while (read_gif_tile(pixels))
+                       (void) write_text_tile(pixels);
+
+               (void) fclose_gif_file();
+               (void) fclose_text_file();
+
+               argc -= 2;
+               argv += 2;
+       }
+       exit(EXIT_SUCCESS);
+       /*NOTREACHED*/
+       return 0;
+}
+#endif