• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /* gzlib.c -- zlib functions common to reading and writing gzip files
2  * Copyright (C) 2004, 2010 Mark Adler
3  * For conditions of distribution and use, see copyright notice in zlib.h
4  */
5 
6 #include "gzguts.h"
7 
8 #if defined(_WIN32) && !defined(__BORLANDC__)
9 #  define LSEEK _lseeki64
10 #elif defined(_LARGEFILE64_SOURCE) && _LFS64_LARGEFILE-0
11 #  define LSEEK lseek64
12 #else
13 #  define LSEEK lseek
14 #endif
15 
16 /* Local functions */
17 local void gz_reset OF((gz_statep));
18 local gzFile gz_open OF((const char *, int, const char *));
19 
20 #if defined UNDER_CE
21 
22 /* Map the Windows error number in ERROR to a locale-dependent error message
23    string and return a pointer to it.  Typically, the values for ERROR come
24    from GetLastError.
25 
26    The string pointed to shall not be modified by the application, but may be
27    overwritten by a subsequent call to gz_strwinerror
28 
29    The gz_strwinerror function does not change the current setting of
30    GetLastError. */
gz_strwinerror(error)31 char ZLIB_INTERNAL *gz_strwinerror (error)
32      DWORD error;
33 {
34     static char buf[1024];
35 
36     wchar_t *msgbuf;
37     DWORD lasterr = GetLastError();
38     DWORD chars = FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM
39         | FORMAT_MESSAGE_ALLOCATE_BUFFER,
40         NULL,
41         error,
42         0, /* Default language */
43         (LPVOID)&msgbuf,
44         0,
45         NULL);
46     if (chars != 0) {
47         /* If there is an \r\n appended, zap it.  */
48         if (chars >= 2
49             && msgbuf[chars - 2] == '\r' && msgbuf[chars - 1] == '\n') {
50             chars -= 2;
51             msgbuf[chars] = 0;
52         }
53 
54         if (chars > sizeof (buf) - 1) {
55             chars = sizeof (buf) - 1;
56             msgbuf[chars] = 0;
57         }
58 
59         wcstombs(buf, msgbuf, chars + 1);
60         LocalFree(msgbuf);
61     }
62     else {
63         sprintf(buf, "unknown win32 error (%ld)", error);
64     }
65 
66     SetLastError(lasterr);
67     return buf;
68 }
69 
70 #endif /* UNDER_CE */
71 
72 /* Reset gzip file state */
gz_reset(state)73 local void gz_reset(state)
74     gz_statep state;
75 {
76     if (state->mode == GZ_READ) {   /* for reading ... */
77         state->have = 0;            /* no output data available */
78         state->eof = 0;             /* not at end of file */
79         state->how = LOOK;          /* look for gzip header */
80         state->direct = 1;          /* default for empty file */
81     }
82     state->seek = 0;                /* no seek request pending */
83     gz_error(state, Z_OK, NULL);    /* clear error */
84     state->pos = 0;                 /* no uncompressed data yet */
85     state->strm.avail_in = 0;       /* no input data yet */
86 }
87 
88 /* Open a gzip file either by name or file descriptor. */
gz_open(path,fd,mode)89 local gzFile gz_open(path, fd, mode)
90     const char *path;
91     int fd;
92     const char *mode;
93 {
94     gz_statep state;
95 
96     /* allocate gzFile structure to return */
97     state = malloc(sizeof(gz_state));
98     if (state == NULL)
99         return NULL;
100     state->size = 0;            /* no buffers allocated yet */
101     state->want = GZBUFSIZE;    /* requested buffer size */
102     state->msg = NULL;          /* no error message yet */
103 
104     /* interpret mode */
105     state->mode = GZ_NONE;
106     state->level = Z_DEFAULT_COMPRESSION;
107     state->strategy = Z_DEFAULT_STRATEGY;
108     while (*mode) {
109         if (*mode >= '0' && *mode <= '9')
110             state->level = *mode - '0';
111         else
112             switch (*mode) {
113             case 'r':
114                 state->mode = GZ_READ;
115                 break;
116 #ifndef NO_GZCOMPRESS
117             case 'w':
118                 state->mode = GZ_WRITE;
119                 break;
120             case 'a':
121                 state->mode = GZ_APPEND;
122                 break;
123 #endif
124             case '+':       /* can't read and write at the same time */
125                 free(state);
126                 return NULL;
127             case 'b':       /* ignore -- will request binary anyway */
128                 break;
129             case 'f':
130                 state->strategy = Z_FILTERED;
131                 break;
132             case 'h':
133                 state->strategy = Z_HUFFMAN_ONLY;
134                 break;
135             case 'R':
136                 state->strategy = Z_RLE;
137                 break;
138             case 'F':
139                 state->strategy = Z_FIXED;
140             default:        /* could consider as an error, but just ignore */
141                 ;
142             }
143         mode++;
144     }
145 
146     /* must provide an "r", "w", or "a" */
147     if (state->mode == GZ_NONE) {
148         free(state);
149         return NULL;
150     }
151 
152     /* save the path name for error messages */
153     state->path = malloc(strlen(path) + 1);
154     if (state->path == NULL) {
155         free(state);
156         return NULL;
157     }
158     strcpy(state->path, path);
159 
160     /* open the file with the appropriate mode (or just use fd) */
161     state->fd = fd != -1 ? fd :
162         open(path,
163 #ifdef O_LARGEFILE
164             O_LARGEFILE |
165 #endif
166 #ifdef O_BINARY
167             O_BINARY |
168 #endif
169             (state->mode == GZ_READ ?
170                 O_RDONLY :
171                 (O_WRONLY | O_CREAT | (
172                     state->mode == GZ_WRITE ?
173                         O_TRUNC :
174                         O_APPEND))),
175             0666);
176     if (state->fd == -1) {
177         free(state->path);
178         free(state);
179         return NULL;
180     }
181     if (state->mode == GZ_APPEND)
182         state->mode = GZ_WRITE;         /* simplify later checks */
183 
184     /* save the current position for rewinding (only if reading) */
185     if (state->mode == GZ_READ) {
186         state->start = LSEEK(state->fd, 0, SEEK_CUR);
187         if (state->start == -1) state->start = 0;
188     }
189 
190     /* initialize stream */
191     gz_reset(state);
192 
193     /* return stream */
194     return (gzFile)state;
195 }
196 
197 /* -- see zlib.h -- */
gzopen(path,mode)198 gzFile ZEXPORT gzopen(path, mode)
199     const char *path;
200     const char *mode;
201 {
202     return gz_open(path, -1, mode);
203 }
204 
205 /* -- see zlib.h -- */
gzopen64(path,mode)206 gzFile ZEXPORT gzopen64(path, mode)
207     const char *path;
208     const char *mode;
209 {
210     return gz_open(path, -1, mode);
211 }
212 
213 /* -- see zlib.h -- */
gzdopen(fd,mode)214 gzFile ZEXPORT gzdopen(fd, mode)
215     int fd;
216     const char *mode;
217 {
218     char *path;         /* identifier for error messages */
219     gzFile gz;
220 
221     if (fd == -1 || (path = malloc(7 + 3 * sizeof(int))) == NULL)
222         return NULL;
223     sprintf(path, "<fd:%d>", fd);   /* for debugging */
224     gz = gz_open(path, fd, mode);
225     free(path);
226     return gz;
227 }
228 
229 /* -- see zlib.h -- */
gzbuffer(file,size)230 int ZEXPORT gzbuffer(file, size)
231     gzFile file;
232     unsigned size;
233 {
234     gz_statep state;
235 
236     /* get internal structure and check integrity */
237     if (file == NULL)
238         return -1;
239     state = (gz_statep)file;
240     if (state->mode != GZ_READ && state->mode != GZ_WRITE)
241         return -1;
242 
243     /* make sure we haven't already allocated memory */
244     if (state->size != 0)
245         return -1;
246 
247     /* check and set requested size */
248     if (size == 0)
249         return -1;
250     state->want = size;
251     return 0;
252 }
253 
254 /* -- see zlib.h -- */
gzrewind(file)255 int ZEXPORT gzrewind(file)
256     gzFile file;
257 {
258     gz_statep state;
259 
260     /* get internal structure */
261     if (file == NULL)
262         return -1;
263     state = (gz_statep)file;
264 
265     /* check that we're reading and that there's no error */
266     if (state->mode != GZ_READ || state->err != Z_OK)
267         return -1;
268 
269     /* back up and start over */
270     if (LSEEK(state->fd, state->start, SEEK_SET) == -1)
271         return -1;
272     gz_reset(state);
273     return 0;
274 }
275 
276 /* -- see zlib.h -- */
gzseek64(file,offset,whence)277 z_off64_t ZEXPORT gzseek64(file, offset, whence)
278     gzFile file;
279     z_off64_t offset;
280     int whence;
281 {
282     unsigned n;
283     z_off64_t ret;
284     gz_statep state;
285 
286     /* get internal structure and check integrity */
287     if (file == NULL)
288         return -1;
289     state = (gz_statep)file;
290     if (state->mode != GZ_READ && state->mode != GZ_WRITE)
291         return -1;
292 
293     /* check that there's no error */
294     if (state->err != Z_OK)
295         return -1;
296 
297     /* can only seek from start or relative to current position */
298     if (whence != SEEK_SET && whence != SEEK_CUR)
299         return -1;
300 
301     /* normalize offset to a SEEK_CUR specification */
302     if (whence == SEEK_SET)
303         offset -= state->pos;
304     else if (state->seek)
305         offset += state->skip;
306     state->seek = 0;
307 
308     /* if within raw area while reading, just go there */
309     if (state->mode == GZ_READ && state->how == COPY &&
310         state->pos + offset >= state->raw) {
311         ret = LSEEK(state->fd, offset - state->have, SEEK_CUR);
312         if (ret == -1)
313             return -1;
314         state->have = 0;
315         state->eof = 0;
316         state->seek = 0;
317         gz_error(state, Z_OK, NULL);
318         state->strm.avail_in = 0;
319         state->pos += offset;
320         return state->pos;
321     }
322 
323     /* calculate skip amount, rewinding if needed for back seek when reading */
324     if (offset < 0) {
325         if (state->mode != GZ_READ)         /* writing -- can't go backwards */
326             return -1;
327         offset += state->pos;
328         if (offset < 0)                     /* before start of file! */
329             return -1;
330         if (gzrewind(file) == -1)           /* rewind, then skip to offset */
331             return -1;
332     }
333 
334     /* if reading, skip what's in output buffer (one less gzgetc() check) */
335     if (state->mode == GZ_READ) {
336         n = GT_OFF(state->have) || (z_off64_t)state->have > offset ?
337             (unsigned)offset : state->have;
338         state->have -= n;
339         state->next += n;
340         state->pos += n;
341         offset -= n;
342     }
343 
344     /* request skip (if not zero) */
345     if (offset) {
346         state->seek = 1;
347         state->skip = offset;
348     }
349     return state->pos + offset;
350 }
351 
352 /* -- see zlib.h -- */
gzseek(file,offset,whence)353 z_off_t ZEXPORT gzseek(file, offset, whence)
354     gzFile file;
355     z_off_t offset;
356     int whence;
357 {
358     z_off64_t ret;
359 
360     ret = gzseek64(file, (z_off64_t)offset, whence);
361     return ret == (z_off_t)ret ? (z_off_t)ret : -1;
362 }
363 
364 /* -- see zlib.h -- */
gztell64(file)365 z_off64_t ZEXPORT gztell64(file)
366     gzFile file;
367 {
368     gz_statep state;
369 
370     /* get internal structure and check integrity */
371     if (file == NULL)
372         return -1;
373     state = (gz_statep)file;
374     if (state->mode != GZ_READ && state->mode != GZ_WRITE)
375         return -1;
376 
377     /* return position */
378     return state->pos + (state->seek ? state->skip : 0);
379 }
380 
381 /* -- see zlib.h -- */
gztell(file)382 z_off_t ZEXPORT gztell(file)
383     gzFile file;
384 {
385     z_off64_t ret;
386 
387     ret = gztell64(file);
388     return ret == (z_off_t)ret ? (z_off_t)ret : -1;
389 }
390 
391 /* -- see zlib.h -- */
gzoffset64(file)392 z_off64_t ZEXPORT gzoffset64(file)
393     gzFile file;
394 {
395     z_off64_t offset;
396     gz_statep state;
397 
398     /* get internal structure and check integrity */
399     if (file == NULL)
400         return -1;
401     state = (gz_statep)file;
402     if (state->mode != GZ_READ && state->mode != GZ_WRITE)
403         return -1;
404 
405     /* compute and return effective offset in file */
406     offset = LSEEK(state->fd, 0, SEEK_CUR);
407     if (offset == -1)
408         return -1;
409     if (state->mode == GZ_READ)             /* reading */
410         offset -= state->strm.avail_in;     /* don't count buffered input */
411     return offset;
412 }
413 
414 /* -- see zlib.h -- */
gzoffset(file)415 z_off_t ZEXPORT gzoffset(file)
416     gzFile file;
417 {
418     z_off64_t ret;
419 
420     ret = gzoffset64(file);
421     return ret == (z_off_t)ret ? (z_off_t)ret : -1;
422 }
423 
424 /* -- see zlib.h -- */
gzeof(file)425 int ZEXPORT gzeof(file)
426     gzFile file;
427 {
428     gz_statep state;
429 
430     /* get internal structure and check integrity */
431     if (file == NULL)
432         return 0;
433     state = (gz_statep)file;
434     if (state->mode != GZ_READ && state->mode != GZ_WRITE)
435         return 0;
436 
437     /* return end-of-file state */
438     return state->mode == GZ_READ ?
439         (state->eof && state->strm.avail_in == 0 && state->have == 0) : 0;
440 }
441 
442 /* -- see zlib.h -- */
gzerror(file,errnum)443 const char * ZEXPORT gzerror(file, errnum)
444     gzFile file;
445     int *errnum;
446 {
447     gz_statep state;
448 
449     /* get internal structure and check integrity */
450     if (file == NULL)
451         return NULL;
452     state = (gz_statep)file;
453     if (state->mode != GZ_READ && state->mode != GZ_WRITE)
454         return NULL;
455 
456     /* return error information */
457     if (errnum != NULL)
458         *errnum = state->err;
459     return state->msg == NULL ? "" : state->msg;
460 }
461 
462 /* -- see zlib.h -- */
gzclearerr(file)463 void ZEXPORT gzclearerr(file)
464     gzFile file;
465 {
466     gz_statep state;
467 
468     /* get internal structure and check integrity */
469     if (file == NULL)
470         return;
471     state = (gz_statep)file;
472     if (state->mode != GZ_READ && state->mode != GZ_WRITE)
473         return;
474 
475     /* clear error and end-of-file */
476     if (state->mode == GZ_READ)
477         state->eof = 0;
478     gz_error(state, Z_OK, NULL);
479 }
480 
481 /* Create an error message in allocated memory and set state->err and
482    state->msg accordingly.  Free any previous error message already there.  Do
483    not try to free or allocate space if the error is Z_MEM_ERROR (out of
484    memory).  Simply save the error message as a static string.  If there is an
485    allocation failure constructing the error message, then convert the error to
486    out of memory. */
gz_error(state,err,msg)487 void ZLIB_INTERNAL gz_error(state, err, msg)
488     gz_statep state;
489     int err;
490     const char *msg;
491 {
492     /* free previously allocated message and clear */
493     if (state->msg != NULL) {
494         if (state->err != Z_MEM_ERROR)
495             free(state->msg);
496         state->msg = NULL;
497     }
498 
499     /* set error code, and if no message, then done */
500     state->err = err;
501     if (msg == NULL)
502         return;
503 
504     /* for an out of memory error, save as static string */
505     if (err == Z_MEM_ERROR) {
506         state->msg = (char *)msg;
507         return;
508     }
509 
510     /* construct error message with path */
511     if ((state->msg = malloc(strlen(state->path) + strlen(msg) + 3)) == NULL) {
512         state->err = Z_MEM_ERROR;
513         state->msg = (char *)"out of memory";
514         return;
515     }
516     strcpy(state->msg, state->path);
517     strcat(state->msg, ": ");
518     strcat(state->msg, msg);
519     return;
520 }
521 
522 #ifndef INT_MAX
523 /* portably return maximum value for an int (when limits.h presumed not
524    available) -- we need to do this to cover cases where 2's complement not
525    used, since C standard permits 1's complement and sign-bit representations,
526    otherwise we could just use ((unsigned)-1) >> 1 */
gz_intmax()527 unsigned ZLIB_INTERNAL gz_intmax()
528 {
529     unsigned p, q;
530 
531     p = 1;
532     do {
533         q = p;
534         p <<= 1;
535         p++;
536     } while (p > q);
537     return q >> 1;
538 }
539 #endif
540