Coverage Report

Created: 2026-06-16 07:20

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/zlib/gzwrite.c
Line
Count
Source
1
/* gzwrite.c -- zlib functions for writing gzip files
2
 * Copyright (C) 2004-2026 Mark Adler
3
 * For conditions of distribution and use, see copyright notice in zlib.h
4
 */
5
6
#include "gzguts.h"
7
8
/* Initialize state for writing a gzip file.  Mark initialization by setting
9
   state->size to non-zero.  Return -1 on a memory allocation failure, or 0 on
10
   success. */
11
0
local int gz_init(gz_statep state) {
12
0
    int ret;
13
0
    z_streamp strm = &(state->strm);
14
15
    /* allocate input buffer (double size for gzprintf) */
16
0
    state->in = (unsigned char *)malloc(state->want << 1);
17
0
    if (state->in == NULL) {
18
0
        gz_error(state, Z_MEM_ERROR, "out of memory");
19
0
        return -1;
20
0
    }
21
22
    /* only need output buffer and deflate state if compressing */
23
0
    if (!state->direct) {
24
        /* allocate output buffer */
25
0
        state->out = (unsigned char *)malloc(state->want);
26
0
        if (state->out == NULL) {
27
0
            free(state->in);
28
0
            gz_error(state, Z_MEM_ERROR, "out of memory");
29
0
            return -1;
30
0
        }
31
32
        /* allocate deflate memory, set up for gzip compression */
33
0
        strm->zalloc = Z_NULL;
34
0
        strm->zfree = Z_NULL;
35
0
        strm->opaque = Z_NULL;
36
0
        ret = deflateInit2(strm, state->level, Z_DEFLATED,
37
0
                           MAX_WBITS + 16, DEF_MEM_LEVEL, state->strategy);
38
0
        if (ret != Z_OK) {
39
0
            free(state->out);
40
0
            free(state->in);
41
0
            gz_error(state, Z_MEM_ERROR, "out of memory");
42
0
            return -1;
43
0
        }
44
0
        strm->next_in = NULL;
45
0
    }
46
47
    /* mark state as initialized */
48
0
    state->size = state->want;
49
50
    /* initialize write buffer if compressing */
51
0
    if (!state->direct) {
52
0
        strm->avail_out = state->size;
53
0
        strm->next_out = state->out;
54
0
        state->x.next = strm->next_out;
55
0
    }
56
0
    return 0;
57
0
}
58
59
/* Compress whatever is at avail_in and next_in and write to the output file.
60
   Return -1 if there is an error writing to the output file or if gz_init()
61
   fails to allocate memory, otherwise 0.  flush is assumed to be a valid
62
   deflate() flush value.  If flush is Z_FINISH, then the deflate() state is
63
   reset to start a new gzip stream.  If gz->direct is true, then simply write
64
   to the output file without compressing, and ignore flush. */
65
0
local int gz_comp(gz_statep state, int flush) {
66
0
    int ret, writ;
67
0
    unsigned have, put, max = ((unsigned)-1 >> 2) + 1;
68
0
    z_streamp strm = &(state->strm);
69
70
    /* allocate memory if this is the first time through */
71
0
    if (state->size == 0 && gz_init(state) == -1)
72
0
        return -1;
73
74
    /* write directly if requested */
75
0
    if (state->direct) {
76
0
        while (strm->avail_in) {
77
0
            errno = 0;
78
0
            state->again = 0;
79
0
            put = strm->avail_in > max ? max : strm->avail_in;
80
0
            writ = (int)write(state->fd, strm->next_in, put);
81
0
            if (writ < 0) {
82
0
                if (errno == EAGAIN || errno == EWOULDBLOCK)
83
0
                    state->again = 1;
84
0
                gz_error(state, Z_ERRNO, zstrerror());
85
0
                return -1;
86
0
            }
87
0
            strm->avail_in -= (unsigned)writ;
88
0
            strm->next_in += writ;
89
0
        }
90
0
        return 0;
91
0
    }
92
93
    /* check for a pending reset */
94
0
    if (state->reset) {
95
        /* don't start a new gzip member unless there is data to write and
96
           we're not flushing */
97
0
        if (strm->avail_in == 0 && flush == Z_NO_FLUSH)
98
0
            return 0;
99
0
        deflateReset(strm);
100
0
        state->reset = 0;
101
0
    }
102
103
    /* run deflate() on provided input until it produces no more output */
104
0
    ret = Z_OK;
105
0
    do {
106
        /* write out current buffer contents if full, or if flushing, but if
107
           doing Z_FINISH then don't write until we get to Z_STREAM_END */
108
0
        if (strm->avail_out == 0 || (flush != Z_NO_FLUSH &&
109
0
            (flush != Z_FINISH || ret == Z_STREAM_END))) {
110
0
            while (strm->next_out > state->x.next) {
111
0
                errno = 0;
112
0
                state->again = 0;
113
0
                put = strm->next_out - state->x.next > (int)max ? max :
114
0
                      (unsigned)(strm->next_out - state->x.next);
115
0
                writ = (int)write(state->fd, state->x.next, put);
116
0
                if (writ < 0) {
117
0
                    if (errno == EAGAIN || errno == EWOULDBLOCK)
118
0
                        state->again = 1;
119
0
                    gz_error(state, Z_ERRNO, zstrerror());
120
0
                    return -1;
121
0
                }
122
0
                state->x.next += writ;
123
0
            }
124
0
            if (strm->avail_out == 0) {
125
0
                strm->avail_out = state->size;
126
0
                strm->next_out = state->out;
127
0
                state->x.next = state->out;
128
0
            }
129
0
        }
130
131
        /* compress */
132
0
        have = strm->avail_out;
133
0
        ret = deflate(strm, flush);
134
0
        if (ret == Z_STREAM_ERROR) {
135
0
            gz_error(state, Z_STREAM_ERROR,
136
0
                      "internal error: deflate stream corrupt");
137
0
            return -1;
138
0
        }
139
0
        have -= strm->avail_out;
140
0
    } while (have);
141
142
    /* if that completed a deflate stream, allow another to start */
143
0
    if (flush == Z_FINISH)
144
0
        state->reset = 1;
145
146
    /* all done, no errors */
147
0
    return 0;
148
0
}
149
150
/* Compress state->skip (> 0) zeros to output.  Return -1 on a write error or
151
   memory allocation failure by gz_comp(), or 0 on success. state->skip is
152
   updated with the number of successfully written zeros, in case there is a
153
   stall on a non-blocking write destination. */
154
0
local int gz_zero(gz_statep state) {
155
0
    int first, ret;
156
0
    unsigned n;
157
0
    z_streamp strm = &(state->strm);
158
159
    /* consume whatever's left in the input buffer */
160
0
    if (strm->avail_in && gz_comp(state, Z_NO_FLUSH) == -1)
161
0
        return -1;
162
163
    /* compress state->skip zeros */
164
0
    first = 1;
165
0
    do {
166
0
        n = GT_OFF(state->size) || (z_off64_t)state->size > state->skip ?
167
0
            (unsigned)state->skip : state->size;
168
0
        if (first) {
169
0
            memset(state->in, 0, n);
170
0
            first = 0;
171
0
        }
172
0
        strm->avail_in = n;
173
0
        strm->next_in = state->in;
174
0
        ret = gz_comp(state, Z_NO_FLUSH);
175
0
        n -= strm->avail_in;
176
0
        state->x.pos += n;
177
0
        state->skip -= n;
178
0
        if (ret == -1)
179
0
            return -1;
180
0
    } while (state->skip);
181
0
    return 0;
182
0
}
183
184
/* Write len bytes from buf to file.  Return the number of bytes written.  If
185
   the returned value is less than len, then there was an error. If the error
186
   was a non-blocking stall, then the number of bytes consumed is returned.
187
   For any other error, 0 is returned. */
188
0
local z_size_t gz_write(gz_statep state, voidpc buf, z_size_t len) {
189
0
    z_size_t put = len;
190
0
    int ret;
191
192
    /* if len is zero, avoid unnecessary operations */
193
0
    if (len == 0)
194
0
        return 0;
195
196
    /* allocate memory if this is the first time through */
197
0
    if (state->size == 0 && gz_init(state) == -1)
198
0
        return 0;
199
200
    /* check for seek request */
201
0
    if (state->skip && gz_zero(state) == -1)
202
0
        return 0;
203
204
    /* for small len, copy to input buffer, otherwise compress directly */
205
0
    if (len < state->size) {
206
        /* copy to input buffer, compress when full */
207
0
        for (;;) {
208
0
            unsigned have, copy;
209
210
0
            if (state->strm.avail_in == 0)
211
0
                state->strm.next_in = state->in;
212
0
            have = (unsigned)((state->strm.next_in + state->strm.avail_in) -
213
0
                              state->in);
214
0
            copy = state->size - have;
215
0
            if (copy > len)
216
0
                copy = (unsigned)len;
217
0
            memcpy(state->in + have, buf, copy);
218
0
            state->strm.avail_in += copy;
219
0
            state->x.pos += copy;
220
0
            buf = (const char *)buf + copy;
221
0
            len -= copy;
222
0
            if (len == 0)
223
0
                break;
224
0
            if (gz_comp(state, Z_NO_FLUSH) == -1)
225
0
                return state->again ? put - len : 0;
226
0
        }
227
0
    }
228
0
    else {
229
        /* consume whatever's left in the input buffer */
230
0
        if (state->strm.avail_in && gz_comp(state, Z_NO_FLUSH) == -1)
231
0
            return 0;
232
233
        /* directly compress user buffer to file */
234
0
        state->strm.next_in = (z_const Bytef *)buf;
235
0
        do {
236
0
            unsigned n = (unsigned)-1;
237
238
0
            if (n > len)
239
0
                n = (unsigned)len;
240
0
            state->strm.avail_in = n;
241
0
            ret = gz_comp(state, Z_NO_FLUSH);
242
0
            n -= state->strm.avail_in;
243
0
            state->x.pos += n;
244
0
            len -= n;
245
0
            if (ret == -1)
246
0
                return state->again ? put - len : 0;
247
0
        } while (len);
248
0
    }
249
250
    /* input was all buffered or compressed */
251
0
    return put;
252
0
}
253
254
/* -- see zlib.h -- */
255
0
int ZEXPORT gzwrite(gzFile file, voidpc buf, unsigned len) {
256
0
    gz_statep state;
257
258
    /* get internal structure */
259
0
    if (file == NULL)
260
0
        return 0;
261
0
    state = (gz_statep)file;
262
263
    /* check that we're writing and that there's no (serious) error */
264
0
    if (state->mode != GZ_WRITE || (state->err != Z_OK && !state->again))
265
0
        return 0;
266
0
    gz_error(state, Z_OK, NULL);
267
268
    /* since an int is returned, make sure len fits in one, otherwise return
269
       with an error (this avoids a flaw in the interface) */
270
0
    if ((int)len < 0) {
271
0
        gz_error(state, Z_DATA_ERROR, "requested length does not fit in int");
272
0
        return 0;
273
0
    }
274
275
    /* write len bytes from buf (the return value will fit in an int) */
276
0
    return (int)gz_write(state, buf, len);
277
0
}
278
279
/* -- see zlib.h -- */
280
z_size_t ZEXPORT gzfwrite(voidpc buf, z_size_t size, z_size_t nitems,
281
0
                          gzFile file) {
282
0
    z_size_t len;
283
0
    gz_statep state;
284
285
    /* get internal structure */
286
0
    if (file == NULL)
287
0
        return 0;
288
0
    state = (gz_statep)file;
289
290
    /* check that we're writing and that there's no (serious) error */
291
0
    if (state->mode != GZ_WRITE || (state->err != Z_OK && !state->again))
292
0
        return 0;
293
0
    gz_error(state, Z_OK, NULL);
294
295
    /* compute bytes to read -- error on overflow */
296
0
    len = nitems * size;
297
0
    if (size && len / size != nitems) {
298
0
        gz_error(state, Z_STREAM_ERROR, "request does not fit in a size_t");
299
0
        return 0;
300
0
    }
301
302
    /* write len bytes to buf, return the number of full items written */
303
0
    return len ? gz_write(state, buf, len) / size : 0;
304
0
}
305
306
/* -- see zlib.h -- */
307
0
int ZEXPORT gzputc(gzFile file, int c) {
308
0
    unsigned have;
309
0
    unsigned char buf[1];
310
0
    gz_statep state;
311
0
    z_streamp strm;
312
313
    /* get internal structure */
314
0
    if (file == NULL)
315
0
        return -1;
316
0
    state = (gz_statep)file;
317
0
    strm = &(state->strm);
318
319
    /* check that we're writing and that there's no (serious) error */
320
0
    if (state->mode != GZ_WRITE || (state->err != Z_OK && !state->again))
321
0
        return -1;
322
0
    gz_error(state, Z_OK, NULL);
323
324
    /* check for seek request */
325
0
    if (state->skip && gz_zero(state) == -1)
326
0
        return -1;
327
328
    /* try writing to input buffer for speed (state->size == 0 if buffer not
329
       initialized) */
330
0
    if (state->size) {
331
0
        if (strm->avail_in == 0)
332
0
            strm->next_in = state->in;
333
0
        have = (unsigned)((strm->next_in + strm->avail_in) - state->in);
334
0
        if (have < state->size) {
335
0
            state->in[have] = (unsigned char)c;
336
0
            strm->avail_in++;
337
0
            state->x.pos++;
338
0
            return c & 0xff;
339
0
        }
340
0
    }
341
342
    /* no room in buffer or not initialized, use gz_write() */
343
0
    buf[0] = (unsigned char)c;
344
0
    if (gz_write(state, buf, 1) != 1)
345
0
        return -1;
346
0
    return c & 0xff;
347
0
}
348
349
/* -- see zlib.h -- */
350
0
int ZEXPORT gzputs(gzFile file, const char *s) {
351
0
    z_size_t len, put;
352
0
    gz_statep state;
353
354
    /* get internal structure */
355
0
    if (file == NULL)
356
0
        return -1;
357
0
    state = (gz_statep)file;
358
359
    /* check that we're writing and that there's no (serious) error */
360
0
    if (state->mode != GZ_WRITE || (state->err != Z_OK && !state->again))
361
0
        return -1;
362
0
    gz_error(state, Z_OK, NULL);
363
364
    /* write string */
365
0
    len = strlen(s);
366
0
    if ((int)len < 0 || (unsigned)len != len) {
367
0
        gz_error(state, Z_STREAM_ERROR, "string length does not fit in int");
368
0
        return -1;
369
0
    }
370
0
    put = gz_write(state, s, len);
371
0
    return len && put == 0 ? -1 : (int)put;
372
0
}
373
374
#if (((!defined(STDC) && !defined(Z_HAVE_STDARG_H)) || !defined(NO_vsnprintf)) && \
375
     (defined(STDC) || defined(Z_HAVE_STDARG_H) || !defined(NO_snprintf))) || \
376
    defined(ZLIB_INSECURE)
377
/* If the second half of the input buffer is occupied, write out the contents.
378
   If there is any input remaining due to a non-blocking stall on write, move
379
   it to the start of the buffer. Return true if this did not open up the
380
   second half of the buffer.  state->err should be checked after this to
381
   handle a gz_comp() error. */
382
0
local int gz_vacate(gz_statep state) {
383
0
    z_streamp strm;
384
385
0
    strm = &(state->strm);
386
0
    if (strm->next_in == NULL ||
387
0
        strm->next_in + strm->avail_in <= state->in + state->size)
388
0
        return 0;
389
0
    (void)gz_comp(state, Z_NO_FLUSH);
390
0
    if (strm->avail_in == 0) {
391
0
        strm->next_in = state->in;
392
0
        return 0;
393
0
    }
394
0
    memmove(state->in, strm->next_in, strm->avail_in);
395
0
    strm->next_in = state->in;
396
0
    return strm->avail_in > state->size;
397
0
}
398
#endif
399
400
#if defined(STDC) || defined(Z_HAVE_STDARG_H)
401
#include <stdarg.h>
402
403
/* -- see zlib.h -- */
404
0
int ZEXPORTVA gzvprintf(gzFile file, const char *format, va_list va) {
405
#if defined(NO_vsnprintf) && !defined(ZLIB_INSECURE)
406
#warning "vsnprintf() not available -- gzprintf() stub returns Z_STREAM_ERROR"
407
#warning "you can recompile with ZLIB_INSECURE defined to use vsprintf()"
408
    /* prevent use of insecure vsprintf(), unless purposefully requested */
409
    (void)file, (void)format, (void)va;
410
    return Z_STREAM_ERROR;
411
#else
412
0
    int len, ret;
413
0
    char *next;
414
0
    gz_statep state;
415
0
    z_streamp strm;
416
417
    /* get internal structure */
418
0
    if (file == NULL)
419
0
        return Z_STREAM_ERROR;
420
0
    state = (gz_statep)file;
421
0
    strm = &(state->strm);
422
423
    /* check that we're writing and that there's no (serious) error */
424
0
    if (state->mode != GZ_WRITE || (state->err != Z_OK && !state->again))
425
0
        return Z_STREAM_ERROR;
426
0
    gz_error(state, Z_OK, NULL);
427
428
    /* make sure we have some buffer space */
429
0
    if (state->size == 0 && gz_init(state) == -1)
430
0
        return state->err;
431
432
    /* check for seek request */
433
0
    if (state->skip && gz_zero(state) == -1)
434
0
        return state->err;
435
436
    /* do the printf() into the input buffer, put length in len -- the input
437
       buffer is double-sized just for this function, so there should be
438
       state->size bytes available after the current contents */
439
0
    ret = gz_vacate(state);
440
0
    if (state->err) {
441
0
        if (ret && state->again) {
442
            /* There was a non-blocking stall on write, resulting in the part
443
               of the second half of the output buffer being occupied.  Return
444
               a Z_BUF_ERROR to let the application know that this gzprintf()
445
               needs to be retried. */
446
0
            gz_error(state, Z_BUF_ERROR, "stalled write on gzprintf");
447
0
        }
448
0
        if (!state->again)
449
0
            return state->err;
450
0
    }
451
0
    if (strm->avail_in == 0)
452
0
        strm->next_in = state->in;
453
0
    next = (char *)(state->in + (strm->next_in - state->in) + strm->avail_in);
454
0
    next[state->size - 1] = 0;
455
#ifdef NO_vsnprintf
456
#  ifdef HAS_vsprintf_void
457
    (void)vsprintf(next, format, va);
458
    for (len = 0; len < state->size; len++)
459
        if (next[len] == 0) break;
460
#  else
461
    len = vsprintf(next, format, va);
462
#  endif
463
#else
464
#  ifdef HAS_vsnprintf_void
465
    (void)vsnprintf(next, state->size, format, va);
466
    len = strlen(next);
467
#  else
468
0
    len = vsnprintf(next, state->size, format, va);
469
0
#  endif
470
0
#endif
471
472
    /* check that printf() results fit in buffer */
473
0
    if (len == 0 || (unsigned)len >= state->size || next[state->size - 1] != 0)
474
0
        return 0;
475
476
    /* update buffer and position */
477
0
    strm->avail_in += (unsigned)len;
478
0
    state->x.pos += len;
479
480
    /* write out buffer if more than half is occupied */
481
0
    ret = gz_vacate(state);
482
0
    if (state->err && !state->again)
483
0
        return state->err;
484
0
    return len;
485
0
#endif
486
0
}
487
488
0
int ZEXPORTVA gzprintf(gzFile file, const char *format, ...) {
489
0
    va_list va;
490
0
    int ret;
491
492
0
    va_start(va, format);
493
0
    ret = gzvprintf(file, format, va);
494
0
    va_end(va);
495
0
    return ret;
496
0
}
497
498
#else /* !STDC && !Z_HAVE_STDARG_H */
499
500
/* -- see zlib.h -- */
501
int ZEXPORTVA gzprintf(gzFile file, const char *format, int a1, int a2, int a3,
502
                       int a4, int a5, int a6, int a7, int a8, int a9, int a10,
503
                       int a11, int a12, int a13, int a14, int a15, int a16,
504
                       int a17, int a18, int a19, int a20) {
505
#if defined(NO_snprintf) && !defined(ZLIB_INSECURE)
506
#warning "snprintf() not available -- gzprintf() stub returns Z_STREAM_ERROR"
507
#warning "you can recompile with ZLIB_INSECURE defined to use sprintf()"
508
    /* prevent use of insecure sprintf(), unless purposefully requested */
509
    (void)file, (void)format, (void)a1, (void)a2, (void)a3, (void)a4, (void)a5,
510
    (void)a6, (void)a7, (void)a8, (void)a9, (void)a10, (void)a11, (void)a12,
511
    (void)a13, (void)a14, (void)a15, (void)a16, (void)a17, (void)a18,
512
    (void)a19, (void)a20;
513
    return Z_STREAM_ERROR;
514
#else
515
    int ret;
516
    unsigned len, left;
517
    char *next;
518
    gz_statep state;
519
    z_streamp strm;
520
521
    /* get internal structure */
522
    if (file == NULL)
523
        return Z_STREAM_ERROR;
524
    state = (gz_statep)file;
525
    strm = &(state->strm);
526
527
    /* check that can really pass pointer in ints */
528
    if (sizeof(int) != sizeof(void *))
529
        return Z_STREAM_ERROR;
530
531
    /* check that we're writing and that there's no (serious) error */
532
    if (state->mode != GZ_WRITE || (state->err != Z_OK && !state->again))
533
        return Z_STREAM_ERROR;
534
    gz_error(state, Z_OK, NULL);
535
536
    /* make sure we have some buffer space */
537
    if (state->size == 0 && gz_init(state) == -1)
538
        return state->err;
539
540
    /* check for seek request */
541
    if (state->skip && gz_zero(state) == -1)
542
        return state->err;
543
544
    /* do the printf() into the input buffer, put length in len -- the input
545
       buffer is double-sized just for this function, so there is guaranteed to
546
       be state->size bytes available after the current contents */
547
    ret = gz_vacate(state);
548
    if (state->err) {
549
        if (ret && state->again) {
550
            /* There was a non-blocking stall on write, resulting in the part
551
               of the second half of the output buffer being occupied.  Return
552
               a Z_BUF_ERROR to let the application know that this gzprintf()
553
               needs to be retried. */
554
            gz_error(state, Z_BUF_ERROR, "stalled write on gzprintf");
555
        }
556
        if (!state->again)
557
            return state->err;
558
    }
559
    if (strm->avail_in == 0)
560
        strm->next_in = state->in;
561
    next = (char *)(strm->next_in + strm->avail_in);
562
    next[state->size - 1] = 0;
563
#ifdef NO_snprintf
564
#  ifdef HAS_sprintf_void
565
    sprintf(next, format, a1, a2, a3, a4, a5, a6, a7, a8, a9, a10, a11, a12,
566
            a13, a14, a15, a16, a17, a18, a19, a20);
567
    for (len = 0; len < size; len++)
568
        if (next[len] == 0)
569
            break;
570
#  else
571
    len = sprintf(next, format, a1, a2, a3, a4, a5, a6, a7, a8, a9, a10, a11,
572
                  a12, a13, a14, a15, a16, a17, a18, a19, a20);
573
#  endif
574
#else
575
#  ifdef HAS_snprintf_void
576
    snprintf(next, state->size, format, a1, a2, a3, a4, a5, a6, a7, a8, a9,
577
             a10, a11, a12, a13, a14, a15, a16, a17, a18, a19, a20);
578
    len = strlen(next);
579
#  else
580
    len = snprintf(next, state->size, format, a1, a2, a3, a4, a5, a6, a7, a8,
581
                   a9, a10, a11, a12, a13, a14, a15, a16, a17, a18, a19, a20);
582
#  endif
583
#endif
584
585
    /* check that printf() results fit in buffer */
586
    if (len == 0 || len >= state->size || next[state->size - 1] != 0)
587
        return 0;
588
589
    /* update buffer and position, compress first half if past that */
590
    strm->avail_in += len;
591
    state->x.pos += len;
592
593
    /* write out buffer if more than half is occupied */
594
    ret = gz_vacate(state);
595
    if (state->err && !state->again)
596
        return state->err;
597
    return (int)len;
598
#endif
599
}
600
601
#endif
602
603
/* -- see zlib.h -- */
604
0
int ZEXPORT gzflush(gzFile file, int flush) {
605
0
    gz_statep state;
606
607
    /* get internal structure */
608
0
    if (file == NULL)
609
0
        return Z_STREAM_ERROR;
610
0
    state = (gz_statep)file;
611
612
    /* check that we're writing and that there's no (serious) error */
613
0
    if (state->mode != GZ_WRITE || (state->err != Z_OK && !state->again))
614
0
        return Z_STREAM_ERROR;
615
0
    gz_error(state, Z_OK, NULL);
616
617
    /* check flush parameter */
618
0
    if (flush < 0 || flush > Z_FINISH)
619
0
        return Z_STREAM_ERROR;
620
621
    /* check for seek request */
622
0
    if (state->skip && gz_zero(state) == -1)
623
0
        return state->err;
624
625
    /* compress remaining data with requested flush */
626
0
    (void)gz_comp(state, flush);
627
0
    return state->err;
628
0
}
629
630
/* -- see zlib.h -- */
631
0
int ZEXPORT gzsetparams(gzFile file, int level, int strategy) {
632
0
    gz_statep state;
633
0
    z_streamp strm;
634
635
    /* get internal structure */
636
0
    if (file == NULL)
637
0
        return Z_STREAM_ERROR;
638
0
    state = (gz_statep)file;
639
0
    strm = &(state->strm);
640
641
    /* check that we're compressing and that there's no (serious) error */
642
0
    if (state->mode != GZ_WRITE || (state->err != Z_OK && !state->again) ||
643
0
            state->direct)
644
0
        return Z_STREAM_ERROR;
645
0
    gz_error(state, Z_OK, NULL);
646
647
    /* if no change is requested, then do nothing */
648
0
    if (level == state->level && strategy == state->strategy)
649
0
        return Z_OK;
650
651
    /* check for seek request */
652
0
    if (state->skip && gz_zero(state) == -1)
653
0
        return state->err;
654
655
    /* change compression parameters for subsequent input */
656
0
    if (state->size) {
657
        /* flush previous input with previous parameters before changing */
658
0
        if (strm->avail_in && gz_comp(state, Z_BLOCK) == -1)
659
0
            return state->err;
660
0
        deflateParams(strm, level, strategy);
661
0
    }
662
0
    state->level = level;
663
0
    state->strategy = strategy;
664
0
    return Z_OK;
665
0
}
666
667
/* -- see zlib.h -- */
668
0
int ZEXPORT gzclose_w(gzFile file) {
669
0
    int ret = Z_OK;
670
0
    gz_statep state;
671
672
    /* get internal structure */
673
0
    if (file == NULL)
674
0
        return Z_STREAM_ERROR;
675
0
    state = (gz_statep)file;
676
677
    /* check that we're writing */
678
0
    if (state->mode != GZ_WRITE)
679
0
        return Z_STREAM_ERROR;
680
681
    /* check for seek request */
682
0
    if (state->skip && gz_zero(state) == -1)
683
0
        ret = state->err;
684
685
    /* flush, free memory, and close file */
686
0
    if (gz_comp(state, Z_FINISH) == -1)
687
0
        ret = state->err;
688
0
    if (state->size) {
689
0
        if (!state->direct) {
690
0
            (void)deflateEnd(&(state->strm));
691
0
            free(state->out);
692
0
        }
693
0
        free(state->in);
694
0
    }
695
0
    gz_error(state, Z_OK, NULL);
696
0
    free(state->path);
697
0
    if (close(state->fd) == -1)
698
0
        ret = Z_ERRNO;
699
0
    free(state);
700
0
    return ret;
701
0
}