unzip.c
上传用户:jnfxsk
上传日期:2022-06-16
资源大小:3675k
文件大小:39k
源码类别:

游戏引擎

开发平台:

Visual C++

  1. /* unzip.c -- IO for uncompress .zip files using zlib
  2. Version 1.01, May 8th, 2004
  3. Copyright (C) 1998-2004 Gilles Vollant
  4. Read unzip.h for more info
  5.  */
  6. #include <stdio.h>
  7. #include <stdlib.h>
  8. #include <string.h>
  9. #include "zlib.h"
  10. #include "unzip.h"
  11. #ifdef STDC
  12.   #include <stddef.h>
  13.   #include <string.h>
  14.   #include <stdlib.h>
  15. #endif
  16. #ifdef NO_ERRNO_H
  17.   extern int errno;
  18. #else
  19.   #include <errno.h>
  20. #endif
  21. /* compile with -Dlocal if your debugger can't find static symbols */
  22. #ifndef CASESENSITIVITYDEFAULT_NO
  23.   #if !defined(unix) && !defined(CASESENSITIVITYDEFAULT_YES)
  24.     #define CASESENSITIVITYDEFAULT_NO
  25.   #endif
  26. #endif
  27. #ifndef UNZ_BUFSIZE
  28.   #define UNZ_BUFSIZE (32768)
  29. #endif
  30. #ifndef UNZ_MAXFILENAMEINZIP
  31.   #define UNZ_MAXFILENAMEINZIP (256)
  32. #endif
  33. #ifndef ALLOC
  34.   #define ALLOC(size) (malloc(size))
  35. #endif
  36. #ifndef TRYFREE
  37.   #define TRYFREE(p) {if (p) free(p);}
  38. #endif
  39. #define SIZECENTRALDIRITEM (0x2e)
  40. #define SIZEZIPLOCALHEADER (0x1e)
  41. /* unz_file_info_interntal contain internal info about a file in zipfile*/
  42. typedef struct unz_file_info_internal_s
  43. {
  44.   DWORD offset_curfile; /* relative offset of local header 4 bytes */
  45. } unz_file_info_internal;
  46. /* file_in_zip_read_info_s contain internal information about a file in zipfile,
  47. when reading and decompress it */
  48. typedef struct
  49. {
  50.   char *read_buffer; /* internal buffer for compressed data */
  51.   z_stream stream; /* zLib stream structure for inflate */
  52.   DWORD pos_in_zipfile; /* position in byte on the zipfile, for fseek*/
  53.   DWORD stream_initialised; /* flag set if stream structure is initialised*/
  54.   DWORD offset_local_extrafield; /* offset of the local extra field */
  55.   DWORD size_local_extrafield; /* size of the local extra field */
  56.   DWORD pos_local_extrafield; /* position in the local extra field in read*/
  57.   DWORD crc32; /* crc32 of all data uncompressed */
  58.   DWORD crc32_wait; /* crc32 we must obtain after decompress all */
  59.   DWORD rest_read_compressed; /* number of byte to be decompressed */
  60.   DWORD rest_read_uncompressed; /*number of byte to be obtained after decomp*/
  61.   zlib_filefunc_def z_filefunc;
  62.   void *filestream; /* io structore of the zipfile */
  63.   DWORD compression_method; /* compression method (0==store) */
  64.   DWORD byte_before_the_zipfile; /* byte before the zipfile, (>0 for sfx)*/
  65.   int raw;
  66. } file_in_zip_read_info_s;
  67. /* unz_s contain internal information about the zipfile
  68.  */
  69. typedef struct
  70. {
  71.   zlib_filefunc_def z_filefunc;
  72.   void *filestream; /* io structore of the zipfile */
  73.   unz_global_info gi; /* public global information */
  74.   DWORD byte_before_the_zipfile; /* byte before the zipfile, (>0 for sfx)*/
  75.   DWORD num_file; /* number of the current file in the zipfile*/
  76.   DWORD pos_in_central_dir; /* pos of the current file in the central dir*/
  77.   DWORD current_file_ok; /* flag about the usability of the current file*/
  78.   DWORD central_pos; /* position of the beginning of the central dir*/
  79.   DWORD size_central_dir; /* size of the central directory  */
  80.   DWORD offset_central_dir; /* offset of start of central directory with respect to the starting disk number */
  81.   unz_file_info cur_file_info; /* public info about the current file in zip*/
  82.   unz_file_info_internal cur_file_info_internal; /* private info about it*/
  83.   file_in_zip_read_info_s *pfile_in_zip_read; /* structure about the current file if we are decompressing it */
  84.   int encrypted;
  85.   #ifndef NOUNCRYPT
  86.     DWORD keys[3]; /* keys defining the pseudo-random sequence */
  87.     const DWORD *pcrc_32_tab;
  88.   #endif
  89. } unz_s;
  90. #ifndef NOUNCRYPT
  91.   #include "crypt.h"
  92. #endif
  93. /* ===========================================================================
  94. Read a byte from a gz_stream; update next_in and avail_in. Return EOF
  95. for end of file.
  96. IN assertion: the stream s has been sucessfully opened for reading.
  97.  */
  98. static int unzlocal_getByte(const zlib_filefunc_def *pzlib_filefunc_def, void *filestream, int *pi)
  99. {
  100.   BYTE c;
  101.   int err = (int)ZREAD(*pzlib_filefunc_def, filestream, &c, 1);
  102.   if (err == 1)
  103.   {
  104.     *pi = (int)c;
  105.     return UNZ_OK;
  106.   }
  107.   else
  108.   {
  109.     if (ZERROR(*pzlib_filefunc_def, filestream))
  110.     {
  111.       return UNZ_ERRNO;
  112.     }
  113.     else
  114.     {
  115.       return UNZ_EOF;
  116.     }
  117.   }
  118. }
  119. /* ===========================================================================
  120. Reads a long in LSB order from the given gz_stream. Sets
  121.  */
  122. static int unzlocal_getShort(const zlib_filefunc_def *pzlib_filefunc_def, void *filestream, DWORD *pX)
  123. {
  124.   DWORD x;
  125.   int i;
  126.   int err;
  127.   err = unzlocal_getByte(pzlib_filefunc_def, filestream, &i);
  128.   x = (DWORD)i;
  129.   if (err == UNZ_OK)
  130.   {
  131.     err = unzlocal_getByte(pzlib_filefunc_def, filestream, &i);
  132.   }
  133.   x += ((DWORD)i) << 8;
  134.   if (err == UNZ_OK)
  135.   {
  136.     *pX = x;
  137.   }
  138.   else
  139.   {
  140.     *pX = 0;
  141.   }
  142.   return err;
  143. }
  144. //-------------------------------------------------------------------------
  145. static int unzlocal_getLong(const zlib_filefunc_def *pzlib_filefunc_def, void *filestream, DWORD *pX)
  146. {
  147.   DWORD x;
  148.   int i;
  149.   int err;
  150.   err = unzlocal_getByte(pzlib_filefunc_def, filestream, &i);
  151.   x = (DWORD)i;
  152.   if (err == UNZ_OK)
  153.   {
  154.     err = unzlocal_getByte(pzlib_filefunc_def, filestream, &i);
  155.   }
  156.   x += ((DWORD)i) << 8;
  157.   if (err == UNZ_OK)
  158.   {
  159.     err = unzlocal_getByte(pzlib_filefunc_def, filestream, &i);
  160.   }
  161.   x += ((DWORD)i) << 16;
  162.   if (err == UNZ_OK)
  163.   {
  164.     err = unzlocal_getByte(pzlib_filefunc_def, filestream, &i);
  165.   }
  166.   x += ((DWORD)i) << 24;
  167.   if (err == UNZ_OK)
  168.   {
  169.     *pX = x;
  170.   }
  171.   else
  172.   {
  173.     *pX = 0;
  174.   }
  175.   return err;
  176. }
  177. //-------------------------------------------------------------------------
  178. #ifdef CASESENSITIVITYDEFAULT_NO
  179.   #define CASESENSITIVITYDEFAULTVALUE 2
  180. #else
  181.   #define CASESENSITIVITYDEFAULTVALUE 1
  182. #endif
  183. #ifndef STRCMPCASENOSENTIVEFUNCTION
  184.   #define STRCMPCASENOSENTIVEFUNCTION strcmpcasenosensitive_internal
  185. #endif
  186. /*
  187. Compare two filename (fileName1,fileName2).
  188. If iCaseSenisivity = 1, comparision is case sensitivity (like strcmp)
  189. If iCaseSenisivity = 2, comparision is not case sensitivity (like strcmpi
  190. or strcasecmp)
  191. If iCaseSenisivity = 0, case sensitivity is defaut of your operating system
  192. (like 1 on Unix, 2 on Windows)
  193. */
  194. /*
  195. extern int ZEXPORT unzStringFileNameCompare(const char *fileName1, const char *fileName2, int iCaseSensitivity)
  196. {
  197.   if (iCaseSensitivity == 0)
  198.   {
  199.     iCaseSensitivity = CASESENSITIVITYDEFAULTVALUE;
  200.   }
  201.   if (iCaseSensitivity == 1)
  202.   {
  203.     return strcmp(fileName1, fileName2);
  204.   }
  205.   return stricmp(fileName1, fileName2);
  206. }
  207. */
  208. //-------------------------------------------------------------------------
  209. #ifndef BUFREADCOMMENT
  210.   #define BUFREADCOMMENT (0x400)
  211. #endif
  212. /*
  213. Locate the Central directory of a zipfile (at the end, just before
  214. the global comment)
  215.  */
  216. static DWORD unzlocal_SearchCentralDir(const zlib_filefunc_def *pzlib_filefunc_def, void *filestream)
  217. {
  218.   BYTE *buf;
  219.   DWORD uSizeFile;
  220.   DWORD uBackRead;
  221.   DWORD uMaxBack = 0xffff; /* maximum size of global comment */
  222.   DWORD uPosFound = 0;
  223.   if (ZSEEK(*pzlib_filefunc_def, filestream, 0, SEEK_END) != 0)
  224.   {
  225.     return 0;
  226.   }
  227.   uSizeFile = ZTELL(*pzlib_filefunc_def, filestream);
  228.   if (uMaxBack > uSizeFile)
  229.   {
  230.     uMaxBack = uSizeFile;
  231.   }
  232.   buf = (BYTE*)ALLOC(BUFREADCOMMENT + 4);
  233.   if (buf == NULL)
  234.   {
  235.     return 0;
  236.   }
  237.   uBackRead = 4;
  238.   while (uBackRead < uMaxBack)
  239.   {
  240.     DWORD uReadSize, uReadPos;
  241.     int i;
  242.     if (uBackRead + BUFREADCOMMENT > uMaxBack)
  243.     {
  244.       uBackRead = uMaxBack;
  245.     }
  246.     else
  247.     {
  248.       uBackRead += BUFREADCOMMENT;
  249.     }
  250.     uReadPos = uSizeFile - uBackRead;
  251.     uReadSize = ((BUFREADCOMMENT + 4) < (uSizeFile - uReadPos)) ? (BUFREADCOMMENT + 4): (uSizeFile - uReadPos);
  252.     if (ZSEEK(*pzlib_filefunc_def, filestream, uReadPos, SEEK_SET) != 0)
  253.     {
  254.       break;
  255.     }
  256.     if (ZREAD(*pzlib_filefunc_def, filestream, buf, uReadSize) != uReadSize)
  257.     {
  258.       break;
  259.     }
  260.     for (i = (int)uReadSize - 3; (i--) > 0;)
  261.     if (((*(buf + i)) == 0x50) && ((*(buf + i + 1)) == 0x4b) && ((*(buf + i + 2)) == 0x05) && ((*(buf + i + 3)) == 0x06))
  262.     {
  263.       uPosFound = uReadPos + i;
  264.       break;
  265.     }
  266.     if (uPosFound != 0)
  267.     {
  268.       break;
  269.     }
  270.   }
  271.   TRYFREE(buf);
  272.   return uPosFound;
  273. }
  274. /*
  275. Open a Zip file. path contain the full pathname (by example,
  276. on a Windows NT computer "c:\test\zlib114.zip" or on an Unix computer
  277. "zlib/zlib114.zip".
  278. If the zipfile cannot be opened (file doesn't exist or in not valid), the
  279. return value is NULL.
  280. Else, the return value is a unzFile Handle, usable with other function
  281. of this unzip package.
  282.  */
  283. extern unzFile ZEXPORT unzOpen2(const char *path, zlib_filefunc_def *pzlib_filefunc_def)
  284. {
  285.   unz_s us;
  286.   unz_s *s;
  287.   DWORD central_pos, uL;
  288.   DWORD number_disk; /* number of the current dist, used for
  289.   spaning ZIP, unsupported, always 0*/
  290.   DWORD number_disk_with_CD; /* number the the disk with central dir, used
  291.   for spaning ZIP, unsupported, always 0*/
  292.   DWORD number_entry_CD; /* total number of entries in
  293.   the central dir
  294.   (same than number_entry on nospan) */
  295.   int err = UNZ_OK;
  296.   if (pzlib_filefunc_def == NULL)
  297.   {
  298.     fill_fopen_filefunc(&us.z_filefunc);
  299.   }
  300.   else
  301.   {
  302.     us.z_filefunc =  *pzlib_filefunc_def;
  303.   }
  304.   us.filestream = (*(us.z_filefunc.zopen_file))(us.z_filefunc.opaque, path, ZLIB_FILEFUNC_MODE_READ | ZLIB_FILEFUNC_MODE_EXISTING);
  305.   if (us.filestream == NULL)
  306.   {
  307.     return NULL;
  308.   }
  309.   central_pos = unzlocal_SearchCentralDir(&us.z_filefunc, us.filestream);
  310.   if (central_pos == 0)
  311.   {
  312.     err = UNZ_ERRNO;
  313.   }
  314.   if (ZSEEK(us.z_filefunc, us.filestream, central_pos, SEEK_SET) != 0)
  315.   {
  316.     err = UNZ_ERRNO;
  317.   }
  318.   /* the signature, already checked */
  319.   if (unzlocal_getLong(&us.z_filefunc, us.filestream, &uL) != UNZ_OK)
  320.   {
  321.     err = UNZ_ERRNO;
  322.   }
  323.   /* number of this disk */
  324.   if (unzlocal_getShort(&us.z_filefunc, us.filestream, &number_disk) != UNZ_OK)
  325.   {
  326.     err = UNZ_ERRNO;
  327.   }
  328.   /* number of the disk with the start of the central directory */
  329.   if (unzlocal_getShort(&us.z_filefunc, us.filestream, &number_disk_with_CD) != UNZ_OK)
  330.   {
  331.     err = UNZ_ERRNO;
  332.   }
  333.   /* total number of entries in the central dir on this disk */
  334.   if (unzlocal_getShort(&us.z_filefunc, us.filestream, &us.gi.number_entry) != UNZ_OK)
  335.   {
  336.     err = UNZ_ERRNO;
  337.   }
  338.   /* total number of entries in the central dir */
  339.   if (unzlocal_getShort(&us.z_filefunc, us.filestream, &number_entry_CD) != UNZ_OK)
  340.   {
  341.     err = UNZ_ERRNO;
  342.   }
  343.   if ((number_entry_CD != us.gi.number_entry) || (number_disk_with_CD != 0) || (number_disk != 0))
  344.   {
  345.     err = UNZ_BADZIPFILE;
  346.   }
  347.   /* size of the central directory */
  348.   if (unzlocal_getLong(&us.z_filefunc, us.filestream, &us.size_central_dir) != UNZ_OK)
  349.   {
  350.     err = UNZ_ERRNO;
  351.   }
  352.   /* offset of start of central directory with respect to the
  353.   starting disk number */
  354.   if (unzlocal_getLong(&us.z_filefunc, us.filestream, &us.offset_central_dir) != UNZ_OK)
  355.   {
  356.     err = UNZ_ERRNO;
  357.   }
  358.   /* zipfile comment length */
  359.   if (unzlocal_getShort(&us.z_filefunc, us.filestream, &us.gi.size_comment) != UNZ_OK)
  360.   {
  361.     err = UNZ_ERRNO;
  362.   }
  363.   if ((central_pos < us.offset_central_dir + us.size_central_dir) && (err == UNZ_OK))
  364.   {
  365.     err = UNZ_BADZIPFILE;
  366.   }
  367.   if (err != UNZ_OK)
  368.   {
  369.     ZCLOSE(us.z_filefunc, us.filestream);
  370.     return NULL;
  371.   }
  372.   us.byte_before_the_zipfile = central_pos - (us.offset_central_dir + us.size_central_dir);
  373.   us.central_pos = central_pos;
  374.   us.pfile_in_zip_read = NULL;
  375.   us.encrypted = 0;
  376.   s = (unz_s*)ALLOC(sizeof(unz_s));
  377.   *s = us;
  378.   unzGoToFirstFile((unzFile)s);
  379.   return (unzFile)s;
  380. }
  381. //-------------------------------------------------------------------------
  382. extern unzFile ZEXPORT unzOpen(const char *path)
  383. {
  384.   return unzOpen2(path, NULL);
  385. }
  386. /*
  387. Close a ZipFile opened with unzipOpen.
  388. If there is files inside the .Zip opened with unzipOpenCurrentFile (see later),
  389. these files MUST be closed with unzipCloseCurrentFile before call unzipClose.
  390. return UNZ_OK if there is no problem. */
  391. extern int ZEXPORT unzClose(unzFile file)
  392. {
  393.   unz_s *s;
  394.   if (file == NULL)
  395.   {
  396.     return UNZ_PARAMERROR;
  397.   }
  398.   s = (unz_s*)file;
  399.   if (s->pfile_in_zip_read != NULL)
  400.   {
  401.     unzCloseCurrentFile(file);
  402.   }
  403.   ZCLOSE(s->z_filefunc, s->filestream);
  404.   TRYFREE(s);
  405.   return UNZ_OK;
  406. }
  407. /*
  408. Write info about the ZipFile in the *pglobal_info structure.
  409. No preparation of the structure is needed
  410. return UNZ_OK if there is no problem. */
  411. /*
  412. extern int ZEXPORT unzGetGlobalInfo(unzFile file, unz_global_info *pglobal_info)
  413. {
  414.   unz_s *s;
  415.   if (file == NULL)
  416.   {
  417.     return UNZ_PARAMERROR;
  418.   }
  419.   s = (unz_s*)file;
  420.   *pglobal_info = s->gi;
  421.   return UNZ_OK;
  422. }
  423. */
  424. /*
  425. Translate date/time from Dos format to tm_unz (readable more easilty)
  426.  */
  427. static void unzlocal_DosDateToTmuDate(DWORD ulDosDate, tm_unz *ptm)
  428. {
  429.   DWORD uDate;
  430.   uDate = (DWORD)(ulDosDate >> 16);
  431.   ptm->tm_mday = (DWORD)(uDate &0x1f);
  432.   ptm->tm_mon = (DWORD)((((uDate) &0x1E0) / 0x20) - 1);
  433.   ptm->tm_year = (DWORD)(((uDate &0x0FE00) / 0x0200) + 1980);
  434.   ptm->tm_hour = (DWORD)((ulDosDate &0xF800) / 0x800);
  435.   ptm->tm_min = (DWORD)((ulDosDate &0x7E0) / 0x20);
  436.   ptm->tm_sec = (DWORD)(2 *(ulDosDate &0x1f));
  437. }
  438. /*
  439. Get Info about the current file in the zipfile, with internal only info
  440.  */
  441. static int unzlocal_GetCurrentFileInfoInternal(unzFile file, unz_file_info *pfile_info, unz_file_info_internal *pfile_info_internal, char *szFileName, DWORD fileNameBufferSize, void *extraField, DWORD extraFieldBufferSize, char *szComment, DWORD commentBufferSize)
  442. {
  443.   unz_s *s;
  444.   unz_file_info file_info;
  445.   unz_file_info_internal file_info_internal;
  446.   int err = UNZ_OK;
  447.   DWORD uMagic;
  448.   long lSeek = 0;
  449.   if (file == NULL)
  450.   {
  451.     return UNZ_PARAMERROR;
  452.   }
  453.   s = (unz_s*)file;
  454.   if (ZSEEK(s->z_filefunc, s->filestream, s->pos_in_central_dir + s->byte_before_the_zipfile, SEEK_SET) != 0)
  455.   {
  456.     err = UNZ_ERRNO;
  457.   }
  458.   /* we check the magic */
  459.   if (err == UNZ_OK)
  460.   {
  461.     if (unzlocal_getLong(&s->z_filefunc, s->filestream, &uMagic) != UNZ_OK)
  462.     {
  463.       err = UNZ_ERRNO;
  464.     }
  465.     else if (uMagic != 0x02014b50)
  466.     {
  467.       err = UNZ_BADZIPFILE;
  468.     }
  469.   }
  470.   if (unzlocal_getShort(&s->z_filefunc, s->filestream, &file_info.version) != UNZ_OK)
  471.   {
  472.     err = UNZ_ERRNO;
  473.   }
  474.   if (unzlocal_getShort(&s->z_filefunc, s->filestream, &file_info.version_needed) != UNZ_OK)
  475.   {
  476.     err = UNZ_ERRNO;
  477.   }
  478.   if (unzlocal_getShort(&s->z_filefunc, s->filestream, &file_info.flag) != UNZ_OK)
  479.   {
  480.     err = UNZ_ERRNO;
  481.   }
  482.   if (unzlocal_getShort(&s->z_filefunc, s->filestream, &file_info.compression_method) != UNZ_OK)
  483.   {
  484.     err = UNZ_ERRNO;
  485.   }
  486.   if (unzlocal_getLong(&s->z_filefunc, s->filestream, &file_info.dosDate) != UNZ_OK)
  487.   {
  488.     err = UNZ_ERRNO;
  489.   }
  490.   unzlocal_DosDateToTmuDate(file_info.dosDate, &file_info.tmu_date);
  491.   if (unzlocal_getLong(&s->z_filefunc, s->filestream, &file_info.crc) != UNZ_OK)
  492.   {
  493.     err = UNZ_ERRNO;
  494.   }
  495.   if (unzlocal_getLong(&s->z_filefunc, s->filestream, &file_info.compressed_size) != UNZ_OK)
  496.   {
  497.     err = UNZ_ERRNO;
  498.   }
  499.   if (unzlocal_getLong(&s->z_filefunc, s->filestream, &file_info.uncompressed_size) != UNZ_OK)
  500.   {
  501.     err = UNZ_ERRNO;
  502.   }
  503.   if (unzlocal_getShort(&s->z_filefunc, s->filestream, &file_info.size_filename) != UNZ_OK)
  504.   {
  505.     err = UNZ_ERRNO;
  506.   }
  507.   if (unzlocal_getShort(&s->z_filefunc, s->filestream, &file_info.size_file_extra) != UNZ_OK)
  508.   {
  509.     err = UNZ_ERRNO;
  510.   }
  511.   if (unzlocal_getShort(&s->z_filefunc, s->filestream, &file_info.size_file_comment) != UNZ_OK)
  512.   {
  513.     err = UNZ_ERRNO;
  514.   }
  515.   if (unzlocal_getShort(&s->z_filefunc, s->filestream, &file_info.disk_num_start) != UNZ_OK)
  516.   {
  517.     err = UNZ_ERRNO;
  518.   }
  519.   if (unzlocal_getShort(&s->z_filefunc, s->filestream, &file_info.internal_fa) != UNZ_OK)
  520.   {
  521.     err = UNZ_ERRNO;
  522.   }
  523.   if (unzlocal_getLong(&s->z_filefunc, s->filestream, &file_info.external_fa) != UNZ_OK)
  524.   {
  525.     err = UNZ_ERRNO;
  526.   }
  527.   if (unzlocal_getLong(&s->z_filefunc, s->filestream, &file_info_internal.offset_curfile) != UNZ_OK)
  528.   {
  529.     err = UNZ_ERRNO;
  530.   }
  531.   lSeek += file_info.size_filename;
  532.   if ((err == UNZ_OK) && (szFileName != NULL))
  533.   {
  534.     DWORD uSizeRead;
  535.     if (file_info.size_filename < fileNameBufferSize)
  536.     {
  537.       *(szFileName + file_info.size_filename) = '';
  538.       uSizeRead = file_info.size_filename;
  539.     }
  540.     else
  541.     {
  542.       uSizeRead = fileNameBufferSize;
  543.     }
  544.     if ((file_info.size_filename > 0) && (fileNameBufferSize > 0))
  545.     if (ZREAD(s->z_filefunc, s->filestream, szFileName, uSizeRead) != uSizeRead)
  546.     {
  547.       err = UNZ_ERRNO;
  548.     }
  549.     lSeek -= uSizeRead;
  550.   }
  551.   if ((err == UNZ_OK) && (extraField != NULL))
  552.   {
  553.     DWORD uSizeRead;
  554.     if (file_info.size_file_extra < extraFieldBufferSize)
  555.     {
  556.       uSizeRead = file_info.size_file_extra;
  557.     }
  558.     else
  559.     {
  560.       uSizeRead = extraFieldBufferSize;
  561.     }
  562.     if (lSeek != 0)
  563.     {
  564.       if (ZSEEK(s->z_filefunc, s->filestream, lSeek, SEEK_CUR) == 0)
  565.       {
  566.         lSeek = 0;
  567.       }
  568.       else
  569.       {
  570.         err = UNZ_ERRNO;
  571.       }
  572.     }
  573.     if ((file_info.size_file_extra > 0) && (extraFieldBufferSize > 0))
  574.     {
  575.       if (ZREAD(s->z_filefunc, s->filestream, extraField, uSizeRead) != uSizeRead)
  576.       {
  577.         err = UNZ_ERRNO;
  578.       }
  579.     }
  580.     lSeek += file_info.size_file_extra - uSizeRead;
  581.   }
  582.   else
  583.   {
  584.     lSeek += file_info.size_file_extra;
  585.   }
  586.   if ((err == UNZ_OK) && (szComment != NULL))
  587.   {
  588.     DWORD uSizeRead;
  589.     if (file_info.size_file_comment < commentBufferSize)
  590.     {
  591.       *(szComment + file_info.size_file_comment) = '';
  592.       uSizeRead = file_info.size_file_comment;
  593.     }
  594.     else
  595.     {
  596.       uSizeRead = commentBufferSize;
  597.     }
  598.     if (lSeek != 0)
  599.     {
  600.       if (ZSEEK(s->z_filefunc, s->filestream, lSeek, SEEK_CUR) == 0)
  601.       {
  602.         lSeek = 0;
  603.       }
  604.       else
  605.       {
  606.         err = UNZ_ERRNO;
  607.       }
  608.     }
  609.     if ((file_info.size_file_comment > 0) && (commentBufferSize > 0))
  610.     {
  611.       if (ZREAD(s->z_filefunc, s->filestream, szComment, uSizeRead) != uSizeRead)
  612.       {
  613.         err = UNZ_ERRNO;
  614.       }
  615.     }
  616.     lSeek += file_info.size_file_comment - uSizeRead;
  617.   }
  618.   else
  619.   {
  620.     lSeek += file_info.size_file_comment;
  621.   }
  622.   if ((err == UNZ_OK) && (pfile_info != NULL))
  623.   {
  624.     *pfile_info = file_info;
  625.   }
  626.   if ((err == UNZ_OK) && (pfile_info_internal != NULL))
  627.   {
  628.     *pfile_info_internal = file_info_internal;
  629.   }
  630.   return err;
  631. }
  632. /*
  633. Write info about the ZipFile in the *pglobal_info structure.
  634. No preparation of the structure is needed
  635. return UNZ_OK if there is no problem.
  636.  */
  637. extern int ZEXPORT unzGetCurrentFileInfo(unzFile file, unz_file_info *pfile_info, char *szFileName, DWORD fileNameBufferSize, void *extraField, DWORD extraFieldBufferSize, char *szComment, DWORD commentBufferSize)
  638. {
  639.   return unzlocal_GetCurrentFileInfoInternal(file, pfile_info, NULL, szFileName, fileNameBufferSize, extraField, extraFieldBufferSize, szComment, commentBufferSize);
  640. }
  641. /*
  642. Set the current file of the zipfile to the first file.
  643. return UNZ_OK if there is no problem
  644.  */
  645. extern int ZEXPORT unzGoToFirstFile(unzFile file)
  646. {
  647.   int err = UNZ_OK;
  648.   unz_s *s;
  649.   if (file == NULL)
  650.   {
  651.     return UNZ_PARAMERROR;
  652.   }
  653.   s = (unz_s*)file;
  654.   s->pos_in_central_dir = s->offset_central_dir;
  655.   s->num_file = 0;
  656.   err = unzlocal_GetCurrentFileInfoInternal(file, &s->cur_file_info, &s->cur_file_info_internal, NULL, 0, NULL, 0, NULL, 0);
  657.   s->current_file_ok = (err == UNZ_OK);
  658.   return err;
  659. }
  660. /*
  661. Set the current file of the zipfile to the next file.
  662. return UNZ_OK if there is no problem
  663. return UNZ_END_OF_LIST_OF_FILE if the actual file was the latest.
  664.  */
  665. extern int ZEXPORT unzGoToNextFile(unzFile file)
  666. {
  667.   unz_s *s;
  668.   int err;
  669.   if (file == NULL)
  670.   {
  671.     return UNZ_PARAMERROR;
  672.   }
  673.   s = (unz_s*)file;
  674.   if (!s->current_file_ok)
  675.   {
  676.     return UNZ_END_OF_LIST_OF_FILE;
  677.   }
  678.   if (s->gi.number_entry != 0xffff)
  679.   {
  680.     /* 2^16 files overflow hack */
  681.     if (s->num_file + 1 == s->gi.number_entry)
  682.     {
  683.       return UNZ_END_OF_LIST_OF_FILE;
  684.     }
  685.   }
  686.   s->pos_in_central_dir += SIZECENTRALDIRITEM + s->cur_file_info.size_filename + s->cur_file_info.size_file_extra + s->cur_file_info.size_file_comment;
  687.   s->num_file++;
  688.   err = unzlocal_GetCurrentFileInfoInternal(file, &s->cur_file_info, &s->cur_file_info_internal, NULL, 0, NULL, 0, NULL, 0);
  689.   s->current_file_ok = (err == UNZ_OK);
  690.   return err;
  691. }
  692. extern int ZEXPORT unzGetCurrentFileID(unzFile file, DWORD *file_num, DWORD *pos_in_central_dir)
  693. {
  694.   unz_s *s;
  695.   if (file == NULL)
  696.   {
  697.     return UNZ_PARAMERROR;
  698.   }
  699.   s = (unz_s*)file;
  700.   *file_num = s->num_file;
  701.   *pos_in_central_dir = s->pos_in_central_dir;
  702.   return UNZ_OK;
  703. }
  704. extern int ZEXPORT unzGoToFileID(unzFile file, DWORD file_num, DWORD pos_in_central_dir)
  705. {
  706.   unz_s *s;
  707.   int err;
  708.   if (file == NULL)
  709.   {
  710.     return UNZ_PARAMERROR;
  711.   }
  712.   
  713.   s = (unz_s*)file;
  714.   s->num_file = file_num;
  715.   s->pos_in_central_dir = pos_in_central_dir;
  716.   err = unzlocal_GetCurrentFileInfoInternal(file, &s->cur_file_info, &s->cur_file_info_internal, NULL, 0, NULL, 0, NULL, 0);
  717.   s->current_file_ok = (err == UNZ_OK);
  718.   return err;
  719. }
  720. /*
  721. Try locate the file szFileName in the zipfile.
  722. For the iCaseSensitivity signification, see unzipStringFileNameCompare
  723. return value :
  724. UNZ_OK if the file is found. It becomes the current file.
  725. UNZ_END_OF_LIST_OF_FILE if the file is not found
  726.  */
  727. //-------------------------------------------------------------------------
  728. /*
  729. // Unzip Helper Functions - should be here?
  730. ///////////////////////////////////////////
  731.  */
  732. /*
  733. Read the local header of the current zipfile
  734. Check the coherency of the local header and info in the end of central
  735. directory about this file
  736. store in *piSizeVar the size of extra info in local header
  737. (filename and size of extra field data)
  738.  */
  739. static int unzlocal_CheckCurrentFileCoherencyHeader(unz_s *s, DWORD *piSizeVar, DWORD *poffset_local_extrafield, DWORD *psize_local_extrafield)
  740. {
  741.   DWORD uMagic, uData, uFlags;
  742.   DWORD size_filename;
  743.   DWORD size_extra_field;
  744.   int err = UNZ_OK;
  745.   *piSizeVar = 0;
  746.   *poffset_local_extrafield = 0;
  747.   *psize_local_extrafield = 0;
  748.   if (ZSEEK(s->z_filefunc, s->filestream, s->cur_file_info_internal.offset_curfile + s->byte_before_the_zipfile, SEEK_SET) != 0)
  749.   {
  750.     return UNZ_ERRNO;
  751.   }
  752.   if (err == UNZ_OK)
  753.   {
  754.     if (unzlocal_getLong(&s->z_filefunc, s->filestream, &uMagic) != UNZ_OK)
  755.     {
  756.       err = UNZ_ERRNO;
  757.     }
  758.     else if (uMagic != 0x04034b50)
  759.     {
  760.       err = UNZ_BADZIPFILE;
  761.     }
  762.   }
  763.   if (unzlocal_getShort(&s->z_filefunc, s->filestream, &uData) != UNZ_OK)
  764.   {
  765.     err = UNZ_ERRNO;
  766.   }
  767.   if (unzlocal_getShort(&s->z_filefunc, s->filestream, &uFlags) != UNZ_OK)
  768.   {
  769.     err = UNZ_ERRNO;
  770.   }
  771.   if (unzlocal_getShort(&s->z_filefunc, s->filestream, &uData) != UNZ_OK)
  772.   {
  773.     err = UNZ_ERRNO;
  774.   }
  775.   else if ((err == UNZ_OK) && (uData != s->cur_file_info.compression_method))
  776.   {
  777.     err = UNZ_BADZIPFILE;
  778.   }
  779.   if ((err == UNZ_OK) && (s->cur_file_info.compression_method != 0) && (s->cur_file_info.compression_method != Z_DEFLATED))
  780.   {
  781.     err = UNZ_BADZIPFILE;
  782.   }
  783.   if (unzlocal_getLong(&s->z_filefunc, s->filestream, &uData) != UNZ_OK)
  784.   // date/time 
  785.   {
  786.     err = UNZ_ERRNO;
  787.   }
  788.   if (unzlocal_getLong(&s->z_filefunc, s->filestream, &uData) != UNZ_OK)
  789.   // crc 
  790.   {
  791.     err = UNZ_ERRNO;
  792.   }
  793.   else if ((err == UNZ_OK) && (uData != s->cur_file_info.crc) && ((uFlags &8) == 0))
  794.   {
  795.     err = UNZ_BADZIPFILE;
  796.   }
  797.   if (unzlocal_getLong(&s->z_filefunc, s->filestream, &uData) != UNZ_OK)
  798.   // size compr 
  799.   {
  800.     err = UNZ_ERRNO;
  801.   }
  802.   else if ((err == UNZ_OK) && (uData != s->cur_file_info.compressed_size) && ((uFlags &8) == 0))
  803.   {
  804.     err = UNZ_BADZIPFILE;
  805.   }
  806.   if (unzlocal_getLong(&s->z_filefunc, s->filestream, &uData) != UNZ_OK)
  807.   // size uncompr 
  808.   {
  809.     err = UNZ_ERRNO;
  810.   }
  811.   else if ((err == UNZ_OK) && (uData != s->cur_file_info.uncompressed_size) && ((uFlags &8) == 0))
  812.   {
  813.     err = UNZ_BADZIPFILE;
  814.   }
  815.   if (unzlocal_getShort(&s->z_filefunc, s->filestream, &size_filename) != UNZ_OK)
  816.   {
  817.     err = UNZ_ERRNO;
  818.   }
  819.   else if ((err == UNZ_OK) && (size_filename != s->cur_file_info.size_filename))
  820.   {
  821.     err = UNZ_BADZIPFILE;
  822.   }
  823.   *piSizeVar += (DWORD)size_filename;
  824.   if (unzlocal_getShort(&s->z_filefunc, s->filestream, &size_extra_field) != UNZ_OK)
  825.   {
  826.     err = UNZ_ERRNO;
  827.   }
  828.   *poffset_local_extrafield = s->cur_file_info_internal.offset_curfile + SIZEZIPLOCALHEADER + size_filename;
  829.   *psize_local_extrafield = (DWORD)size_extra_field;
  830.   *piSizeVar += (DWORD)size_extra_field;
  831.   return err;
  832. }
  833. /*
  834. Open for reading data the current file in the zipfile.
  835. If there is no error and the file is opened, the return value is UNZ_OK.
  836.  */
  837. extern int ZEXPORT unzOpenCurrentFile3(unzFile file, int *method, int *level, int raw, const char *password)
  838. {
  839.   int err = UNZ_OK;
  840.   DWORD iSizeVar;
  841.   unz_s *s;
  842.   file_in_zip_read_info_s *pfile_in_zip_read_info;
  843.   DWORD offset_local_extrafield; // offset of the local extra field 
  844.   DWORD size_local_extrafield; // size of the local extra field 
  845.   #ifndef NOUNCRYPT
  846.     char source[12];
  847.   #else
  848.     if (password != NULL)
  849.     {
  850.       return UNZ_PARAMERROR;
  851.     }
  852.   #endif
  853.   if (file == NULL)
  854.   {
  855.     return UNZ_PARAMERROR;
  856.   }
  857.   s = (unz_s*)file;
  858.   if (!s->current_file_ok)
  859.   {
  860.     return UNZ_PARAMERROR;
  861.   }
  862.   if (s->pfile_in_zip_read != NULL)
  863.   {
  864.     unzCloseCurrentFile(file);
  865.   }
  866.   if (unzlocal_CheckCurrentFileCoherencyHeader(s, &iSizeVar, &offset_local_extrafield, &size_local_extrafield) != UNZ_OK)
  867.   {
  868.     return UNZ_BADZIPFILE;
  869.   }
  870.   pfile_in_zip_read_info = (file_in_zip_read_info_s*)ALLOC(sizeof(file_in_zip_read_info_s));
  871.   if (pfile_in_zip_read_info == NULL)
  872.   {
  873.     return UNZ_INTERNALERROR;
  874.   }
  875.   pfile_in_zip_read_info->read_buffer = (char*)ALLOC(UNZ_BUFSIZE);
  876.   pfile_in_zip_read_info->offset_local_extrafield = offset_local_extrafield;
  877.   pfile_in_zip_read_info->size_local_extrafield = size_local_extrafield;
  878.   pfile_in_zip_read_info->pos_local_extrafield = 0;
  879.   pfile_in_zip_read_info->raw = raw;
  880.   if (pfile_in_zip_read_info->read_buffer == NULL)
  881.   {
  882.     TRYFREE(pfile_in_zip_read_info);
  883.     return UNZ_INTERNALERROR;
  884.   }
  885.   pfile_in_zip_read_info->stream_initialised = 0;
  886.   if (method != NULL)
  887.   {
  888.     *method = (int)s->cur_file_info.compression_method;
  889.   }
  890.   if (level != NULL)
  891.   {
  892.     *level = 6;
  893.     switch (s->cur_file_info.flag &0x06)
  894.     {
  895.       case 6:
  896.         *level = 1;
  897.         break;
  898.       case 4:
  899.         *level = 2;
  900.         break;
  901.       case 2:
  902.         *level = 9;
  903.         break;
  904.     }
  905.   }
  906.   if ((s->cur_file_info.compression_method != 0) && (s->cur_file_info.compression_method != Z_DEFLATED))
  907.   {
  908.     err = UNZ_BADZIPFILE;
  909.   }
  910.   pfile_in_zip_read_info->crc32_wait = s->cur_file_info.crc;
  911.   pfile_in_zip_read_info->crc32 = 0;
  912.   pfile_in_zip_read_info->compression_method = s->cur_file_info.compression_method;
  913.   pfile_in_zip_read_info->filestream = s->filestream;
  914.   pfile_in_zip_read_info->z_filefunc = s->z_filefunc;
  915.   pfile_in_zip_read_info->byte_before_the_zipfile = s->byte_before_the_zipfile;
  916.   pfile_in_zip_read_info->stream.total_out = 0;
  917.   if ((s->cur_file_info.compression_method == Z_DEFLATED) && (!raw))
  918.   {
  919.     pfile_in_zip_read_info->stream.zalloc = (alloc_func)0;
  920.     pfile_in_zip_read_info->stream.zfree = (free_func)0;
  921.     pfile_in_zip_read_info->stream.opaque = (void*)0;
  922.     pfile_in_zip_read_info->stream.next_in = (void*)0;
  923.     pfile_in_zip_read_info->stream.avail_in = 0;
  924.     err = inflateInit2(&pfile_in_zip_read_info->stream,  - MAX_WBITS);
  925.     if (err == Z_OK)
  926.     {
  927.       pfile_in_zip_read_info->stream_initialised = 1;
  928.     }
  929.     else
  930.     {
  931.       return err;
  932.     }
  933.     /* windowBits is passed < 0 to tell that there is no zlib header.
  934.      * Note that in this case inflate *requires* an extra "dummy" byte
  935.      * after the compressed stream in order to complete decompression and
  936.      * return Z_STREAM_END.
  937.      * In unzip, i don't wait absolutely Z_STREAM_END because I known the
  938.      * size of both compressed and uncompressed data
  939.      */
  940.   }
  941.   pfile_in_zip_read_info->rest_read_compressed = s->cur_file_info.compressed_size;
  942.   pfile_in_zip_read_info->rest_read_uncompressed = s->cur_file_info.uncompressed_size;
  943.   pfile_in_zip_read_info->pos_in_zipfile = s->cur_file_info_internal.offset_curfile + SIZEZIPLOCALHEADER + iSizeVar;
  944.   pfile_in_zip_read_info->stream.avail_in = (DWORD)0;
  945.   s->pfile_in_zip_read = pfile_in_zip_read_info;
  946.   #ifndef NOUNCRYPT
  947.     if (password != NULL)
  948.     {
  949.       int i;
  950.       s->pcrc_32_tab = get_crc_table();
  951.       init_keys(password, s->keys, s->pcrc_32_tab);
  952.       if (ZSEEK(s->z_filefunc, s->filestream, s->pfile_in_zip_read->pos_in_zipfile + s->pfile_in_zip_read->byte_before_the_zipfile, SEEK_SET) != 0)
  953.       {
  954.         return UNZ_INTERNALERROR;
  955.       }
  956.       if (ZREAD(s->z_filefunc, s->filestream, source, 12) < 12)
  957.       {
  958.         return UNZ_INTERNALERROR;
  959.       }
  960.       for (i = 0; i < 12; i++)
  961.       {
  962.         zdecode(s->keys, s->pcrc_32_tab, source[i]);
  963.       }
  964.       s->pfile_in_zip_read->pos_in_zipfile += 12;
  965.       s->encrypted = 1;
  966.     }
  967.   #endif
  968.   return UNZ_OK;
  969. }
  970. //-------------------------------------------------------------------------
  971. extern int ZEXPORT unzOpenCurrentFile(unzFile file)
  972. {
  973.   return unzOpenCurrentFile3(file, NULL, NULL, 0, NULL);
  974. }
  975. //-------------------------------------------------------------------------
  976. extern int ZEXPORT unzOpenCurrentFilePassword(unzFile file, const char *password)
  977. {
  978.   return unzOpenCurrentFile3(file, NULL, NULL, 0, password);
  979. }
  980. //-------------------------------------------------------------------------
  981. /*
  982. Read bytes from the current file.
  983. buf contain buffer where data must be copied
  984. len the size of buf.
  985. return the number of byte copied if somes bytes are copied
  986. return 0 if the end of file was reached
  987. return <0 with error code if there is an error
  988. (UNZ_ERRNO for IO error, or zLib error for uncompress error)
  989.  */
  990. extern int ZEXPORT unzReadCurrentFile(unzFile file, void *buf, DWORD len)
  991. {
  992.   int err = UNZ_OK;
  993.   DWORD iRead = 0;
  994.   unz_s *s;
  995.   file_in_zip_read_info_s *pfile_in_zip_read_info;
  996.   if (file == NULL)
  997.   {
  998.     return UNZ_PARAMERROR;
  999.   }
  1000.   s = (unz_s*)file;
  1001.   pfile_in_zip_read_info = s->pfile_in_zip_read;
  1002.   if (pfile_in_zip_read_info == NULL)
  1003.   {
  1004.     return UNZ_PARAMERROR;
  1005.   }
  1006.   if ((pfile_in_zip_read_info->read_buffer == NULL))
  1007.   {
  1008.     return UNZ_END_OF_LIST_OF_FILE;
  1009.   }
  1010.   if (len == 0)
  1011.   {
  1012.     return 0;
  1013.   }
  1014.   pfile_in_zip_read_info->stream.next_out = (BYTE*)buf;
  1015.   pfile_in_zip_read_info->stream.avail_out = (DWORD)len;
  1016.   if (len > pfile_in_zip_read_info->rest_read_uncompressed)
  1017.   {
  1018.     pfile_in_zip_read_info->stream.avail_out = (DWORD)pfile_in_zip_read_info->rest_read_uncompressed;
  1019.   }
  1020.   while (pfile_in_zip_read_info->stream.avail_out > 0)
  1021.   {
  1022.     if ((pfile_in_zip_read_info->stream.avail_in == 0) && (pfile_in_zip_read_info->rest_read_compressed > 0))
  1023.     {
  1024.       DWORD uReadThis = UNZ_BUFSIZE;
  1025.       if (pfile_in_zip_read_info->rest_read_compressed < uReadThis)
  1026.       {
  1027.         uReadThis = (DWORD)pfile_in_zip_read_info->rest_read_compressed;
  1028.       }
  1029.       if (uReadThis == 0)
  1030.       {
  1031.         return UNZ_EOF;
  1032.       }
  1033.       if (ZSEEK(pfile_in_zip_read_info->z_filefunc, pfile_in_zip_read_info->filestream, pfile_in_zip_read_info->pos_in_zipfile + pfile_in_zip_read_info->byte_before_the_zipfile, SEEK_SET) != 0)
  1034.       {
  1035.         return UNZ_ERRNO;
  1036.       }
  1037.       if (ZREAD(pfile_in_zip_read_info->z_filefunc, pfile_in_zip_read_info->filestream, pfile_in_zip_read_info->read_buffer, uReadThis) != uReadThis)
  1038.       {
  1039.         return UNZ_ERRNO;
  1040.       }
  1041.       #ifndef NOUNCRYPT
  1042.         if (s->encrypted)
  1043.         {
  1044.           DWORD i;
  1045.           for (i = 0; i < uReadThis; i++)
  1046.           {
  1047.             pfile_in_zip_read_info->read_buffer[i] = zdecode(s->keys, s->pcrc_32_tab, pfile_in_zip_read_info->read_buffer[i]);
  1048.           }
  1049.         }
  1050.       #endif
  1051.       pfile_in_zip_read_info->pos_in_zipfile += uReadThis;
  1052.       pfile_in_zip_read_info->rest_read_compressed -= uReadThis;
  1053.       pfile_in_zip_read_info->stream.next_in = (BYTE*)pfile_in_zip_read_info->read_buffer;
  1054.       pfile_in_zip_read_info->stream.avail_in = (DWORD)uReadThis;
  1055.     }
  1056.     if ((pfile_in_zip_read_info->compression_method == 0) || (pfile_in_zip_read_info->raw))
  1057.     {
  1058.       DWORD uDoCopy, i;
  1059.       if ((pfile_in_zip_read_info->stream.avail_in == 0) && (pfile_in_zip_read_info->rest_read_compressed == 0))
  1060.       {
  1061.         return (iRead == 0) ? UNZ_EOF : iRead;
  1062.       }
  1063.       if (pfile_in_zip_read_info->stream.avail_out < pfile_in_zip_read_info->stream.avail_in)
  1064.       {
  1065.         uDoCopy = pfile_in_zip_read_info->stream.avail_out;
  1066.       }
  1067.       else
  1068.       {
  1069.         uDoCopy = pfile_in_zip_read_info->stream.avail_in;
  1070.       }
  1071.       for (i = 0; i < uDoCopy; i++)
  1072.       {
  1073.         *(pfile_in_zip_read_info->stream.next_out + i) = *(pfile_in_zip_read_info->stream.next_in + i);
  1074.       }
  1075.       pfile_in_zip_read_info->crc32 = crc32(pfile_in_zip_read_info->crc32, pfile_in_zip_read_info->stream.next_out, uDoCopy);
  1076.       pfile_in_zip_read_info->rest_read_uncompressed -= uDoCopy;
  1077.       pfile_in_zip_read_info->stream.avail_in -= uDoCopy;
  1078.       pfile_in_zip_read_info->stream.avail_out -= uDoCopy;
  1079.       pfile_in_zip_read_info->stream.next_out += uDoCopy;
  1080.       pfile_in_zip_read_info->stream.next_in += uDoCopy;
  1081.       pfile_in_zip_read_info->stream.total_out += uDoCopy;
  1082.       iRead += uDoCopy;
  1083.     }
  1084.     else
  1085.     {
  1086.       DWORD uTotalOutBefore, uTotalOutAfter;
  1087.       const BYTE *bufBefore;
  1088.       DWORD uOutThis;
  1089.       int flush = Z_SYNC_FLUSH;
  1090.       uTotalOutBefore = pfile_in_zip_read_info->stream.total_out;
  1091.       bufBefore = pfile_in_zip_read_info->stream.next_out;
  1092.       err = inflate(&pfile_in_zip_read_info->stream, flush);
  1093.       uTotalOutAfter = pfile_in_zip_read_info->stream.total_out;
  1094.       uOutThis = uTotalOutAfter - uTotalOutBefore;
  1095.       pfile_in_zip_read_info->crc32 = crc32(pfile_in_zip_read_info->crc32, bufBefore, (DWORD)(uOutThis));
  1096.       pfile_in_zip_read_info->rest_read_uncompressed -= uOutThis;
  1097.       iRead += (DWORD)(uTotalOutAfter - uTotalOutBefore);
  1098.       if (err == Z_STREAM_END)
  1099.       {
  1100.         return (iRead == 0) ? UNZ_EOF : iRead;
  1101.       }
  1102.       if (err != Z_OK)
  1103.       {
  1104.         break;
  1105.       }
  1106.     }
  1107.   }
  1108.   if (err == Z_OK)
  1109.   {
  1110.     return iRead;
  1111.   }
  1112.   return err;
  1113. }
  1114. /*
  1115. Give the current position in uncompressed data
  1116.  */
  1117. /*
  1118. extern z_off_t ZEXPORT unztell(unzFile file)
  1119. {
  1120.   unz_s *s;
  1121.   file_in_zip_read_info_s *pfile_in_zip_read_info;
  1122.   if (file == NULL)
  1123.   {
  1124.     return UNZ_PARAMERROR;
  1125.   }
  1126.   s = (unz_s*)file;
  1127.   pfile_in_zip_read_info = s->pfile_in_zip_read;
  1128.   if (pfile_in_zip_read_info == NULL)
  1129.   {
  1130.     return UNZ_PARAMERROR;
  1131.   }
  1132.   return (z_off_t)pfile_in_zip_read_info->stream.total_out;
  1133. }
  1134. */
  1135. /*
  1136. return 1 if the end of file was reached, 0 elsewhere
  1137.  */
  1138. /*
  1139. extern int ZEXPORT unzeof(unzFile file)
  1140. {
  1141.   unz_s *s;
  1142.   file_in_zip_read_info_s *pfile_in_zip_read_info;
  1143.   if (file == NULL)
  1144.   {
  1145.     return UNZ_PARAMERROR;
  1146.   }
  1147.   s = (unz_s*)file;
  1148.   pfile_in_zip_read_info = s->pfile_in_zip_read;
  1149.   if (pfile_in_zip_read_info == NULL)
  1150.   {
  1151.     return UNZ_PARAMERROR;
  1152.   }
  1153.   if (pfile_in_zip_read_info->rest_read_uncompressed == 0)
  1154.   {
  1155.     return 1;
  1156.   }
  1157.   else
  1158.   {
  1159.     return 0;
  1160.   }
  1161. }
  1162. */
  1163. /*
  1164. Read extra field from the current file (opened by unzOpenCurrentFile)
  1165. This is the local-header version of the extra field (sometimes, there is
  1166. more info in the local-header version than in the central-header)
  1167. if buf==NULL, it return the size of the local extra field that can be read
  1168. if buf!=NULL, len is the size of the buffer, the extra header is copied in
  1169. buf.
  1170. the return value is the number of bytes copied in buf, or (if <0)
  1171. the error code
  1172.  */
  1173. /*
  1174. extern int ZEXPORT unzGetLocalExtrafield(unzFile file, void *buf, DWORD len)
  1175. {
  1176.   unz_s *s;
  1177.   file_in_zip_read_info_s *pfile_in_zip_read_info;
  1178.   DWORD read_now;
  1179.   DWORD size_to_read;
  1180.   if (file == NULL)
  1181.   {
  1182.     return UNZ_PARAMERROR;
  1183.   }
  1184.   s = (unz_s*)file;
  1185.   pfile_in_zip_read_info = s->pfile_in_zip_read;
  1186.   if (pfile_in_zip_read_info == NULL)
  1187.   {
  1188.     return UNZ_PARAMERROR;
  1189.   }
  1190.   size_to_read = (pfile_in_zip_read_info->size_local_extrafield - pfile_in_zip_read_info->pos_local_extrafield);
  1191.   if (buf == NULL)
  1192.   {
  1193.     return (int)size_to_read;
  1194.   }
  1195.   if (len > size_to_read)
  1196.   {
  1197.     read_now = (DWORD)size_to_read;
  1198.   }
  1199.   else
  1200.   {
  1201.     read_now = (DWORD)len;
  1202.   }
  1203.   if (read_now == 0)
  1204.   {
  1205.     return 0;
  1206.   }
  1207.   if (ZSEEK(pfile_in_zip_read_info->z_filefunc, pfile_in_zip_read_info->filestream, pfile_in_zip_read_info->offset_local_extrafield + pfile_in_zip_read_info->pos_local_extrafield, SEEK_SET) != 0)
  1208.   {
  1209.     return UNZ_ERRNO;
  1210.   }
  1211.   if (ZREAD(pfile_in_zip_read_info->z_filefunc, pfile_in_zip_read_info->filestream, buf, read_now) != read_now)
  1212.   {
  1213.     return UNZ_ERRNO;
  1214.   }
  1215.   return (int)read_now;
  1216. }
  1217. */
  1218. /*
  1219. Close the file in zip opened with unzipOpenCurrentFile
  1220. Return UNZ_CRCERROR if all the file was read but the CRC is not good
  1221.  */
  1222. extern int ZEXPORT unzCloseCurrentFile(unzFile file)
  1223. {
  1224.   int err = UNZ_OK;
  1225.   unz_s *s;
  1226.   file_in_zip_read_info_s *pfile_in_zip_read_info;
  1227.   if (file == NULL)
  1228.   {
  1229.     return UNZ_PARAMERROR;
  1230.   }
  1231.   s = (unz_s*)file;
  1232.   pfile_in_zip_read_info = s->pfile_in_zip_read;
  1233.   if (pfile_in_zip_read_info == NULL)
  1234.   {
  1235.     return UNZ_PARAMERROR;
  1236.   }
  1237.   if ((pfile_in_zip_read_info->rest_read_uncompressed == 0) && (!pfile_in_zip_read_info->raw))
  1238.   {
  1239.     if (pfile_in_zip_read_info->crc32 != pfile_in_zip_read_info->crc32_wait)
  1240.     {
  1241.       err = UNZ_CRCERROR;
  1242.     }
  1243.   }
  1244.   TRYFREE(pfile_in_zip_read_info->read_buffer);
  1245.   pfile_in_zip_read_info->read_buffer = NULL;
  1246.   if (pfile_in_zip_read_info->stream_initialised)
  1247.   {
  1248.     inflateEnd(&pfile_in_zip_read_info->stream);
  1249.   }
  1250.   pfile_in_zip_read_info->stream_initialised = 0;
  1251.   TRYFREE(pfile_in_zip_read_info);
  1252.   s->pfile_in_zip_read = NULL;
  1253.   return err;
  1254. }
  1255. /*
  1256. Get the global comment string of the ZipFile, in the szComment buffer.
  1257. uSizeBuf is the size of the szComment buffer.
  1258. return the number of byte copied or an error code <0
  1259.  */
  1260. /*
  1261. extern int ZEXPORT unzGetGlobalComment(unzFile file, char *szComment, DWORD uSizeBuf)
  1262. {
  1263.   //int err = UNZ_OK;
  1264.   unz_s *s;
  1265.   DWORD uReadThis;
  1266.   if (file == NULL)
  1267.   {
  1268.     return UNZ_PARAMERROR;
  1269.   }
  1270.   s = (unz_s*)file;
  1271.   uReadThis = uSizeBuf;
  1272.   if (uReadThis > s->gi.size_comment)
  1273.   {
  1274.     uReadThis = s->gi.size_comment;
  1275.   }
  1276.   if (ZSEEK(s->z_filefunc, s->filestream, s->central_pos + 22, SEEK_SET) != 0)
  1277.   {
  1278.     return UNZ_ERRNO;
  1279.   }
  1280.   if (uReadThis > 0)
  1281.   {
  1282.     *szComment = '';
  1283.     if (ZREAD(s->z_filefunc, s->filestream, szComment, uReadThis) != uReadThis)
  1284.     {
  1285.       return UNZ_ERRNO;
  1286.     }
  1287.   }
  1288.   if ((szComment != NULL) && (uSizeBuf > s->gi.size_comment))
  1289.   {
  1290.     *(szComment + s->gi.size_comment) = '';
  1291.   }
  1292.   return (int)uReadThis;
  1293. }
  1294. */