mirror of
				https://github.com/pocoproject/poco.git
				synced 2025-10-25 18:22:59 +02:00 
			
		
		
		
	zlib updated to the version 1.2.7
This commit is contained in:
		
							
								
								
									
										25
									
								
								Foundation/src/gzclose.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								Foundation/src/gzclose.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,25 @@ | |||||||
|  | /* gzclose.c -- zlib gzclose() function | ||||||
|  |  * Copyright (C) 2004, 2010 Mark Adler | ||||||
|  |  * For conditions of distribution and use, see copyright notice in zlib.h | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | #include "gzguts.h" | ||||||
|  |  | ||||||
|  | /* gzclose() is in a separate file so that it is linked in only if it is used. | ||||||
|  |    That way the other gzclose functions can be used instead to avoid linking in | ||||||
|  |    unneeded compression or decompression routines. */ | ||||||
|  | int ZEXPORT gzclose(file) | ||||||
|  |     gzFile file; | ||||||
|  | { | ||||||
|  | #ifndef NO_GZCOMPRESS | ||||||
|  |     gz_statep state; | ||||||
|  |  | ||||||
|  |     if (file == NULL) | ||||||
|  |         return Z_STREAM_ERROR; | ||||||
|  |     state = (gz_statep)file; | ||||||
|  |  | ||||||
|  |     return state->mode == GZ_READ ? gzclose_r(file) : gzclose_w(file); | ||||||
|  | #else | ||||||
|  |     return gzclose_r(file); | ||||||
|  | #endif | ||||||
|  | } | ||||||
							
								
								
									
										193
									
								
								Foundation/src/gzguts.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										193
									
								
								Foundation/src/gzguts.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,193 @@ | |||||||
|  | /* gzguts.h -- zlib internal header definitions for gz* operations | ||||||
|  |  * Copyright (C) 2004, 2005, 2010, 2011, 2012 Mark Adler | ||||||
|  |  * For conditions of distribution and use, see copyright notice in zlib.h | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | #ifdef _LARGEFILE64_SOURCE | ||||||
|  | #  ifndef _LARGEFILE_SOURCE | ||||||
|  | #    define _LARGEFILE_SOURCE 1 | ||||||
|  | #  endif | ||||||
|  | #  ifdef _FILE_OFFSET_BITS | ||||||
|  | #    undef _FILE_OFFSET_BITS | ||||||
|  | #  endif | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | #ifdef HAVE_HIDDEN | ||||||
|  | #  define ZLIB_INTERNAL __attribute__((visibility ("hidden"))) | ||||||
|  | #else | ||||||
|  | #  define ZLIB_INTERNAL | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | #include <stdio.h> | ||||||
|  | #include "zlib.h" | ||||||
|  | #ifdef STDC | ||||||
|  | #  include <string.h> | ||||||
|  | #  include <stdlib.h> | ||||||
|  | #  include <limits.h> | ||||||
|  | #endif | ||||||
|  | #include <fcntl.h> | ||||||
|  |  | ||||||
|  | #ifdef _WIN32 | ||||||
|  | #  include <stddef.h> | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | #if defined(__TURBOC__) || defined(_MSC_VER) || defined(_WIN32) | ||||||
|  | #  include <io.h> | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | #ifdef NO_DEFLATE       /* for compatibility with old definition */ | ||||||
|  | #  define NO_GZCOMPRESS | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | #if defined(STDC99) || (defined(__TURBOC__) && __TURBOC__ >= 0x550) | ||||||
|  | #  ifndef HAVE_VSNPRINTF | ||||||
|  | #    define HAVE_VSNPRINTF | ||||||
|  | #  endif | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | #if defined(__CYGWIN__) | ||||||
|  | #  ifndef HAVE_VSNPRINTF | ||||||
|  | #    define HAVE_VSNPRINTF | ||||||
|  | #  endif | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | #if defined(MSDOS) && defined(__BORLANDC__) && (BORLANDC > 0x410) | ||||||
|  | #  ifndef HAVE_VSNPRINTF | ||||||
|  | #    define HAVE_VSNPRINTF | ||||||
|  | #  endif | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | #ifndef HAVE_VSNPRINTF | ||||||
|  | #  ifdef MSDOS | ||||||
|  | /* vsnprintf may exist on some MS-DOS compilers (DJGPP?), | ||||||
|  |  but for now we just assume it doesn't. */ | ||||||
|  | #    define NO_vsnprintf | ||||||
|  | #  endif | ||||||
|  | #  ifdef __TURBOC__ | ||||||
|  | #    define NO_vsnprintf | ||||||
|  | #  endif | ||||||
|  | #  ifdef WIN32 | ||||||
|  | /* In Win32, vsnprintf is available as the "non-ANSI" _vsnprintf. */ | ||||||
|  | #    if !defined(vsnprintf) && !defined(NO_vsnprintf) | ||||||
|  | #      if !defined(_MSC_VER) || ( defined(_MSC_VER) && _MSC_VER < 1500 ) | ||||||
|  | #         define vsnprintf _vsnprintf | ||||||
|  | #      endif | ||||||
|  | #    endif | ||||||
|  | #  endif | ||||||
|  | #  ifdef __SASC | ||||||
|  | #    define NO_vsnprintf | ||||||
|  | #  endif | ||||||
|  | #  ifdef VMS | ||||||
|  | #    define NO_vsnprintf | ||||||
|  | #  endif | ||||||
|  | #  ifdef __OS400__ | ||||||
|  | #    define NO_vsnprintf | ||||||
|  | #  endif | ||||||
|  | #  ifdef __MVS__ | ||||||
|  | #    define NO_vsnprintf | ||||||
|  | #  endif | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | #ifndef local | ||||||
|  | #  define local static | ||||||
|  | #endif | ||||||
|  | /* compile with -Dlocal if your debugger can't find static symbols */ | ||||||
|  |  | ||||||
|  | /* gz* functions always use library allocation functions */ | ||||||
|  | #ifndef STDC | ||||||
|  |   extern voidp  malloc OF((uInt size)); | ||||||
|  |   extern void   free   OF((voidpf ptr)); | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | /* get errno and strerror definition */ | ||||||
|  | #if defined UNDER_CE | ||||||
|  | #  include <windows.h> | ||||||
|  | #  define zstrerror() gz_strwinerror((DWORD)GetLastError()) | ||||||
|  | #else | ||||||
|  | #  ifndef NO_STRERROR | ||||||
|  | #    include <errno.h> | ||||||
|  | #    define zstrerror() strerror(errno) | ||||||
|  | #  else | ||||||
|  | #    define zstrerror() "stdio error (consult errno)" | ||||||
|  | #  endif | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | /* provide prototypes for these when building zlib without LFS */ | ||||||
|  | #if !defined(_LARGEFILE64_SOURCE) || _LFS64_LARGEFILE-0 == 0 | ||||||
|  |     ZEXTERN gzFile ZEXPORT gzopen64 OF((const char *, const char *)); | ||||||
|  |     ZEXTERN z_off64_t ZEXPORT gzseek64 OF((gzFile, z_off64_t, int)); | ||||||
|  |     ZEXTERN z_off64_t ZEXPORT gztell64 OF((gzFile)); | ||||||
|  |     ZEXTERN z_off64_t ZEXPORT gzoffset64 OF((gzFile)); | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | /* default memLevel */ | ||||||
|  | #if MAX_MEM_LEVEL >= 8 | ||||||
|  | #  define DEF_MEM_LEVEL 8 | ||||||
|  | #else | ||||||
|  | #  define DEF_MEM_LEVEL  MAX_MEM_LEVEL | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | /* default i/o buffer size -- double this for output when reading */ | ||||||
|  | #define GZBUFSIZE 8192 | ||||||
|  |  | ||||||
|  | /* gzip modes, also provide a little integrity check on the passed structure */ | ||||||
|  | #define GZ_NONE 0 | ||||||
|  | #define GZ_READ 7247 | ||||||
|  | #define GZ_WRITE 31153 | ||||||
|  | #define GZ_APPEND 1     /* mode set to GZ_WRITE after the file is opened */ | ||||||
|  |  | ||||||
|  | /* values for gz_state how */ | ||||||
|  | #define LOOK 0      /* look for a gzip header */ | ||||||
|  | #define COPY 1      /* copy input directly */ | ||||||
|  | #define GZIP 2      /* decompress a gzip stream */ | ||||||
|  |  | ||||||
|  | /* internal gzip file state data structure */ | ||||||
|  | typedef struct { | ||||||
|  |         /* exposed contents for gzgetc() macro */ | ||||||
|  |     struct gzFile_s x;      /* "x" for exposed */ | ||||||
|  |                             /* x.have: number of bytes available at x.next */ | ||||||
|  |                             /* x.next: next output data to deliver or write */ | ||||||
|  |                             /* x.pos: current position in uncompressed data */ | ||||||
|  |         /* used for both reading and writing */ | ||||||
|  |     int mode;               /* see gzip modes above */ | ||||||
|  |     int fd;                 /* file descriptor */ | ||||||
|  |     char *path;             /* path or fd for error messages */ | ||||||
|  |     unsigned size;          /* buffer size, zero if not allocated yet */ | ||||||
|  |     unsigned want;          /* requested buffer size, default is GZBUFSIZE */ | ||||||
|  |     unsigned char *in;      /* input buffer */ | ||||||
|  |     unsigned char *out;     /* output buffer (double-sized when reading) */ | ||||||
|  |     int direct;             /* 0 if processing gzip, 1 if transparent */ | ||||||
|  |         /* just for reading */ | ||||||
|  |     int how;                /* 0: get header, 1: copy, 2: decompress */ | ||||||
|  |     z_off64_t start;        /* where the gzip data started, for rewinding */ | ||||||
|  |     int eof;                /* true if end of input file reached */ | ||||||
|  |     int past;               /* true if read requested past end */ | ||||||
|  |         /* just for writing */ | ||||||
|  |     int level;              /* compression level */ | ||||||
|  |     int strategy;           /* compression strategy */ | ||||||
|  |         /* seek request */ | ||||||
|  |     z_off64_t skip;         /* amount to skip (already rewound if backwards) */ | ||||||
|  |     int seek;               /* true if seek request pending */ | ||||||
|  |         /* error information */ | ||||||
|  |     int err;                /* error code */ | ||||||
|  |     char *msg;              /* error message */ | ||||||
|  |         /* zlib inflate or deflate stream */ | ||||||
|  |     z_stream strm;          /* stream structure in-place (not a pointer) */ | ||||||
|  | } gz_state; | ||||||
|  | typedef gz_state FAR *gz_statep; | ||||||
|  |  | ||||||
|  | /* shared functions */ | ||||||
|  | void ZLIB_INTERNAL gz_error OF((gz_statep, int, const char *)); | ||||||
|  | #if defined UNDER_CE | ||||||
|  | char ZLIB_INTERNAL *gz_strwinerror OF((DWORD error)); | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | /* GT_OFF(x), where x is an unsigned value, is true if x > maximum z_off64_t | ||||||
|  |    value -- needed when comparing unsigned to z_off64_t, which is signed | ||||||
|  |    (possible z_off64_t types off_t, off64_t, and long are all signed) */ | ||||||
|  | #ifdef INT_MAX | ||||||
|  | #  define GT_OFF(x) (sizeof(int) == sizeof(z_off64_t) && (x) > INT_MAX) | ||||||
|  | #else | ||||||
|  | unsigned ZLIB_INTERNAL gz_intmax OF((void)); | ||||||
|  | #  define GT_OFF(x) (sizeof(int) == sizeof(z_off64_t) && (x) > gz_intmax()) | ||||||
|  | #endif | ||||||
							
								
								
									
										620
									
								
								Foundation/src/gzlib.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										620
									
								
								Foundation/src/gzlib.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,620 @@ | |||||||
|  | /* gzlib.c -- zlib functions common to reading and writing gzip files | ||||||
|  |  * Copyright (C) 2004, 2010, 2011, 2012 Mark Adler | ||||||
|  |  * For conditions of distribution and use, see copyright notice in zlib.h | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | #include "gzguts.h" | ||||||
|  |  | ||||||
|  | #if defined(_WIN32) && !defined(__BORLANDC__) | ||||||
|  | #  define LSEEK _lseeki64 | ||||||
|  | #else | ||||||
|  | #if defined(_LARGEFILE64_SOURCE) && _LFS64_LARGEFILE-0 | ||||||
|  | #  define LSEEK lseek64 | ||||||
|  | #else | ||||||
|  | #  define LSEEK lseek | ||||||
|  | #endif | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | /* Local functions */ | ||||||
|  | local void gz_reset OF((gz_statep)); | ||||||
|  | local gzFile gz_open OF((const void *, int, const char *)); | ||||||
|  |  | ||||||
|  | #if defined UNDER_CE | ||||||
|  |  | ||||||
|  | /* Map the Windows error number in ERROR to a locale-dependent error message | ||||||
|  |    string and return a pointer to it.  Typically, the values for ERROR come | ||||||
|  |    from GetLastError. | ||||||
|  |  | ||||||
|  |    The string pointed to shall not be modified by the application, but may be | ||||||
|  |    overwritten by a subsequent call to gz_strwinerror | ||||||
|  |  | ||||||
|  |    The gz_strwinerror function does not change the current setting of | ||||||
|  |    GetLastError. */ | ||||||
|  | char ZLIB_INTERNAL *gz_strwinerror (error) | ||||||
|  |      DWORD error; | ||||||
|  | { | ||||||
|  |     static char buf[1024]; | ||||||
|  |  | ||||||
|  |     wchar_t *msgbuf; | ||||||
|  |     DWORD lasterr = GetLastError(); | ||||||
|  |     DWORD chars = FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM | ||||||
|  |         | FORMAT_MESSAGE_ALLOCATE_BUFFER, | ||||||
|  |         NULL, | ||||||
|  |         error, | ||||||
|  |         0, /* Default language */ | ||||||
|  |         (LPVOID)&msgbuf, | ||||||
|  |         0, | ||||||
|  |         NULL); | ||||||
|  |     if (chars != 0) { | ||||||
|  |         /* If there is an \r\n appended, zap it.  */ | ||||||
|  |         if (chars >= 2 | ||||||
|  |             && msgbuf[chars - 2] == '\r' && msgbuf[chars - 1] == '\n') { | ||||||
|  |             chars -= 2; | ||||||
|  |             msgbuf[chars] = 0; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (chars > sizeof (buf) - 1) { | ||||||
|  |             chars = sizeof (buf) - 1; | ||||||
|  |             msgbuf[chars] = 0; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         wcstombs(buf, msgbuf, chars + 1); | ||||||
|  |         LocalFree(msgbuf); | ||||||
|  |     } | ||||||
|  |     else { | ||||||
|  |         sprintf(buf, "unknown win32 error (%ld)", error); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     SetLastError(lasterr); | ||||||
|  |     return buf; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #endif /* UNDER_CE */ | ||||||
|  |  | ||||||
|  | /* Reset gzip file state */ | ||||||
|  | local void gz_reset(state) | ||||||
|  |     gz_statep state; | ||||||
|  | { | ||||||
|  |     state->x.have = 0;              /* no output data available */ | ||||||
|  |     if (state->mode == GZ_READ) {   /* for reading ... */ | ||||||
|  |         state->eof = 0;             /* not at end of file */ | ||||||
|  |         state->past = 0;            /* have not read past end yet */ | ||||||
|  |         state->how = LOOK;          /* look for gzip header */ | ||||||
|  |     } | ||||||
|  |     state->seek = 0;                /* no seek request pending */ | ||||||
|  |     gz_error(state, Z_OK, NULL);    /* clear error */ | ||||||
|  |     state->x.pos = 0;               /* no uncompressed data yet */ | ||||||
|  |     state->strm.avail_in = 0;       /* no input data yet */ | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Open a gzip file either by name or file descriptor. */ | ||||||
|  | local gzFile gz_open(path, fd, mode) | ||||||
|  |     const void *path; | ||||||
|  |     int fd; | ||||||
|  |     const char *mode; | ||||||
|  | { | ||||||
|  |     gz_statep state; | ||||||
|  |     size_t len; | ||||||
|  |     int oflag; | ||||||
|  | #ifdef O_CLOEXEC | ||||||
|  |     int cloexec = 0; | ||||||
|  | #endif | ||||||
|  | #ifdef O_EXCL | ||||||
|  |     int exclusive = 0; | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  |     /* check input */ | ||||||
|  |     if (path == NULL) | ||||||
|  |         return NULL; | ||||||
|  |  | ||||||
|  |     /* allocate gzFile structure to return */ | ||||||
|  |     state = malloc(sizeof(gz_state)); | ||||||
|  |     if (state == NULL) | ||||||
|  |         return NULL; | ||||||
|  |     state->size = 0;            /* no buffers allocated yet */ | ||||||
|  |     state->want = GZBUFSIZE;    /* requested buffer size */ | ||||||
|  |     state->msg = NULL;          /* no error message yet */ | ||||||
|  |  | ||||||
|  |     /* interpret mode */ | ||||||
|  |     state->mode = GZ_NONE; | ||||||
|  |     state->level = Z_DEFAULT_COMPRESSION; | ||||||
|  |     state->strategy = Z_DEFAULT_STRATEGY; | ||||||
|  |     state->direct = 0; | ||||||
|  |     while (*mode) { | ||||||
|  |         if (*mode >= '0' && *mode <= '9') | ||||||
|  |             state->level = *mode - '0'; | ||||||
|  |         else | ||||||
|  |             switch (*mode) { | ||||||
|  |             case 'r': | ||||||
|  |                 state->mode = GZ_READ; | ||||||
|  |                 break; | ||||||
|  | #ifndef NO_GZCOMPRESS | ||||||
|  |             case 'w': | ||||||
|  |                 state->mode = GZ_WRITE; | ||||||
|  |                 break; | ||||||
|  |             case 'a': | ||||||
|  |                 state->mode = GZ_APPEND; | ||||||
|  |                 break; | ||||||
|  | #endif | ||||||
|  |             case '+':       /* can't read and write at the same time */ | ||||||
|  |                 free(state); | ||||||
|  |                 return NULL; | ||||||
|  |             case 'b':       /* ignore -- will request binary anyway */ | ||||||
|  |                 break; | ||||||
|  | #ifdef O_CLOEXEC | ||||||
|  |             case 'e': | ||||||
|  |                 cloexec = 1; | ||||||
|  |                 break; | ||||||
|  | #endif | ||||||
|  | #ifdef O_EXCL | ||||||
|  |             case 'x': | ||||||
|  |                 exclusive = 1; | ||||||
|  |                 break; | ||||||
|  | #endif | ||||||
|  |             case 'f': | ||||||
|  |                 state->strategy = Z_FILTERED; | ||||||
|  |                 break; | ||||||
|  |             case 'h': | ||||||
|  |                 state->strategy = Z_HUFFMAN_ONLY; | ||||||
|  |                 break; | ||||||
|  |             case 'R': | ||||||
|  |                 state->strategy = Z_RLE; | ||||||
|  |                 break; | ||||||
|  |             case 'F': | ||||||
|  |                 state->strategy = Z_FIXED; | ||||||
|  |             case 'T': | ||||||
|  |                 state->direct = 1; | ||||||
|  |             default:        /* could consider as an error, but just ignore */ | ||||||
|  |                 ; | ||||||
|  |             } | ||||||
|  |         mode++; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* must provide an "r", "w", or "a" */ | ||||||
|  |     if (state->mode == GZ_NONE) { | ||||||
|  |         free(state); | ||||||
|  |         return NULL; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* can't force transparent read */ | ||||||
|  |     if (state->mode == GZ_READ) { | ||||||
|  |         if (state->direct) { | ||||||
|  |             free(state); | ||||||
|  |             return NULL; | ||||||
|  |         } | ||||||
|  |         state->direct = 1;      /* for empty file */ | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* save the path name for error messages */ | ||||||
|  | #ifdef _WIN32 | ||||||
|  |     if (fd == -2) { | ||||||
|  |         len = wcstombs(NULL, path, 0); | ||||||
|  |         if (len == (size_t)-1) | ||||||
|  |             len = 0; | ||||||
|  |     } | ||||||
|  |     else | ||||||
|  | #endif | ||||||
|  |         len = strlen(path); | ||||||
|  |     state->path = malloc(len + 1); | ||||||
|  |     if (state->path == NULL) { | ||||||
|  |         free(state); | ||||||
|  |         return NULL; | ||||||
|  |     } | ||||||
|  | #ifdef _WIN32 | ||||||
|  |     if (fd == -2) | ||||||
|  |         if (len) | ||||||
|  |             wcstombs(state->path, path, len + 1); | ||||||
|  |         else | ||||||
|  |             *(state->path) = 0; | ||||||
|  |     else | ||||||
|  | #endif | ||||||
|  |         strcpy(state->path, path); | ||||||
|  |  | ||||||
|  |     /* compute the flags for open() */ | ||||||
|  |     oflag = | ||||||
|  | #ifdef O_LARGEFILE | ||||||
|  |         O_LARGEFILE | | ||||||
|  | #endif | ||||||
|  | #ifdef O_BINARY | ||||||
|  |         O_BINARY | | ||||||
|  | #endif | ||||||
|  | #ifdef O_CLOEXEC | ||||||
|  |         (cloexec ? O_CLOEXEC : 0) | | ||||||
|  | #endif | ||||||
|  |         (state->mode == GZ_READ ? | ||||||
|  |          O_RDONLY : | ||||||
|  |          (O_WRONLY | O_CREAT | | ||||||
|  | #ifdef O_EXCL | ||||||
|  |           (exclusive ? O_EXCL : 0) | | ||||||
|  | #endif | ||||||
|  |           (state->mode == GZ_WRITE ? | ||||||
|  |            O_TRUNC : | ||||||
|  |            O_APPEND))); | ||||||
|  |  | ||||||
|  |     /* open the file with the appropriate flags (or just use fd) */ | ||||||
|  |     state->fd = fd > -1 ? fd : ( | ||||||
|  | #ifdef _WIN32 | ||||||
|  |         fd == -2 ? _wopen(path, oflag, 0666) : | ||||||
|  | #endif | ||||||
|  |         open(path, oflag, 0666)); | ||||||
|  |     if (state->fd == -1) { | ||||||
|  |         free(state->path); | ||||||
|  |         free(state); | ||||||
|  |         return NULL; | ||||||
|  |     } | ||||||
|  |     if (state->mode == GZ_APPEND) | ||||||
|  |         state->mode = GZ_WRITE;         /* simplify later checks */ | ||||||
|  |  | ||||||
|  |     /* save the current position for rewinding (only if reading) */ | ||||||
|  |     if (state->mode == GZ_READ) { | ||||||
|  |         state->start = LSEEK(state->fd, 0, SEEK_CUR); | ||||||
|  |         if (state->start == -1) state->start = 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* initialize stream */ | ||||||
|  |     gz_reset(state); | ||||||
|  |  | ||||||
|  |     /* return stream */ | ||||||
|  |     return (gzFile)state; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* -- see zlib.h -- */ | ||||||
|  | gzFile ZEXPORT gzopen(path, mode) | ||||||
|  |     const char *path; | ||||||
|  |     const char *mode; | ||||||
|  | { | ||||||
|  |     return gz_open(path, -1, mode); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* -- see zlib.h -- */ | ||||||
|  | gzFile ZEXPORT gzopen64(path, mode) | ||||||
|  |     const char *path; | ||||||
|  |     const char *mode; | ||||||
|  | { | ||||||
|  |     return gz_open(path, -1, mode); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* -- see zlib.h -- */ | ||||||
|  | gzFile ZEXPORT gzdopen(fd, mode) | ||||||
|  |     int fd; | ||||||
|  |     const char *mode; | ||||||
|  | { | ||||||
|  |     char *path;         /* identifier for error messages */ | ||||||
|  |     gzFile gz; | ||||||
|  |  | ||||||
|  |     if (fd == -1 || (path = malloc(7 + 3 * sizeof(int))) == NULL) | ||||||
|  |         return NULL; | ||||||
|  |     sprintf(path, "<fd:%d>", fd);   /* for debugging */ | ||||||
|  |     gz = gz_open(path, fd, mode); | ||||||
|  |     free(path); | ||||||
|  |     return gz; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* -- see zlib.h -- */ | ||||||
|  | #ifdef _WIN32 | ||||||
|  | gzFile ZEXPORT gzopen_w(path, mode) | ||||||
|  |     const wchar_t *path; | ||||||
|  |     const char *mode; | ||||||
|  | { | ||||||
|  |     return gz_open(path, -2, mode); | ||||||
|  | } | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | /* -- see zlib.h -- */ | ||||||
|  | int ZEXPORT gzbuffer(file, size) | ||||||
|  |     gzFile file; | ||||||
|  |     unsigned size; | ||||||
|  | { | ||||||
|  |     gz_statep state; | ||||||
|  |  | ||||||
|  |     /* get internal structure and check integrity */ | ||||||
|  |     if (file == NULL) | ||||||
|  |         return -1; | ||||||
|  |     state = (gz_statep)file; | ||||||
|  |     if (state->mode != GZ_READ && state->mode != GZ_WRITE) | ||||||
|  |         return -1; | ||||||
|  |  | ||||||
|  |     /* make sure we haven't already allocated memory */ | ||||||
|  |     if (state->size != 0) | ||||||
|  |         return -1; | ||||||
|  |  | ||||||
|  |     /* check and set requested size */ | ||||||
|  |     if (size < 2) | ||||||
|  |         size = 2;               /* need two bytes to check magic header */ | ||||||
|  |     state->want = size; | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* -- see zlib.h -- */ | ||||||
|  | int ZEXPORT gzrewind(file) | ||||||
|  |     gzFile file; | ||||||
|  | { | ||||||
|  |     gz_statep state; | ||||||
|  |  | ||||||
|  |     /* get internal structure */ | ||||||
|  |     if (file == NULL) | ||||||
|  |         return -1; | ||||||
|  |     state = (gz_statep)file; | ||||||
|  |  | ||||||
|  |     /* check that we're reading and that there's no error */ | ||||||
|  |     if (state->mode != GZ_READ || | ||||||
|  |             (state->err != Z_OK && state->err != Z_BUF_ERROR)) | ||||||
|  |         return -1; | ||||||
|  |  | ||||||
|  |     /* back up and start over */ | ||||||
|  |     if (LSEEK(state->fd, state->start, SEEK_SET) == -1) | ||||||
|  |         return -1; | ||||||
|  |     gz_reset(state); | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* -- see zlib.h -- */ | ||||||
|  | z_off64_t ZEXPORT gzseek64(file, offset, whence) | ||||||
|  |     gzFile file; | ||||||
|  |     z_off64_t offset; | ||||||
|  |     int whence; | ||||||
|  | { | ||||||
|  |     unsigned n; | ||||||
|  |     z_off64_t ret; | ||||||
|  |     gz_statep state; | ||||||
|  |  | ||||||
|  |     /* get internal structure and check integrity */ | ||||||
|  |     if (file == NULL) | ||||||
|  |         return -1; | ||||||
|  |     state = (gz_statep)file; | ||||||
|  |     if (state->mode != GZ_READ && state->mode != GZ_WRITE) | ||||||
|  |         return -1; | ||||||
|  |  | ||||||
|  |     /* check that there's no error */ | ||||||
|  |     if (state->err != Z_OK && state->err != Z_BUF_ERROR) | ||||||
|  |         return -1; | ||||||
|  |  | ||||||
|  |     /* can only seek from start or relative to current position */ | ||||||
|  |     if (whence != SEEK_SET && whence != SEEK_CUR) | ||||||
|  |         return -1; | ||||||
|  |  | ||||||
|  |     /* normalize offset to a SEEK_CUR specification */ | ||||||
|  |     if (whence == SEEK_SET) | ||||||
|  |         offset -= state->x.pos; | ||||||
|  |     else if (state->seek) | ||||||
|  |         offset += state->skip; | ||||||
|  |     state->seek = 0; | ||||||
|  |  | ||||||
|  |     /* if within raw area while reading, just go there */ | ||||||
|  |     if (state->mode == GZ_READ && state->how == COPY && | ||||||
|  |             state->x.pos + offset >= 0) { | ||||||
|  |         ret = LSEEK(state->fd, offset - state->x.have, SEEK_CUR); | ||||||
|  |         if (ret == -1) | ||||||
|  |             return -1; | ||||||
|  |         state->x.have = 0; | ||||||
|  |         state->eof = 0; | ||||||
|  |         state->past = 0; | ||||||
|  |         state->seek = 0; | ||||||
|  |         gz_error(state, Z_OK, NULL); | ||||||
|  |         state->strm.avail_in = 0; | ||||||
|  |         state->x.pos += offset; | ||||||
|  |         return state->x.pos; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* calculate skip amount, rewinding if needed for back seek when reading */ | ||||||
|  |     if (offset < 0) { | ||||||
|  |         if (state->mode != GZ_READ)         /* writing -- can't go backwards */ | ||||||
|  |             return -1; | ||||||
|  |         offset += state->x.pos; | ||||||
|  |         if (offset < 0)                     /* before start of file! */ | ||||||
|  |             return -1; | ||||||
|  |         if (gzrewind(file) == -1)           /* rewind, then skip to offset */ | ||||||
|  |             return -1; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* if reading, skip what's in output buffer (one less gzgetc() check) */ | ||||||
|  |     if (state->mode == GZ_READ) { | ||||||
|  |         n = GT_OFF(state->x.have) || (z_off64_t)state->x.have > offset ? | ||||||
|  |             (unsigned)offset : state->x.have; | ||||||
|  |         state->x.have -= n; | ||||||
|  |         state->x.next += n; | ||||||
|  |         state->x.pos += n; | ||||||
|  |         offset -= n; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* request skip (if not zero) */ | ||||||
|  |     if (offset) { | ||||||
|  |         state->seek = 1; | ||||||
|  |         state->skip = offset; | ||||||
|  |     } | ||||||
|  |     return state->x.pos + offset; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* -- see zlib.h -- */ | ||||||
|  | z_off_t ZEXPORT gzseek(file, offset, whence) | ||||||
|  |     gzFile file; | ||||||
|  |     z_off_t offset; | ||||||
|  |     int whence; | ||||||
|  | { | ||||||
|  |     z_off64_t ret; | ||||||
|  |  | ||||||
|  |     ret = gzseek64(file, (z_off64_t)offset, whence); | ||||||
|  |     return ret == (z_off_t)ret ? (z_off_t)ret : -1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* -- see zlib.h -- */ | ||||||
|  | z_off64_t ZEXPORT gztell64(file) | ||||||
|  |     gzFile file; | ||||||
|  | { | ||||||
|  |     gz_statep state; | ||||||
|  |  | ||||||
|  |     /* get internal structure and check integrity */ | ||||||
|  |     if (file == NULL) | ||||||
|  |         return -1; | ||||||
|  |     state = (gz_statep)file; | ||||||
|  |     if (state->mode != GZ_READ && state->mode != GZ_WRITE) | ||||||
|  |         return -1; | ||||||
|  |  | ||||||
|  |     /* return position */ | ||||||
|  |     return state->x.pos + (state->seek ? state->skip : 0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* -- see zlib.h -- */ | ||||||
|  | z_off_t ZEXPORT gztell(file) | ||||||
|  |     gzFile file; | ||||||
|  | { | ||||||
|  |     z_off64_t ret; | ||||||
|  |  | ||||||
|  |     ret = gztell64(file); | ||||||
|  |     return ret == (z_off_t)ret ? (z_off_t)ret : -1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* -- see zlib.h -- */ | ||||||
|  | z_off64_t ZEXPORT gzoffset64(file) | ||||||
|  |     gzFile file; | ||||||
|  | { | ||||||
|  |     z_off64_t offset; | ||||||
|  |     gz_statep state; | ||||||
|  |  | ||||||
|  |     /* get internal structure and check integrity */ | ||||||
|  |     if (file == NULL) | ||||||
|  |         return -1; | ||||||
|  |     state = (gz_statep)file; | ||||||
|  |     if (state->mode != GZ_READ && state->mode != GZ_WRITE) | ||||||
|  |         return -1; | ||||||
|  |  | ||||||
|  |     /* compute and return effective offset in file */ | ||||||
|  |     offset = LSEEK(state->fd, 0, SEEK_CUR); | ||||||
|  |     if (offset == -1) | ||||||
|  |         return -1; | ||||||
|  |     if (state->mode == GZ_READ)             /* reading */ | ||||||
|  |         offset -= state->strm.avail_in;     /* don't count buffered input */ | ||||||
|  |     return offset; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* -- see zlib.h -- */ | ||||||
|  | z_off_t ZEXPORT gzoffset(file) | ||||||
|  |     gzFile file; | ||||||
|  | { | ||||||
|  |     z_off64_t ret; | ||||||
|  |  | ||||||
|  |     ret = gzoffset64(file); | ||||||
|  |     return ret == (z_off_t)ret ? (z_off_t)ret : -1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* -- see zlib.h -- */ | ||||||
|  | int ZEXPORT gzeof(file) | ||||||
|  |     gzFile file; | ||||||
|  | { | ||||||
|  |     gz_statep state; | ||||||
|  |  | ||||||
|  |     /* get internal structure and check integrity */ | ||||||
|  |     if (file == NULL) | ||||||
|  |         return 0; | ||||||
|  |     state = (gz_statep)file; | ||||||
|  |     if (state->mode != GZ_READ && state->mode != GZ_WRITE) | ||||||
|  |         return 0; | ||||||
|  |  | ||||||
|  |     /* return end-of-file state */ | ||||||
|  |     return state->mode == GZ_READ ? state->past : 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* -- see zlib.h -- */ | ||||||
|  | const char * ZEXPORT gzerror(file, errnum) | ||||||
|  |     gzFile file; | ||||||
|  |     int *errnum; | ||||||
|  | { | ||||||
|  |     gz_statep state; | ||||||
|  |  | ||||||
|  |     /* get internal structure and check integrity */ | ||||||
|  |     if (file == NULL) | ||||||
|  |         return NULL; | ||||||
|  |     state = (gz_statep)file; | ||||||
|  |     if (state->mode != GZ_READ && state->mode != GZ_WRITE) | ||||||
|  |         return NULL; | ||||||
|  |  | ||||||
|  |     /* return error information */ | ||||||
|  |     if (errnum != NULL) | ||||||
|  |         *errnum = state->err; | ||||||
|  |     return state->msg == NULL ? "" : state->msg; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* -- see zlib.h -- */ | ||||||
|  | void ZEXPORT gzclearerr(file) | ||||||
|  |     gzFile file; | ||||||
|  | { | ||||||
|  |     gz_statep state; | ||||||
|  |  | ||||||
|  |     /* get internal structure and check integrity */ | ||||||
|  |     if (file == NULL) | ||||||
|  |         return; | ||||||
|  |     state = (gz_statep)file; | ||||||
|  |     if (state->mode != GZ_READ && state->mode != GZ_WRITE) | ||||||
|  |         return; | ||||||
|  |  | ||||||
|  |     /* clear error and end-of-file */ | ||||||
|  |     if (state->mode == GZ_READ) { | ||||||
|  |         state->eof = 0; | ||||||
|  |         state->past = 0; | ||||||
|  |     } | ||||||
|  |     gz_error(state, Z_OK, NULL); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Create an error message in allocated memory and set state->err and | ||||||
|  |    state->msg accordingly.  Free any previous error message already there.  Do | ||||||
|  |    not try to free or allocate space if the error is Z_MEM_ERROR (out of | ||||||
|  |    memory).  Simply save the error message as a static string.  If there is an | ||||||
|  |    allocation failure constructing the error message, then convert the error to | ||||||
|  |    out of memory. */ | ||||||
|  | void ZLIB_INTERNAL gz_error(state, err, msg) | ||||||
|  |     gz_statep state; | ||||||
|  |     int err; | ||||||
|  |     const char *msg; | ||||||
|  | { | ||||||
|  |     /* free previously allocated message and clear */ | ||||||
|  |     if (state->msg != NULL) { | ||||||
|  |         if (state->err != Z_MEM_ERROR) | ||||||
|  |             free(state->msg); | ||||||
|  |         state->msg = NULL; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* if fatal, set state->x.have to 0 so that the gzgetc() macro fails */ | ||||||
|  |     if (err != Z_OK && err != Z_BUF_ERROR) | ||||||
|  |         state->x.have = 0; | ||||||
|  |  | ||||||
|  |     /* set error code, and if no message, then done */ | ||||||
|  |     state->err = err; | ||||||
|  |     if (msg == NULL) | ||||||
|  |         return; | ||||||
|  |  | ||||||
|  |     /* for an out of memory error, save as static string */ | ||||||
|  |     if (err == Z_MEM_ERROR) { | ||||||
|  |         state->msg = (char *)msg; | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* construct error message with path */ | ||||||
|  |     if ((state->msg = malloc(strlen(state->path) + strlen(msg) + 3)) == NULL) { | ||||||
|  |         state->err = Z_MEM_ERROR; | ||||||
|  |         state->msg = (char *)"out of memory"; | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  |     strcpy(state->msg, state->path); | ||||||
|  |     strcat(state->msg, ": "); | ||||||
|  |     strcat(state->msg, msg); | ||||||
|  |     return; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #ifndef INT_MAX | ||||||
|  | /* portably return maximum value for an int (when limits.h presumed not | ||||||
|  |    available) -- we need to do this to cover cases where 2's complement not | ||||||
|  |    used, since C standard permits 1's complement and sign-bit representations, | ||||||
|  |    otherwise we could just use ((unsigned)-1) >> 1 */ | ||||||
|  | unsigned ZLIB_INTERNAL gz_intmax() | ||||||
|  | { | ||||||
|  |     unsigned p, q; | ||||||
|  |  | ||||||
|  |     p = 1; | ||||||
|  |     do { | ||||||
|  |         q = p; | ||||||
|  |         p <<= 1; | ||||||
|  |         p++; | ||||||
|  |     } while (p > q); | ||||||
|  |     return q >> 1; | ||||||
|  | } | ||||||
|  | #endif | ||||||
							
								
								
									
										589
									
								
								Foundation/src/gzread.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										589
									
								
								Foundation/src/gzread.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,589 @@ | |||||||
|  | /* gzread.c -- zlib functions for reading gzip files | ||||||
|  |  * Copyright (C) 2004, 2005, 2010, 2011, 2012 Mark Adler | ||||||
|  |  * For conditions of distribution and use, see copyright notice in zlib.h | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | #include "gzguts.h" | ||||||
|  |  | ||||||
|  | /* Local functions */ | ||||||
|  | local int gz_load OF((gz_statep, unsigned char *, unsigned, unsigned *)); | ||||||
|  | local int gz_avail OF((gz_statep)); | ||||||
|  | local int gz_look OF((gz_statep)); | ||||||
|  | local int gz_decomp OF((gz_statep)); | ||||||
|  | local int gz_fetch OF((gz_statep)); | ||||||
|  | local int gz_skip OF((gz_statep, z_off64_t)); | ||||||
|  |  | ||||||
|  | /* Use read() to load a buffer -- return -1 on error, otherwise 0.  Read from | ||||||
|  |    state->fd, and update state->eof, state->err, and state->msg as appropriate. | ||||||
|  |    This function needs to loop on read(), since read() is not guaranteed to | ||||||
|  |    read the number of bytes requested, depending on the type of descriptor. */ | ||||||
|  | local int gz_load(state, buf, len, have) | ||||||
|  |     gz_statep state; | ||||||
|  |     unsigned char *buf; | ||||||
|  |     unsigned len; | ||||||
|  |     unsigned *have; | ||||||
|  | { | ||||||
|  |     int ret; | ||||||
|  |  | ||||||
|  |     *have = 0; | ||||||
|  |     do { | ||||||
|  |         ret = read(state->fd, buf + *have, len - *have); | ||||||
|  |         if (ret <= 0) | ||||||
|  |             break; | ||||||
|  |         *have += ret; | ||||||
|  |     } while (*have < len); | ||||||
|  |     if (ret < 0) { | ||||||
|  |         gz_error(state, Z_ERRNO, zstrerror()); | ||||||
|  |         return -1; | ||||||
|  |     } | ||||||
|  |     if (ret == 0) | ||||||
|  |         state->eof = 1; | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Load up input buffer and set eof flag if last data loaded -- return -1 on | ||||||
|  |    error, 0 otherwise.  Note that the eof flag is set when the end of the input | ||||||
|  |    file is reached, even though there may be unused data in the buffer.  Once | ||||||
|  |    that data has been used, no more attempts will be made to read the file. | ||||||
|  |    If strm->avail_in != 0, then the current data is moved to the beginning of | ||||||
|  |    the input buffer, and then the remainder of the buffer is loaded with the | ||||||
|  |    available data from the input file. */ | ||||||
|  | local int gz_avail(state) | ||||||
|  |     gz_statep state; | ||||||
|  | { | ||||||
|  |     unsigned got; | ||||||
|  |     z_streamp strm = &(state->strm); | ||||||
|  |  | ||||||
|  |     if (state->err != Z_OK && state->err != Z_BUF_ERROR) | ||||||
|  |         return -1; | ||||||
|  |     if (state->eof == 0) { | ||||||
|  |         if (strm->avail_in) {       /* copy what's there to the start */ | ||||||
|  |             unsigned char *p = state->in, *q = strm->next_in; | ||||||
|  |             unsigned n = strm->avail_in; | ||||||
|  |             do { | ||||||
|  |                 *p++ = *q++; | ||||||
|  |             } while (--n); | ||||||
|  |         } | ||||||
|  |         if (gz_load(state, state->in + strm->avail_in, | ||||||
|  |                     state->size - strm->avail_in, &got) == -1) | ||||||
|  |             return -1; | ||||||
|  |         strm->avail_in += got; | ||||||
|  |         strm->next_in = state->in; | ||||||
|  |     } | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Look for gzip header, set up for inflate or copy.  state->x.have must be 0. | ||||||
|  |    If this is the first time in, allocate required memory.  state->how will be | ||||||
|  |    left unchanged if there is no more input data available, will be set to COPY | ||||||
|  |    if there is no gzip header and direct copying will be performed, or it will | ||||||
|  |    be set to GZIP for decompression.  If direct copying, then leftover input | ||||||
|  |    data from the input buffer will be copied to the output buffer.  In that | ||||||
|  |    case, all further file reads will be directly to either the output buffer or | ||||||
|  |    a user buffer.  If decompressing, the inflate state will be initialized. | ||||||
|  |    gz_look() will return 0 on success or -1 on failure. */ | ||||||
|  | local int gz_look(state) | ||||||
|  |     gz_statep state; | ||||||
|  | { | ||||||
|  |     z_streamp strm = &(state->strm); | ||||||
|  |  | ||||||
|  |     /* allocate read buffers and inflate memory */ | ||||||
|  |     if (state->size == 0) { | ||||||
|  |         /* allocate buffers */ | ||||||
|  |         state->in = malloc(state->want); | ||||||
|  |         state->out = malloc(state->want << 1); | ||||||
|  |         if (state->in == NULL || state->out == NULL) { | ||||||
|  |             if (state->out != NULL) | ||||||
|  |                 free(state->out); | ||||||
|  |             if (state->in != NULL) | ||||||
|  |                 free(state->in); | ||||||
|  |             gz_error(state, Z_MEM_ERROR, "out of memory"); | ||||||
|  |             return -1; | ||||||
|  |         } | ||||||
|  |         state->size = state->want; | ||||||
|  |  | ||||||
|  |         /* allocate inflate memory */ | ||||||
|  |         state->strm.zalloc = Z_NULL; | ||||||
|  |         state->strm.zfree = Z_NULL; | ||||||
|  |         state->strm.opaque = Z_NULL; | ||||||
|  |         state->strm.avail_in = 0; | ||||||
|  |         state->strm.next_in = Z_NULL; | ||||||
|  |         if (inflateInit2(&(state->strm), 15 + 16) != Z_OK) {    /* gunzip */ | ||||||
|  |             free(state->out); | ||||||
|  |             free(state->in); | ||||||
|  |             state->size = 0; | ||||||
|  |             gz_error(state, Z_MEM_ERROR, "out of memory"); | ||||||
|  |             return -1; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* get at least the magic bytes in the input buffer */ | ||||||
|  |     if (strm->avail_in < 2) { | ||||||
|  |         if (gz_avail(state) == -1) | ||||||
|  |             return -1; | ||||||
|  |         if (strm->avail_in == 0) | ||||||
|  |             return 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* look for gzip magic bytes -- if there, do gzip decoding (note: there is | ||||||
|  |        a logical dilemma here when considering the case of a partially written | ||||||
|  |        gzip file, to wit, if a single 31 byte is written, then we cannot tell | ||||||
|  |        whether this is a single-byte file, or just a partially written gzip | ||||||
|  |        file -- for here we assume that if a gzip file is being written, then | ||||||
|  |        the header will be written in a single operation, so that reading a | ||||||
|  |        single byte is sufficient indication that it is not a gzip file) */ | ||||||
|  |     if (strm->avail_in > 1 && | ||||||
|  |             strm->next_in[0] == 31 && strm->next_in[1] == 139) { | ||||||
|  |         inflateReset(strm); | ||||||
|  |         state->how = GZIP; | ||||||
|  |         state->direct = 0; | ||||||
|  |         return 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* no gzip header -- if we were decoding gzip before, then this is trailing | ||||||
|  |        garbage.  Ignore the trailing garbage and finish. */ | ||||||
|  |     if (state->direct == 0) { | ||||||
|  |         strm->avail_in = 0; | ||||||
|  |         state->eof = 1; | ||||||
|  |         state->x.have = 0; | ||||||
|  |         return 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* doing raw i/o, copy any leftover input to output -- this assumes that | ||||||
|  |        the output buffer is larger than the input buffer, which also assures | ||||||
|  |        space for gzungetc() */ | ||||||
|  |     state->x.next = state->out; | ||||||
|  |     if (strm->avail_in) { | ||||||
|  |         memcpy(state->x.next, strm->next_in, strm->avail_in); | ||||||
|  |         state->x.have = strm->avail_in; | ||||||
|  |         strm->avail_in = 0; | ||||||
|  |     } | ||||||
|  |     state->how = COPY; | ||||||
|  |     state->direct = 1; | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Decompress from input to the provided next_out and avail_out in the state. | ||||||
|  |    On return, state->x.have and state->x.next point to the just decompressed | ||||||
|  |    data.  If the gzip stream completes, state->how is reset to LOOK to look for | ||||||
|  |    the next gzip stream or raw data, once state->x.have is depleted.  Returns 0 | ||||||
|  |    on success, -1 on failure. */ | ||||||
|  | local int gz_decomp(state) | ||||||
|  |     gz_statep state; | ||||||
|  | { | ||||||
|  |     int ret = Z_OK; | ||||||
|  |     unsigned had; | ||||||
|  |     z_streamp strm = &(state->strm); | ||||||
|  |  | ||||||
|  |     /* fill output buffer up to end of deflate stream */ | ||||||
|  |     had = strm->avail_out; | ||||||
|  |     do { | ||||||
|  |         /* get more input for inflate() */ | ||||||
|  |         if (strm->avail_in == 0 && gz_avail(state) == -1) | ||||||
|  |             return -1; | ||||||
|  |         if (strm->avail_in == 0) { | ||||||
|  |             gz_error(state, Z_BUF_ERROR, "unexpected end of file"); | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         /* decompress and handle errors */ | ||||||
|  |         ret = inflate(strm, Z_NO_FLUSH); | ||||||
|  |         if (ret == Z_STREAM_ERROR || ret == Z_NEED_DICT) { | ||||||
|  |             gz_error(state, Z_STREAM_ERROR, | ||||||
|  |                      "internal error: inflate stream corrupt"); | ||||||
|  |             return -1; | ||||||
|  |         } | ||||||
|  |         if (ret == Z_MEM_ERROR) { | ||||||
|  |             gz_error(state, Z_MEM_ERROR, "out of memory"); | ||||||
|  |             return -1; | ||||||
|  |         } | ||||||
|  |         if (ret == Z_DATA_ERROR) {              /* deflate stream invalid */ | ||||||
|  |             gz_error(state, Z_DATA_ERROR, | ||||||
|  |                      strm->msg == NULL ? "compressed data error" : strm->msg); | ||||||
|  |             return -1; | ||||||
|  |         } | ||||||
|  |     } while (strm->avail_out && ret != Z_STREAM_END); | ||||||
|  |  | ||||||
|  |     /* update available output */ | ||||||
|  |     state->x.have = had - strm->avail_out; | ||||||
|  |     state->x.next = strm->next_out - state->x.have; | ||||||
|  |  | ||||||
|  |     /* if the gzip stream completed successfully, look for another */ | ||||||
|  |     if (ret == Z_STREAM_END) | ||||||
|  |         state->how = LOOK; | ||||||
|  |  | ||||||
|  |     /* good decompression */ | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Fetch data and put it in the output buffer.  Assumes state->x.have is 0. | ||||||
|  |    Data is either copied from the input file or decompressed from the input | ||||||
|  |    file depending on state->how.  If state->how is LOOK, then a gzip header is | ||||||
|  |    looked for to determine whether to copy or decompress.  Returns -1 on error, | ||||||
|  |    otherwise 0.  gz_fetch() will leave state->how as COPY or GZIP unless the | ||||||
|  |    end of the input file has been reached and all data has been processed.  */ | ||||||
|  | local int gz_fetch(state) | ||||||
|  |     gz_statep state; | ||||||
|  | { | ||||||
|  |     z_streamp strm = &(state->strm); | ||||||
|  |  | ||||||
|  |     do { | ||||||
|  |         switch(state->how) { | ||||||
|  |         case LOOK:      /* -> LOOK, COPY (only if never GZIP), or GZIP */ | ||||||
|  |             if (gz_look(state) == -1) | ||||||
|  |                 return -1; | ||||||
|  |             if (state->how == LOOK) | ||||||
|  |                 return 0; | ||||||
|  |             break; | ||||||
|  |         case COPY:      /* -> COPY */ | ||||||
|  |             if (gz_load(state, state->out, state->size << 1, &(state->x.have)) | ||||||
|  |                     == -1) | ||||||
|  |                 return -1; | ||||||
|  |             state->x.next = state->out; | ||||||
|  |             return 0; | ||||||
|  |         case GZIP:      /* -> GZIP or LOOK (if end of gzip stream) */ | ||||||
|  |             strm->avail_out = state->size << 1; | ||||||
|  |             strm->next_out = state->out; | ||||||
|  |             if (gz_decomp(state) == -1) | ||||||
|  |                 return -1; | ||||||
|  |         } | ||||||
|  |     } while (state->x.have == 0 && (!state->eof || strm->avail_in)); | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Skip len uncompressed bytes of output.  Return -1 on error, 0 on success. */ | ||||||
|  | local int gz_skip(state, len) | ||||||
|  |     gz_statep state; | ||||||
|  |     z_off64_t len; | ||||||
|  | { | ||||||
|  |     unsigned n; | ||||||
|  |  | ||||||
|  |     /* skip over len bytes or reach end-of-file, whichever comes first */ | ||||||
|  |     while (len) | ||||||
|  |         /* skip over whatever is in output buffer */ | ||||||
|  |         if (state->x.have) { | ||||||
|  |             n = GT_OFF(state->x.have) || (z_off64_t)state->x.have > len ? | ||||||
|  |                 (unsigned)len : state->x.have; | ||||||
|  |             state->x.have -= n; | ||||||
|  |             state->x.next += n; | ||||||
|  |             state->x.pos += n; | ||||||
|  |             len -= n; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         /* output buffer empty -- return if we're at the end of the input */ | ||||||
|  |         else if (state->eof && state->strm.avail_in == 0) | ||||||
|  |             break; | ||||||
|  |  | ||||||
|  |         /* need more data to skip -- load up output buffer */ | ||||||
|  |         else { | ||||||
|  |             /* get more output, looking for header if required */ | ||||||
|  |             if (gz_fetch(state) == -1) | ||||||
|  |                 return -1; | ||||||
|  |         } | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* -- see zlib.h -- */ | ||||||
|  | int ZEXPORT gzread(file, buf, len) | ||||||
|  |     gzFile file; | ||||||
|  |     voidp buf; | ||||||
|  |     unsigned len; | ||||||
|  | { | ||||||
|  |     unsigned got, n; | ||||||
|  |     gz_statep state; | ||||||
|  |     z_streamp strm; | ||||||
|  |  | ||||||
|  |     /* get internal structure */ | ||||||
|  |     if (file == NULL) | ||||||
|  |         return -1; | ||||||
|  |     state = (gz_statep)file; | ||||||
|  |     strm = &(state->strm); | ||||||
|  |  | ||||||
|  |     /* check that we're reading and that there's no (serious) error */ | ||||||
|  |     if (state->mode != GZ_READ || | ||||||
|  |             (state->err != Z_OK && state->err != Z_BUF_ERROR)) | ||||||
|  |         return -1; | ||||||
|  |  | ||||||
|  |     /* since an int is returned, make sure len fits in one, otherwise return | ||||||
|  |        with an error (this avoids the flaw in the interface) */ | ||||||
|  |     if ((int)len < 0) { | ||||||
|  |         gz_error(state, Z_DATA_ERROR, "requested length does not fit in int"); | ||||||
|  |         return -1; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* if len is zero, avoid unnecessary operations */ | ||||||
|  |     if (len == 0) | ||||||
|  |         return 0; | ||||||
|  |  | ||||||
|  |     /* process a skip request */ | ||||||
|  |     if (state->seek) { | ||||||
|  |         state->seek = 0; | ||||||
|  |         if (gz_skip(state, state->skip) == -1) | ||||||
|  |             return -1; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* get len bytes to buf, or less than len if at the end */ | ||||||
|  |     got = 0; | ||||||
|  |     do { | ||||||
|  |         /* first just try copying data from the output buffer */ | ||||||
|  |         if (state->x.have) { | ||||||
|  |             n = state->x.have > len ? len : state->x.have; | ||||||
|  |             memcpy(buf, state->x.next, n); | ||||||
|  |             state->x.next += n; | ||||||
|  |             state->x.have -= n; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         /* output buffer empty -- return if we're at the end of the input */ | ||||||
|  |         else if (state->eof && strm->avail_in == 0) { | ||||||
|  |             state->past = 1;        /* tried to read past end */ | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         /* need output data -- for small len or new stream load up our output | ||||||
|  |            buffer */ | ||||||
|  |         else if (state->how == LOOK || len < (state->size << 1)) { | ||||||
|  |             /* get more output, looking for header if required */ | ||||||
|  |             if (gz_fetch(state) == -1) | ||||||
|  |                 return -1; | ||||||
|  |             continue;       /* no progress yet -- go back to copy above */ | ||||||
|  |             /* the copy above assures that we will leave with space in the | ||||||
|  |                output buffer, allowing at least one gzungetc() to succeed */ | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         /* large len -- read directly into user buffer */ | ||||||
|  |         else if (state->how == COPY) {      /* read directly */ | ||||||
|  |             if (gz_load(state, buf, len, &n) == -1) | ||||||
|  |                 return -1; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         /* large len -- decompress directly into user buffer */ | ||||||
|  |         else {  /* state->how == GZIP */ | ||||||
|  |             strm->avail_out = len; | ||||||
|  |             strm->next_out = buf; | ||||||
|  |             if (gz_decomp(state) == -1) | ||||||
|  |                 return -1; | ||||||
|  |             n = state->x.have; | ||||||
|  |             state->x.have = 0; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         /* update progress */ | ||||||
|  |         len -= n; | ||||||
|  |         buf = (char *)buf + n; | ||||||
|  |         got += n; | ||||||
|  |         state->x.pos += n; | ||||||
|  |     } while (len); | ||||||
|  |  | ||||||
|  |     /* return number of bytes read into user buffer (will fit in int) */ | ||||||
|  |     return (int)got; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* -- see zlib.h -- */ | ||||||
|  | #undef gzgetc | ||||||
|  | int ZEXPORT gzgetc(file) | ||||||
|  |     gzFile file; | ||||||
|  | { | ||||||
|  |     int ret; | ||||||
|  |     unsigned char buf[1]; | ||||||
|  |     gz_statep state; | ||||||
|  |  | ||||||
|  |     /* get internal structure */ | ||||||
|  |     if (file == NULL) | ||||||
|  |         return -1; | ||||||
|  |     state = (gz_statep)file; | ||||||
|  |  | ||||||
|  |     /* check that we're reading and that there's no (serious) error */ | ||||||
|  |     if (state->mode != GZ_READ || | ||||||
|  |         (state->err != Z_OK && state->err != Z_BUF_ERROR)) | ||||||
|  |         return -1; | ||||||
|  |  | ||||||
|  |     /* try output buffer (no need to check for skip request) */ | ||||||
|  |     if (state->x.have) { | ||||||
|  |         state->x.have--; | ||||||
|  |         state->x.pos++; | ||||||
|  |         return *(state->x.next)++; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* nothing there -- try gzread() */ | ||||||
|  |     ret = gzread(file, buf, 1); | ||||||
|  |     return ret < 1 ? -1 : buf[0]; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int ZEXPORT gzgetc_(file) | ||||||
|  | gzFile file; | ||||||
|  | { | ||||||
|  |     return gzgetc(file); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* -- see zlib.h -- */ | ||||||
|  | int ZEXPORT gzungetc(c, file) | ||||||
|  |     int c; | ||||||
|  |     gzFile file; | ||||||
|  | { | ||||||
|  |     gz_statep state; | ||||||
|  |  | ||||||
|  |     /* get internal structure */ | ||||||
|  |     if (file == NULL) | ||||||
|  |         return -1; | ||||||
|  |     state = (gz_statep)file; | ||||||
|  |  | ||||||
|  |     /* check that we're reading and that there's no (serious) error */ | ||||||
|  |     if (state->mode != GZ_READ || | ||||||
|  |         (state->err != Z_OK && state->err != Z_BUF_ERROR)) | ||||||
|  |         return -1; | ||||||
|  |  | ||||||
|  |     /* process a skip request */ | ||||||
|  |     if (state->seek) { | ||||||
|  |         state->seek = 0; | ||||||
|  |         if (gz_skip(state, state->skip) == -1) | ||||||
|  |             return -1; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* can't push EOF */ | ||||||
|  |     if (c < 0) | ||||||
|  |         return -1; | ||||||
|  |  | ||||||
|  |     /* if output buffer empty, put byte at end (allows more pushing) */ | ||||||
|  |     if (state->x.have == 0) { | ||||||
|  |         state->x.have = 1; | ||||||
|  |         state->x.next = state->out + (state->size << 1) - 1; | ||||||
|  |         state->x.next[0] = c; | ||||||
|  |         state->x.pos--; | ||||||
|  |         state->past = 0; | ||||||
|  |         return c; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* if no room, give up (must have already done a gzungetc()) */ | ||||||
|  |     if (state->x.have == (state->size << 1)) { | ||||||
|  |         gz_error(state, Z_DATA_ERROR, "out of room to push characters"); | ||||||
|  |         return -1; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* slide output data if needed and insert byte before existing data */ | ||||||
|  |     if (state->x.next == state->out) { | ||||||
|  |         unsigned char *src = state->out + state->x.have; | ||||||
|  |         unsigned char *dest = state->out + (state->size << 1); | ||||||
|  |         while (src > state->out) | ||||||
|  |             *--dest = *--src; | ||||||
|  |         state->x.next = dest; | ||||||
|  |     } | ||||||
|  |     state->x.have++; | ||||||
|  |     state->x.next--; | ||||||
|  |     state->x.next[0] = c; | ||||||
|  |     state->x.pos--; | ||||||
|  |     state->past = 0; | ||||||
|  |     return c; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* -- see zlib.h -- */ | ||||||
|  | char * ZEXPORT gzgets(file, buf, len) | ||||||
|  |     gzFile file; | ||||||
|  |     char *buf; | ||||||
|  |     int len; | ||||||
|  | { | ||||||
|  |     unsigned left, n; | ||||||
|  |     char *str; | ||||||
|  |     unsigned char *eol; | ||||||
|  |     gz_statep state; | ||||||
|  |  | ||||||
|  |     /* check parameters and get internal structure */ | ||||||
|  |     if (file == NULL || buf == NULL || len < 1) | ||||||
|  |         return NULL; | ||||||
|  |     state = (gz_statep)file; | ||||||
|  |  | ||||||
|  |     /* check that we're reading and that there's no (serious) error */ | ||||||
|  |     if (state->mode != GZ_READ || | ||||||
|  |         (state->err != Z_OK && state->err != Z_BUF_ERROR)) | ||||||
|  |         return NULL; | ||||||
|  |  | ||||||
|  |     /* process a skip request */ | ||||||
|  |     if (state->seek) { | ||||||
|  |         state->seek = 0; | ||||||
|  |         if (gz_skip(state, state->skip) == -1) | ||||||
|  |             return NULL; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* copy output bytes up to new line or len - 1, whichever comes first -- | ||||||
|  |        append a terminating zero to the string (we don't check for a zero in | ||||||
|  |        the contents, let the user worry about that) */ | ||||||
|  |     str = buf; | ||||||
|  |     left = (unsigned)len - 1; | ||||||
|  |     if (left) do { | ||||||
|  |         /* assure that something is in the output buffer */ | ||||||
|  |         if (state->x.have == 0 && gz_fetch(state) == -1) | ||||||
|  |             return NULL;                /* error */ | ||||||
|  |         if (state->x.have == 0) {       /* end of file */ | ||||||
|  |             state->past = 1;            /* read past end */ | ||||||
|  |             break;                      /* return what we have */ | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         /* look for end-of-line in current output buffer */ | ||||||
|  |         n = state->x.have > left ? left : state->x.have; | ||||||
|  |         eol = memchr(state->x.next, '\n', n); | ||||||
|  |         if (eol != NULL) | ||||||
|  |             n = (unsigned)(eol - state->x.next) + 1; | ||||||
|  |  | ||||||
|  |         /* copy through end-of-line, or remainder if not found */ | ||||||
|  |         memcpy(buf, state->x.next, n); | ||||||
|  |         state->x.have -= n; | ||||||
|  |         state->x.next += n; | ||||||
|  |         state->x.pos += n; | ||||||
|  |         left -= n; | ||||||
|  |         buf += n; | ||||||
|  |     } while (left && eol == NULL); | ||||||
|  |  | ||||||
|  |     /* return terminated string, or if nothing, end of file */ | ||||||
|  |     if (buf == str) | ||||||
|  |         return NULL; | ||||||
|  |     buf[0] = 0; | ||||||
|  |     return str; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* -- see zlib.h -- */ | ||||||
|  | int ZEXPORT gzdirect(file) | ||||||
|  |     gzFile file; | ||||||
|  | { | ||||||
|  |     gz_statep state; | ||||||
|  |  | ||||||
|  |     /* get internal structure */ | ||||||
|  |     if (file == NULL) | ||||||
|  |         return 0; | ||||||
|  |     state = (gz_statep)file; | ||||||
|  |  | ||||||
|  |     /* if the state is not known, but we can find out, then do so (this is | ||||||
|  |        mainly for right after a gzopen() or gzdopen()) */ | ||||||
|  |     if (state->mode == GZ_READ && state->how == LOOK && state->x.have == 0) | ||||||
|  |         (void)gz_look(state); | ||||||
|  |  | ||||||
|  |     /* return 1 if transparent, 0 if processing a gzip stream */ | ||||||
|  |     return state->direct; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* -- see zlib.h -- */ | ||||||
|  | int ZEXPORT gzclose_r(file) | ||||||
|  |     gzFile file; | ||||||
|  | { | ||||||
|  |     int ret, err; | ||||||
|  |     gz_statep state; | ||||||
|  |  | ||||||
|  |     /* get internal structure */ | ||||||
|  |     if (file == NULL) | ||||||
|  |         return Z_STREAM_ERROR; | ||||||
|  |     state = (gz_statep)file; | ||||||
|  |  | ||||||
|  |     /* check that we're reading */ | ||||||
|  |     if (state->mode != GZ_READ) | ||||||
|  |         return Z_STREAM_ERROR; | ||||||
|  |  | ||||||
|  |     /* free memory and close file */ | ||||||
|  |     if (state->size) { | ||||||
|  |         inflateEnd(&(state->strm)); | ||||||
|  |         free(state->out); | ||||||
|  |         free(state->in); | ||||||
|  |     } | ||||||
|  |     err = state->err == Z_BUF_ERROR ? Z_BUF_ERROR : Z_OK; | ||||||
|  |     gz_error(state, Z_OK, NULL); | ||||||
|  |     free(state->path); | ||||||
|  |     ret = close(state->fd); | ||||||
|  |     free(state); | ||||||
|  |     return ret ? Z_ERRNO : err; | ||||||
|  | } | ||||||
							
								
								
									
										565
									
								
								Foundation/src/gzwrite.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										565
									
								
								Foundation/src/gzwrite.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,565 @@ | |||||||
|  | /* gzwrite.c -- zlib functions for writing gzip files | ||||||
|  |  * Copyright (C) 2004, 2005, 2010, 2011, 2012 Mark Adler | ||||||
|  |  * For conditions of distribution and use, see copyright notice in zlib.h | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | #include "gzguts.h" | ||||||
|  |  | ||||||
|  | /* Local functions */ | ||||||
|  | local int gz_init OF((gz_statep)); | ||||||
|  | local int gz_comp OF((gz_statep, int)); | ||||||
|  | local int gz_zero OF((gz_statep, z_off64_t)); | ||||||
|  |  | ||||||
|  | /* Initialize state for writing a gzip file.  Mark initialization by setting | ||||||
|  |    state->size to non-zero.  Return -1 on failure or 0 on success. */ | ||||||
|  | local int gz_init(state) | ||||||
|  |     gz_statep state; | ||||||
|  | { | ||||||
|  |     int ret; | ||||||
|  |     z_streamp strm = &(state->strm); | ||||||
|  |  | ||||||
|  |     /* allocate input buffer */ | ||||||
|  |     state->in = malloc(state->want); | ||||||
|  |     if (state->in == NULL) { | ||||||
|  |         gz_error(state, Z_MEM_ERROR, "out of memory"); | ||||||
|  |         return -1; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* only need output buffer and deflate state if compressing */ | ||||||
|  |     if (!state->direct) { | ||||||
|  |         /* allocate output buffer */ | ||||||
|  |         state->out = malloc(state->want); | ||||||
|  |         if (state->out == NULL) { | ||||||
|  |             free(state->in); | ||||||
|  |             gz_error(state, Z_MEM_ERROR, "out of memory"); | ||||||
|  |             return -1; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         /* allocate deflate memory, set up for gzip compression */ | ||||||
|  |         strm->zalloc = Z_NULL; | ||||||
|  |         strm->zfree = Z_NULL; | ||||||
|  |         strm->opaque = Z_NULL; | ||||||
|  |         ret = deflateInit2(strm, state->level, Z_DEFLATED, | ||||||
|  |                            MAX_WBITS + 16, DEF_MEM_LEVEL, state->strategy); | ||||||
|  |         if (ret != Z_OK) { | ||||||
|  |             free(state->out); | ||||||
|  |             free(state->in); | ||||||
|  |             gz_error(state, Z_MEM_ERROR, "out of memory"); | ||||||
|  |             return -1; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* mark state as initialized */ | ||||||
|  |     state->size = state->want; | ||||||
|  |  | ||||||
|  |     /* initialize write buffer if compressing */ | ||||||
|  |     if (!state->direct) { | ||||||
|  |         strm->avail_out = state->size; | ||||||
|  |         strm->next_out = state->out; | ||||||
|  |         state->x.next = strm->next_out; | ||||||
|  |     } | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Compress whatever is at avail_in and next_in and write to the output file. | ||||||
|  |    Return -1 if there is an error writing to the output file, otherwise 0. | ||||||
|  |    flush is assumed to be a valid deflate() flush value.  If flush is Z_FINISH, | ||||||
|  |    then the deflate() state is reset to start a new gzip stream.  If gz->direct | ||||||
|  |    is true, then simply write to the output file without compressing, and | ||||||
|  |    ignore flush. */ | ||||||
|  | local int gz_comp(state, flush) | ||||||
|  |     gz_statep state; | ||||||
|  |     int flush; | ||||||
|  | { | ||||||
|  |     int ret, got; | ||||||
|  |     unsigned have; | ||||||
|  |     z_streamp strm = &(state->strm); | ||||||
|  |  | ||||||
|  |     /* allocate memory if this is the first time through */ | ||||||
|  |     if (state->size == 0 && gz_init(state) == -1) | ||||||
|  |         return -1; | ||||||
|  |  | ||||||
|  |     /* write directly if requested */ | ||||||
|  |     if (state->direct) { | ||||||
|  |         got = write(state->fd, strm->next_in, strm->avail_in); | ||||||
|  |         if (got < 0 || (unsigned)got != strm->avail_in) { | ||||||
|  |             gz_error(state, Z_ERRNO, zstrerror()); | ||||||
|  |             return -1; | ||||||
|  |         } | ||||||
|  |         strm->avail_in = 0; | ||||||
|  |         return 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* run deflate() on provided input until it produces no more output */ | ||||||
|  |     ret = Z_OK; | ||||||
|  |     do { | ||||||
|  |         /* write out current buffer contents if full, or if flushing, but if | ||||||
|  |            doing Z_FINISH then don't write until we get to Z_STREAM_END */ | ||||||
|  |         if (strm->avail_out == 0 || (flush != Z_NO_FLUSH && | ||||||
|  |             (flush != Z_FINISH || ret == Z_STREAM_END))) { | ||||||
|  |             have = (unsigned)(strm->next_out - state->x.next); | ||||||
|  |             if (have && ((got = write(state->fd, state->x.next, have)) < 0 || | ||||||
|  |                          (unsigned)got != have)) { | ||||||
|  |                 gz_error(state, Z_ERRNO, zstrerror()); | ||||||
|  |                 return -1; | ||||||
|  |             } | ||||||
|  |             if (strm->avail_out == 0) { | ||||||
|  |                 strm->avail_out = state->size; | ||||||
|  |                 strm->next_out = state->out; | ||||||
|  |             } | ||||||
|  |             state->x.next = strm->next_out; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         /* compress */ | ||||||
|  |         have = strm->avail_out; | ||||||
|  |         ret = deflate(strm, flush); | ||||||
|  |         if (ret == Z_STREAM_ERROR) { | ||||||
|  |             gz_error(state, Z_STREAM_ERROR, | ||||||
|  |                       "internal error: deflate stream corrupt"); | ||||||
|  |             return -1; | ||||||
|  |         } | ||||||
|  |         have -= strm->avail_out; | ||||||
|  |     } while (have); | ||||||
|  |  | ||||||
|  |     /* if that completed a deflate stream, allow another to start */ | ||||||
|  |     if (flush == Z_FINISH) | ||||||
|  |         deflateReset(strm); | ||||||
|  |  | ||||||
|  |     /* all done, no errors */ | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* Compress len zeros to output.  Return -1 on error, 0 on success. */ | ||||||
|  | local int gz_zero(state, len) | ||||||
|  |     gz_statep state; | ||||||
|  |     z_off64_t len; | ||||||
|  | { | ||||||
|  |     int first; | ||||||
|  |     unsigned n; | ||||||
|  |     z_streamp strm = &(state->strm); | ||||||
|  |  | ||||||
|  |     /* consume whatever's left in the input buffer */ | ||||||
|  |     if (strm->avail_in && gz_comp(state, Z_NO_FLUSH) == -1) | ||||||
|  |         return -1; | ||||||
|  |  | ||||||
|  |     /* compress len zeros (len guaranteed > 0) */ | ||||||
|  |     first = 1; | ||||||
|  |     while (len) { | ||||||
|  |         n = GT_OFF(state->size) || (z_off64_t)state->size > len ? | ||||||
|  |             (unsigned)len : state->size; | ||||||
|  |         if (first) { | ||||||
|  |             memset(state->in, 0, n); | ||||||
|  |             first = 0; | ||||||
|  |         } | ||||||
|  |         strm->avail_in = n; | ||||||
|  |         strm->next_in = state->in; | ||||||
|  |         state->x.pos += n; | ||||||
|  |         if (gz_comp(state, Z_NO_FLUSH) == -1) | ||||||
|  |             return -1; | ||||||
|  |         len -= n; | ||||||
|  |     } | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* -- see zlib.h -- */ | ||||||
|  | int ZEXPORT gzwrite(file, buf, len) | ||||||
|  |     gzFile file; | ||||||
|  |     voidpc buf; | ||||||
|  |     unsigned len; | ||||||
|  | { | ||||||
|  |     unsigned put = len; | ||||||
|  |     unsigned n; | ||||||
|  |     gz_statep state; | ||||||
|  |     z_streamp strm; | ||||||
|  |  | ||||||
|  |     /* get internal structure */ | ||||||
|  |     if (file == NULL) | ||||||
|  |         return 0; | ||||||
|  |     state = (gz_statep)file; | ||||||
|  |     strm = &(state->strm); | ||||||
|  |  | ||||||
|  |     /* check that we're writing and that there's no error */ | ||||||
|  |     if (state->mode != GZ_WRITE || state->err != Z_OK) | ||||||
|  |         return 0; | ||||||
|  |  | ||||||
|  |     /* since an int is returned, make sure len fits in one, otherwise return | ||||||
|  |        with an error (this avoids the flaw in the interface) */ | ||||||
|  |     if ((int)len < 0) { | ||||||
|  |         gz_error(state, Z_DATA_ERROR, "requested length does not fit in int"); | ||||||
|  |         return 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* if len is zero, avoid unnecessary operations */ | ||||||
|  |     if (len == 0) | ||||||
|  |         return 0; | ||||||
|  |  | ||||||
|  |     /* allocate memory if this is the first time through */ | ||||||
|  |     if (state->size == 0 && gz_init(state) == -1) | ||||||
|  |         return 0; | ||||||
|  |  | ||||||
|  |     /* check for seek request */ | ||||||
|  |     if (state->seek) { | ||||||
|  |         state->seek = 0; | ||||||
|  |         if (gz_zero(state, state->skip) == -1) | ||||||
|  |             return 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* for small len, copy to input buffer, otherwise compress directly */ | ||||||
|  |     if (len < state->size) { | ||||||
|  |         /* copy to input buffer, compress when full */ | ||||||
|  |         do { | ||||||
|  |             if (strm->avail_in == 0) | ||||||
|  |                 strm->next_in = state->in; | ||||||
|  |             n = state->size - strm->avail_in; | ||||||
|  |             if (n > len) | ||||||
|  |                 n = len; | ||||||
|  |             memcpy(strm->next_in + strm->avail_in, buf, n); | ||||||
|  |             strm->avail_in += n; | ||||||
|  |             state->x.pos += n; | ||||||
|  |             buf = (char *)buf + n; | ||||||
|  |             len -= n; | ||||||
|  |             if (len && gz_comp(state, Z_NO_FLUSH) == -1) | ||||||
|  |                 return 0; | ||||||
|  |         } while (len); | ||||||
|  |     } | ||||||
|  |     else { | ||||||
|  |         /* consume whatever's left in the input buffer */ | ||||||
|  |         if (strm->avail_in && gz_comp(state, Z_NO_FLUSH) == -1) | ||||||
|  |             return 0; | ||||||
|  |  | ||||||
|  |         /* directly compress user buffer to file */ | ||||||
|  |         strm->avail_in = len; | ||||||
|  |         strm->next_in = (voidp)buf; | ||||||
|  |         state->x.pos += len; | ||||||
|  |         if (gz_comp(state, Z_NO_FLUSH) == -1) | ||||||
|  |             return 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* input was all buffered or compressed (put will fit in int) */ | ||||||
|  |     return (int)put; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* -- see zlib.h -- */ | ||||||
|  | int ZEXPORT gzputc(file, c) | ||||||
|  |     gzFile file; | ||||||
|  |     int c; | ||||||
|  | { | ||||||
|  |     unsigned char buf[1]; | ||||||
|  |     gz_statep state; | ||||||
|  |     z_streamp strm; | ||||||
|  |  | ||||||
|  |     /* get internal structure */ | ||||||
|  |     if (file == NULL) | ||||||
|  |         return -1; | ||||||
|  |     state = (gz_statep)file; | ||||||
|  |     strm = &(state->strm); | ||||||
|  |  | ||||||
|  |     /* check that we're writing and that there's no error */ | ||||||
|  |     if (state->mode != GZ_WRITE || state->err != Z_OK) | ||||||
|  |         return -1; | ||||||
|  |  | ||||||
|  |     /* check for seek request */ | ||||||
|  |     if (state->seek) { | ||||||
|  |         state->seek = 0; | ||||||
|  |         if (gz_zero(state, state->skip) == -1) | ||||||
|  |             return -1; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* try writing to input buffer for speed (state->size == 0 if buffer not | ||||||
|  |        initialized) */ | ||||||
|  |     if (strm->avail_in < state->size) { | ||||||
|  |         if (strm->avail_in == 0) | ||||||
|  |             strm->next_in = state->in; | ||||||
|  |         strm->next_in[strm->avail_in++] = c; | ||||||
|  |         state->x.pos++; | ||||||
|  |         return c & 0xff; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* no room in buffer or not initialized, use gz_write() */ | ||||||
|  |     buf[0] = c; | ||||||
|  |     if (gzwrite(file, buf, 1) != 1) | ||||||
|  |         return -1; | ||||||
|  |     return c & 0xff; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* -- see zlib.h -- */ | ||||||
|  | int ZEXPORT gzputs(file, str) | ||||||
|  |     gzFile file; | ||||||
|  |     const char *str; | ||||||
|  | { | ||||||
|  |     int ret; | ||||||
|  |     unsigned len; | ||||||
|  |  | ||||||
|  |     /* write string */ | ||||||
|  |     len = (unsigned)strlen(str); | ||||||
|  |     ret = gzwrite(file, str, len); | ||||||
|  |     return ret == 0 && len != 0 ? -1 : ret; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #if defined(STDC) || defined(Z_HAVE_STDARG_H) | ||||||
|  | #include <stdarg.h> | ||||||
|  |  | ||||||
|  | /* -- see zlib.h -- */ | ||||||
|  | int ZEXPORTVA gzprintf (gzFile file, const char *format, ...) | ||||||
|  | { | ||||||
|  |     int size, len; | ||||||
|  |     gz_statep state; | ||||||
|  |     z_streamp strm; | ||||||
|  |     va_list va; | ||||||
|  |  | ||||||
|  |     /* get internal structure */ | ||||||
|  |     if (file == NULL) | ||||||
|  |         return -1; | ||||||
|  |     state = (gz_statep)file; | ||||||
|  |     strm = &(state->strm); | ||||||
|  |  | ||||||
|  |     /* check that we're writing and that there's no error */ | ||||||
|  |     if (state->mode != GZ_WRITE || state->err != Z_OK) | ||||||
|  |         return 0; | ||||||
|  |  | ||||||
|  |     /* make sure we have some buffer space */ | ||||||
|  |     if (state->size == 0 && gz_init(state) == -1) | ||||||
|  |         return 0; | ||||||
|  |  | ||||||
|  |     /* check for seek request */ | ||||||
|  |     if (state->seek) { | ||||||
|  |         state->seek = 0; | ||||||
|  |         if (gz_zero(state, state->skip) == -1) | ||||||
|  |             return 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* consume whatever's left in the input buffer */ | ||||||
|  |     if (strm->avail_in && gz_comp(state, Z_NO_FLUSH) == -1) | ||||||
|  |         return 0; | ||||||
|  |  | ||||||
|  |     /* do the printf() into the input buffer, put length in len */ | ||||||
|  |     size = (int)(state->size); | ||||||
|  |     state->in[size - 1] = 0; | ||||||
|  |     va_start(va, format); | ||||||
|  | #ifdef NO_vsnprintf | ||||||
|  | #  ifdef HAS_vsprintf_void | ||||||
|  |     (void)vsprintf((char *)(state->in), format, va); | ||||||
|  |     va_end(va); | ||||||
|  |     for (len = 0; len < size; len++) | ||||||
|  |         if (state->in[len] == 0) break; | ||||||
|  | #  else | ||||||
|  |     len = vsprintf((char *)(state->in), format, va); | ||||||
|  |     va_end(va); | ||||||
|  | #  endif | ||||||
|  | #else | ||||||
|  | #  ifdef HAS_vsnprintf_void | ||||||
|  |     (void)vsnprintf((char *)(state->in), size, format, va); | ||||||
|  |     va_end(va); | ||||||
|  |     len = strlen((char *)(state->in)); | ||||||
|  | #  else | ||||||
|  |     len = vsnprintf((char *)(state->in), size, format, va); | ||||||
|  |     va_end(va); | ||||||
|  | #  endif | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  |     /* check that printf() results fit in buffer */ | ||||||
|  |     if (len <= 0 || len >= (int)size || state->in[size - 1] != 0) | ||||||
|  |         return 0; | ||||||
|  |  | ||||||
|  |     /* update buffer and position, defer compression until needed */ | ||||||
|  |     strm->avail_in = (unsigned)len; | ||||||
|  |     strm->next_in = state->in; | ||||||
|  |     state->x.pos += len; | ||||||
|  |     return len; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #else /* !STDC && !Z_HAVE_STDARG_H */ | ||||||
|  |  | ||||||
|  | /* -- see zlib.h -- */ | ||||||
|  | int ZEXPORTVA gzprintf (file, format, a1, a2, a3, a4, a5, a6, a7, a8, a9, a10, | ||||||
|  |                        a11, a12, a13, a14, a15, a16, a17, a18, a19, a20) | ||||||
|  |     gzFile file; | ||||||
|  |     const char *format; | ||||||
|  |     int a1, a2, a3, a4, a5, a6, a7, a8, a9, a10, | ||||||
|  |         a11, a12, a13, a14, a15, a16, a17, a18, a19, a20; | ||||||
|  | { | ||||||
|  |     int size, len; | ||||||
|  |     gz_statep state; | ||||||
|  |     z_streamp strm; | ||||||
|  |  | ||||||
|  |     /* get internal structure */ | ||||||
|  |     if (file == NULL) | ||||||
|  |         return -1; | ||||||
|  |     state = (gz_statep)file; | ||||||
|  |     strm = &(state->strm); | ||||||
|  |  | ||||||
|  |     /* check that can really pass pointer in ints */ | ||||||
|  |     if (sizeof(int) != sizeof(void *)) | ||||||
|  |         return 0; | ||||||
|  |  | ||||||
|  |     /* check that we're writing and that there's no error */ | ||||||
|  |     if (state->mode != GZ_WRITE || state->err != Z_OK) | ||||||
|  |         return 0; | ||||||
|  |  | ||||||
|  |     /* make sure we have some buffer space */ | ||||||
|  |     if (state->size == 0 && gz_init(state) == -1) | ||||||
|  |         return 0; | ||||||
|  |  | ||||||
|  |     /* check for seek request */ | ||||||
|  |     if (state->seek) { | ||||||
|  |         state->seek = 0; | ||||||
|  |         if (gz_zero(state, state->skip) == -1) | ||||||
|  |             return 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* consume whatever's left in the input buffer */ | ||||||
|  |     if (strm->avail_in && gz_comp(state, Z_NO_FLUSH) == -1) | ||||||
|  |         return 0; | ||||||
|  |  | ||||||
|  |     /* do the printf() into the input buffer, put length in len */ | ||||||
|  |     size = (int)(state->size); | ||||||
|  |     state->in[size - 1] = 0; | ||||||
|  | #ifdef NO_snprintf | ||||||
|  | #  ifdef HAS_sprintf_void | ||||||
|  |     sprintf((char *)(state->in), format, a1, a2, a3, a4, a5, a6, a7, a8, | ||||||
|  |             a9, a10, a11, a12, a13, a14, a15, a16, a17, a18, a19, a20); | ||||||
|  |     for (len = 0; len < size; len++) | ||||||
|  |         if (state->in[len] == 0) break; | ||||||
|  | #  else | ||||||
|  |     len = sprintf((char *)(state->in), format, a1, a2, a3, a4, a5, a6, a7, a8, | ||||||
|  |                   a9, a10, a11, a12, a13, a14, a15, a16, a17, a18, a19, a20); | ||||||
|  | #  endif | ||||||
|  | #else | ||||||
|  | #  ifdef HAS_snprintf_void | ||||||
|  |     snprintf((char *)(state->in), size, format, a1, a2, a3, a4, a5, a6, a7, a8, | ||||||
|  |              a9, a10, a11, a12, a13, a14, a15, a16, a17, a18, a19, a20); | ||||||
|  |     len = strlen((char *)(state->in)); | ||||||
|  | #  else | ||||||
|  |     len = snprintf((char *)(state->in), size, format, a1, a2, a3, a4, a5, a6, | ||||||
|  |                    a7, a8, a9, a10, a11, a12, a13, a14, a15, a16, a17, a18, | ||||||
|  |                    a19, a20); | ||||||
|  | #  endif | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  |     /* check that printf() results fit in buffer */ | ||||||
|  |     if (len <= 0 || len >= (int)size || state->in[size - 1] != 0) | ||||||
|  |         return 0; | ||||||
|  |  | ||||||
|  |     /* update buffer and position, defer compression until needed */ | ||||||
|  |     strm->avail_in = (unsigned)len; | ||||||
|  |     strm->next_in = state->in; | ||||||
|  |     state->x.pos += len; | ||||||
|  |     return len; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | /* -- see zlib.h -- */ | ||||||
|  | int ZEXPORT gzflush(file, flush) | ||||||
|  |     gzFile file; | ||||||
|  |     int flush; | ||||||
|  | { | ||||||
|  |     gz_statep state; | ||||||
|  |  | ||||||
|  |     /* get internal structure */ | ||||||
|  |     if (file == NULL) | ||||||
|  |         return -1; | ||||||
|  |     state = (gz_statep)file; | ||||||
|  |  | ||||||
|  |     /* check that we're writing and that there's no error */ | ||||||
|  |     if (state->mode != GZ_WRITE || state->err != Z_OK) | ||||||
|  |         return Z_STREAM_ERROR; | ||||||
|  |  | ||||||
|  |     /* check flush parameter */ | ||||||
|  |     if (flush < 0 || flush > Z_FINISH) | ||||||
|  |         return Z_STREAM_ERROR; | ||||||
|  |  | ||||||
|  |     /* check for seek request */ | ||||||
|  |     if (state->seek) { | ||||||
|  |         state->seek = 0; | ||||||
|  |         if (gz_zero(state, state->skip) == -1) | ||||||
|  |             return -1; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* compress remaining data with requested flush */ | ||||||
|  |     gz_comp(state, flush); | ||||||
|  |     return state->err; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* -- see zlib.h -- */ | ||||||
|  | int ZEXPORT gzsetparams(file, level, strategy) | ||||||
|  |     gzFile file; | ||||||
|  |     int level; | ||||||
|  |     int strategy; | ||||||
|  | { | ||||||
|  |     gz_statep state; | ||||||
|  |     z_streamp strm; | ||||||
|  |  | ||||||
|  |     /* get internal structure */ | ||||||
|  |     if (file == NULL) | ||||||
|  |         return Z_STREAM_ERROR; | ||||||
|  |     state = (gz_statep)file; | ||||||
|  |     strm = &(state->strm); | ||||||
|  |  | ||||||
|  |     /* check that we're writing and that there's no error */ | ||||||
|  |     if (state->mode != GZ_WRITE || state->err != Z_OK) | ||||||
|  |         return Z_STREAM_ERROR; | ||||||
|  |  | ||||||
|  |     /* if no change is requested, then do nothing */ | ||||||
|  |     if (level == state->level && strategy == state->strategy) | ||||||
|  |         return Z_OK; | ||||||
|  |  | ||||||
|  |     /* check for seek request */ | ||||||
|  |     if (state->seek) { | ||||||
|  |         state->seek = 0; | ||||||
|  |         if (gz_zero(state, state->skip) == -1) | ||||||
|  |             return -1; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* change compression parameters for subsequent input */ | ||||||
|  |     if (state->size) { | ||||||
|  |         /* flush previous input with previous parameters before changing */ | ||||||
|  |         if (strm->avail_in && gz_comp(state, Z_PARTIAL_FLUSH) == -1) | ||||||
|  |             return state->err; | ||||||
|  |         deflateParams(strm, level, strategy); | ||||||
|  |     } | ||||||
|  |     state->level = level; | ||||||
|  |     state->strategy = strategy; | ||||||
|  |     return Z_OK; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* -- see zlib.h -- */ | ||||||
|  | int ZEXPORT gzclose_w(file) | ||||||
|  |     gzFile file; | ||||||
|  | { | ||||||
|  |     int ret = Z_OK; | ||||||
|  |     gz_statep state; | ||||||
|  |  | ||||||
|  |     /* get internal structure */ | ||||||
|  |     if (file == NULL) | ||||||
|  |         return Z_STREAM_ERROR; | ||||||
|  |     state = (gz_statep)file; | ||||||
|  |  | ||||||
|  |     /* check that we're writing */ | ||||||
|  |     if (state->mode != GZ_WRITE) | ||||||
|  |         return Z_STREAM_ERROR; | ||||||
|  |  | ||||||
|  |     /* check for seek request */ | ||||||
|  |     if (state->seek) { | ||||||
|  |         state->seek = 0; | ||||||
|  |         if (gz_zero(state, state->skip) == -1) | ||||||
|  |             ret = state->err; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* flush, free memory, and close file */ | ||||||
|  |     if (state->size) { | ||||||
|  |         if (gz_comp(state, Z_FINISH) == -1) | ||||||
|  |             ret = state->err; | ||||||
|  |         if (!state->direct) { | ||||||
|  |             (void)deflateEnd(&(state->strm)); | ||||||
|  |             free(state->out); | ||||||
|  |         } | ||||||
|  |         free(state->in); | ||||||
|  |     } | ||||||
|  |     gz_error(state, Z_OK, NULL); | ||||||
|  |     free(state->path); | ||||||
|  |     if (close(state->fd) == -1) | ||||||
|  |         ret = Z_ERRNO; | ||||||
|  |     free(state); | ||||||
|  |     return ret; | ||||||
|  | } | ||||||
							
								
								
									
										59
									
								
								Foundation/src/uncompr.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								Foundation/src/uncompr.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,59 @@ | |||||||
|  | /* uncompr.c -- decompress a memory buffer | ||||||
|  |  * Copyright (C) 1995-2003, 2010 Jean-loup Gailly. | ||||||
|  |  * For conditions of distribution and use, see copyright notice in zlib.h | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | /* @(#) $Id$ */ | ||||||
|  |  | ||||||
|  | #define ZLIB_INTERNAL | ||||||
|  | #include "zlib.h" | ||||||
|  |  | ||||||
|  | /* =========================================================================== | ||||||
|  |      Decompresses the source buffer into the destination buffer.  sourceLen is | ||||||
|  |    the byte length of the source buffer. Upon entry, destLen is the total | ||||||
|  |    size of the destination buffer, which must be large enough to hold the | ||||||
|  |    entire uncompressed data. (The size of the uncompressed data must have | ||||||
|  |    been saved previously by the compressor and transmitted to the decompressor | ||||||
|  |    by some mechanism outside the scope of this compression library.) | ||||||
|  |    Upon exit, destLen is the actual size of the compressed buffer. | ||||||
|  |  | ||||||
|  |      uncompress returns Z_OK if success, Z_MEM_ERROR if there was not | ||||||
|  |    enough memory, Z_BUF_ERROR if there was not enough room in the output | ||||||
|  |    buffer, or Z_DATA_ERROR if the input data was corrupted. | ||||||
|  | */ | ||||||
|  | int ZEXPORT uncompress (dest, destLen, source, sourceLen) | ||||||
|  |     Bytef *dest; | ||||||
|  |     uLongf *destLen; | ||||||
|  |     const Bytef *source; | ||||||
|  |     uLong sourceLen; | ||||||
|  | { | ||||||
|  |     z_stream stream; | ||||||
|  |     int err; | ||||||
|  |  | ||||||
|  |     stream.next_in = (Bytef*)source; | ||||||
|  |     stream.avail_in = (uInt)sourceLen; | ||||||
|  |     /* Check for source > 64K on 16-bit machine: */ | ||||||
|  |     if ((uLong)stream.avail_in != sourceLen) return Z_BUF_ERROR; | ||||||
|  |  | ||||||
|  |     stream.next_out = dest; | ||||||
|  |     stream.avail_out = (uInt)*destLen; | ||||||
|  |     if ((uLong)stream.avail_out != *destLen) return Z_BUF_ERROR; | ||||||
|  |  | ||||||
|  |     stream.zalloc = (alloc_func)0; | ||||||
|  |     stream.zfree = (free_func)0; | ||||||
|  |  | ||||||
|  |     err = inflateInit(&stream); | ||||||
|  |     if (err != Z_OK) return err; | ||||||
|  |  | ||||||
|  |     err = inflate(&stream, Z_FINISH); | ||||||
|  |     if (err != Z_STREAM_END) { | ||||||
|  |         inflateEnd(&stream); | ||||||
|  |         if (err == Z_NEED_DICT || (err == Z_BUF_ERROR && stream.avail_in == 0)) | ||||||
|  |             return Z_DATA_ERROR; | ||||||
|  |         return err; | ||||||
|  |     } | ||||||
|  |     *destLen = stream.total_out; | ||||||
|  |  | ||||||
|  |     err = inflateEnd(&stream); | ||||||
|  |     return err; | ||||||
|  | } | ||||||
		Reference in New Issue
	
	Block a user
	 Marian Krivos
					Marian Krivos