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