home *** CD-ROM | disk | FTP | other *** search
/ Media Share 9 / MEDIASHARE_09.ISO / graphics / gif-util.zip / GIFASM.C < prev    next >
C/C++ Source or Header  |  1989-08-01  |  11KB  |  302 lines

  1. /*****************************************************************************
  2. *   "Gif-Lib" - Yet another gif library.                     *
  3. *                                         *
  4. * Written by:  Gershon Elber            IBM PC Ver 0.1,    Jul. 1989    *
  5. ******************************************************************************
  6. * Program to assemble/disassemble GIF files: disassembles multi image file   *
  7. * into seperated files, or assembles few single image GIF files into one.    *
  8. * Options:                                     *
  9. * -a : assemble few files into one (default)                     *
  10. * -d name : disassmble given GIF file into seperate files derived from name. *
  11. * -h : on line help.                                 *
  12. ******************************************************************************
  13. * History:                                     *
  14. * 7 Jul 89 - Version 1.0 by Gershon Elber.                     *
  15. *****************************************************************************/
  16.  
  17. #include <stdio.h>
  18. #include <stdlib.h>
  19. #include <ctype.h>
  20. #include <alloc.h>
  21. #include <string.h>
  22. #include "gif_lib.h"
  23. #include "getarg.h"
  24.  
  25. #define PROGRAM_NAME    "GifAsm"
  26. #define VERSION        "ß Version 1.0, "
  27.  
  28. extern unsigned int
  29.     _stklen = 16384;                  /* Increase default stack size */
  30.  
  31. static char
  32.     *VersionStr =
  33.     PROGRAM_NAME
  34.     "    IBMPC "
  35.     VERSION
  36.     "    Gershon Elber,    "
  37.     __DATE__ ",   " __TIME__ "\n"
  38.     "(C) Copyright 1989 Gershon Elber, Non commercial use only.\n";
  39. static char
  40.     *CtrlStr =
  41.     PROGRAM_NAME
  42.     " a%- d%-OutFileName!s h%- GifFile(s)!*s";
  43. static char
  44.     *ProgramName;
  45. static int
  46.     AsmFlag = FALSE;
  47.  
  48. static void DoAssembly(int NumFiles, char **FileNames);
  49. static void DoDisassembly(char *InFileName, char *OutFileName);
  50. static void QuitGifError(GifFileType *GifFileIn, GifFileType *GifFileOut);
  51.  
  52. /******************************************************************************
  53. * Interpret the command line and scan the given GIF file.              *
  54. ******************************************************************************/
  55. void main(int argc, char **argv)
  56. {
  57.     int    Error, NumFiles, DisasmFlag = FALSE, HelpFlag = FALSE;
  58.     char **FileNames = NULL, *OutFileName;
  59.  
  60.     if (strlen(ProgramName = argv[0]) == 0)            /* DOS 3.x only! */
  61.     ProgramName = PROGRAM_NAME;      /* Do something reasonable for 2.x */
  62.  
  63.     if ((Error = GAGetArgs(argc, argv, CtrlStr,
  64.         &AsmFlag, &DisasmFlag, &OutFileName,
  65.         &HelpFlag, &NumFiles, &FileNames)) != FALSE) {
  66.     GAPrintErrMsg(Error);
  67.     GAPrintHowTo(CtrlStr);
  68.     exit(1);
  69.     }
  70.  
  71.     if (HelpFlag) {
  72.     fprintf(stderr, VersionStr);
  73.     GAPrintHowTo(CtrlStr);
  74.     exit(0);
  75.     }
  76.  
  77.     if (!AsmFlag && !DisasmFlag) AsmFlag = TRUE;  /* Make default - assemble */
  78.     if (AsmFlag && NumFiles < 2) {
  79.     MESSAGE("At list two GIF files are required to assembly operation\n");
  80.     GAPrintHowTo(CtrlStr);
  81.     exit(1);
  82.     }
  83.     if (!AsmFlag && NumFiles > 1) {
  84.     MESSAGE("Error in command line parsing - one GIF file please\n");
  85.     GAPrintHowTo(CtrlStr);
  86.     exit(1);
  87.     }
  88.  
  89.     if (AsmFlag) DoAssembly(NumFiles, FileNames);
  90.     else DoDisassembly(NumFiles == 1 ? *FileNames : NULL, OutFileName);
  91. }
  92.  
  93. /******************************************************************************
  94. * Perform the assembly operation - take few input files into one output.      *
  95. ******************************************************************************/
  96. static void DoAssembly(int NumFiles, char **FileNames)
  97. {
  98.     int    i, ExtCode, CodeSize;
  99.     GifRecordType RecordType;
  100.     ByteType *Extension, *CodeBlock;
  101.     GifFileType *GifFileIn = NULL, *GifFileOut = NULL;
  102.  
  103.     /* Open stdout for the output file: */
  104.     if ((GifFileOut = EGifOpenFileHandle(1)) == NULL)
  105.     QuitGifError(GifFileIn, GifFileOut);
  106.  
  107.     /* Scan the content of the GIF file and load the image(s) in: */
  108.     for (i=0; i<NumFiles; i++) {
  109.     if ((GifFileIn = DGifOpenFileName(FileNames[i])) == NULL)
  110.         QuitGifError(GifFileIn, GifFileOut);
  111.  
  112.     /* And dump out screen descriptor iff its first image.    */
  113.     if (i == 0)
  114.         if (EGifPutScreenDesc(GifFileOut,
  115.         GifFileIn -> SWidth, GifFileIn -> SHeight,
  116.         GifFileIn -> SColorResolution, GifFileIn -> SBackGroundColor,
  117.         GifFileIn -> SBitsPerPixel, GifFileIn -> SColorMap) == ERROR)
  118.         QuitGifError(GifFileIn, GifFileOut);
  119.  
  120.     do {
  121.         if (DGifGetRecordType(GifFileIn, &RecordType) == ERROR)
  122.         QuitGifError(GifFileIn, GifFileOut);
  123.  
  124.         switch (RecordType) {
  125.         case IMAGE_DESC_RECORD_TYPE:
  126.             if (DGifGetImageDesc(GifFileIn) == ERROR)
  127.             QuitGifError(GifFileIn, GifFileOut);
  128.             /* Put image descriptor to out file: */
  129.             if (EGifPutImageDesc(GifFileOut,
  130.             GifFileIn -> ILeft, GifFileIn -> ITop,
  131.             GifFileIn -> IWidth, GifFileIn -> IHeight,
  132.             GifFileIn -> IInterlace, GifFileIn -> IBitsPerPixel,
  133.             GifFileIn -> IColorMap) == ERROR)
  134.             QuitGifError(GifFileIn, GifFileOut);
  135.  
  136.             /* Now read image itself in decoded form as we dont      */
  137.             /* dont care what is there, and this is much faster.     */
  138.             if (DGifGetCode(GifFileIn, &CodeSize, &CodeBlock) == ERROR
  139.              || EGifPutCode(GifFileOut, CodeSize, CodeBlock) == ERROR)
  140.             QuitGifError(GifFileIn, GifFileOut);
  141.             while (CodeBlock != NULL)
  142.             if (DGifGetCodeNext(GifFileIn, &CodeBlock) == ERROR ||
  143.                 EGifPutCodeNext(GifFileOut, CodeBlock) == ERROR)
  144.                 QuitGifError(GifFileIn, GifFileOut);
  145.             break;
  146.         case EXTENSION_RECORD_TYPE:
  147.             /* Skip any extension blocks in file: */
  148.             if (DGifGetExtension(GifFileIn, &ExtCode, &Extension)
  149.             == ERROR)
  150.             QuitGifError(GifFileIn, GifFileOut);
  151.             if (EGifPutExtension(GifFileOut, ExtCode, Extension[0],
  152.                             Extension) == ERROR)
  153.             QuitGifError(GifFileIn, GifFileOut);
  154.  
  155.             /* No support to more than one extension blocks, discard */
  156.             while (Extension != NULL)
  157.             if (DGifGetExtensionNext(GifFileIn, &Extension)
  158.                 == ERROR)
  159.                 QuitGifError(GifFileIn, GifFileOut);
  160.             break;
  161.         case TERMINATE_RECORD_TYPE:
  162.             break;
  163.         default:         /* Should be traps by DGifGetRecordType */
  164.             break;
  165.         }
  166.     }
  167.     while (RecordType != TERMINATE_RECORD_TYPE);
  168.  
  169.     if (DGifCloseFile(GifFileIn) == ERROR)
  170.         QuitGifError(GifFileIn, GifFileOut);
  171.     }
  172.  
  173.     if (EGifCloseFile(GifFileOut) == ERROR)
  174.     QuitGifError(GifFileIn, GifFileOut);
  175. }
  176.  
  177. /******************************************************************************
  178. * Perform the disassembly operation - take one input files into few output.   *
  179. ******************************************************************************/
  180. static void DoDisassembly(char *InFileName, char *OutFileName)
  181. {
  182.     int    ExtCode, CodeSize, FileNum = 0, FileEmpty;
  183.     GifRecordType RecordType;
  184.     char CrntFileName[80], *p;
  185.     ByteType *Extension, *CodeBlock;
  186.     GifFileType *GifFileIn = NULL, *GifFileOut = NULL;
  187.  
  188.     /* If name has type postfix, strip it out, and make sure name is less    */
  189.     /* or equal to 6 chars, so we will have 2 chars in name for numbers.     */
  190.     strupr(OutFileName);              /* Make sure all is upper case */
  191.     if ((p = strrchr(OutFileName, '.')) != NULL && strlen(p) <= 4) p[0] = 0;
  192.     if ((p = strrchr(OutFileName, '/')) != NULL ||
  193.     (p = strrchr(OutFileName, '\\')) != NULL ||
  194.     (p = strrchr(OutFileName, ':')) != NULL) {
  195.     if (strlen(p) > 7) p[7] = 0;   /* p includes the '/', '\\', ':' char */
  196.     }
  197.     else {
  198.     /* Only name is given for current directory: */
  199.     if (strlen(OutFileName) > 6) OutFileName[6] = 0;
  200.     }
  201.  
  202.     /* Open input file: */
  203.     if (InFileName != NULL) {
  204.     if ((GifFileIn = DGifOpenFileName(InFileName)) == NULL)
  205.         QuitGifError(GifFileIn, GifFileOut);
  206.     }
  207.     else {
  208.     /* Use the stdin instead: */
  209.     if ((GifFileIn = DGifOpenFileHandle(0)) == NULL)
  210.         QuitGifError(GifFileIn, GifFileOut);
  211.     }
  212.  
  213.     /* Scan the content of GIF file and dump image(s) to seperate file(s): */
  214.     do {
  215.     sprintf(CrntFileName, "%s%02d.gif", OutFileName, FileNum++);
  216.     if ((GifFileOut = EGifOpenFileName(CrntFileName, TRUE)) == NULL)
  217.         QuitGifError(GifFileIn, GifFileOut);
  218.     FileEmpty = TRUE;
  219.  
  220.     /* And dump out its exactly same screen information: */
  221.     if (EGifPutScreenDesc(GifFileOut,
  222.         GifFileIn -> SWidth, GifFileIn -> SHeight,
  223.         GifFileIn -> SColorResolution, GifFileIn -> SBackGroundColor,
  224.         GifFileIn -> SBitsPerPixel, GifFileIn -> SColorMap) == ERROR)
  225.         QuitGifError(GifFileIn, GifFileOut);
  226.  
  227.     do {
  228.         if (DGifGetRecordType(GifFileIn, &RecordType) == ERROR)
  229.         QuitGifError(GifFileIn, GifFileOut);
  230.  
  231.         switch (RecordType) {
  232.         case IMAGE_DESC_RECORD_TYPE:
  233.             FileEmpty = FALSE;
  234.             if (DGifGetImageDesc(GifFileIn) == ERROR)
  235.             QuitGifError(GifFileIn, GifFileOut);
  236.             /* Put same image descriptor to out file: */
  237.             if (EGifPutImageDesc(GifFileOut,
  238.             GifFileIn -> ILeft, GifFileIn -> ITop,
  239.             GifFileIn -> IWidth, GifFileIn -> IHeight,
  240.             GifFileIn -> IInterlace, GifFileIn -> IBitsPerPixel,
  241.             GifFileIn -> IColorMap) == ERROR)
  242.             QuitGifError(GifFileIn, GifFileOut);
  243.  
  244.             /* Now read image itself in decoded form as we dont      */
  245.             /* really care what is there, and this is much faster.   */
  246.             if (DGifGetCode(GifFileIn, &CodeSize, &CodeBlock) == ERROR
  247.              || EGifPutCode(GifFileOut, CodeSize, CodeBlock) == ERROR)
  248.             QuitGifError(GifFileIn, GifFileOut);
  249.             while (CodeBlock != NULL)
  250.             if (DGifGetCodeNext(GifFileIn, &CodeBlock) == ERROR ||
  251.                 EGifPutCodeNext(GifFileOut, CodeBlock) == ERROR)
  252.             QuitGifError(GifFileIn, GifFileOut);
  253.             break;
  254.         case EXTENSION_RECORD_TYPE:
  255.             FileEmpty = FALSE;
  256.             /* Skip any extension blocks in file: */
  257.             if (DGifGetExtension(GifFileIn, &ExtCode, &Extension)
  258.             == ERROR)
  259.             QuitGifError(GifFileIn, GifFileOut);
  260.             if (EGifPutExtension(GifFileOut, ExtCode, Extension[0],
  261.                             Extension) == ERROR)
  262.             QuitGifError(GifFileIn, GifFileOut);
  263.  
  264.             /* No support to more than one extension blocks, discard */
  265.             while (Extension != NULL)
  266.             if (DGifGetExtensionNext(GifFileIn, &Extension)
  267.                 == ERROR)
  268.                 QuitGifError(GifFileIn, GifFileOut);
  269.             break;
  270.         case TERMINATE_RECORD_TYPE:
  271.             break;
  272.         default:         /* Should be traps by DGifGetRecordType */
  273.             break;
  274.         }
  275.     }
  276.     while (RecordType != IMAGE_DESC_RECORD_TYPE &&
  277.            RecordType != TERMINATE_RECORD_TYPE);
  278.  
  279.     if (EGifCloseFile(GifFileOut) == ERROR)
  280.         QuitGifError(GifFileIn, GifFileOut);
  281.     if (FileEmpty) {
  282.         /* Might happen on last file - delete it if so: */
  283.         unlink(CrntFileName);
  284.     }
  285.    }
  286.     while (RecordType != TERMINATE_RECORD_TYPE);
  287.  
  288.     if (DGifCloseFile(GifFileIn) == ERROR)
  289.     QuitGifError(GifFileIn, GifFileOut);
  290. }
  291.  
  292. /******************************************************************************
  293. * Close both input and output file (if open), and exit.                  *
  294. ******************************************************************************/
  295. static void QuitGifError(GifFileType *GifFileIn, GifFileType *GifFileOut)
  296. {
  297.     PrintGifError();
  298.     if (GifFileIn != NULL) DGifCloseFile(GifFileIn);
  299.     if (GifFileOut != NULL) EGifCloseFile(GifFileOut);
  300.     exit(1);
  301. }
  302.