xref: /netbsd-src/common/dist/zlib/contrib/iostream/zfstream.cpp (revision aaf4ece63a859a04e37cf3a7229b5fab0157cc06)
1*aaf4ece6Schristos 
2*aaf4ece6Schristos #include "zfstream.h"
3*aaf4ece6Schristos 
gzfilebuf()4*aaf4ece6Schristos gzfilebuf::gzfilebuf() :
5*aaf4ece6Schristos   file(NULL),
6*aaf4ece6Schristos   mode(0),
7*aaf4ece6Schristos   own_file_descriptor(0)
8*aaf4ece6Schristos { }
9*aaf4ece6Schristos 
~gzfilebuf()10*aaf4ece6Schristos gzfilebuf::~gzfilebuf() {
11*aaf4ece6Schristos 
12*aaf4ece6Schristos   sync();
13*aaf4ece6Schristos   if ( own_file_descriptor )
14*aaf4ece6Schristos     close();
15*aaf4ece6Schristos 
16*aaf4ece6Schristos }
17*aaf4ece6Schristos 
open(const char * name,int io_mode)18*aaf4ece6Schristos gzfilebuf *gzfilebuf::open( const char *name,
19*aaf4ece6Schristos                             int io_mode ) {
20*aaf4ece6Schristos 
21*aaf4ece6Schristos   if ( is_open() )
22*aaf4ece6Schristos     return NULL;
23*aaf4ece6Schristos 
24*aaf4ece6Schristos   char char_mode[10];
25*aaf4ece6Schristos   char *p = char_mode;
26*aaf4ece6Schristos 
27*aaf4ece6Schristos   if ( io_mode & ios::in ) {
28*aaf4ece6Schristos     mode = ios::in;
29*aaf4ece6Schristos     *p++ = 'r';
30*aaf4ece6Schristos   } else if ( io_mode & ios::app ) {
31*aaf4ece6Schristos     mode = ios::app;
32*aaf4ece6Schristos     *p++ = 'a';
33*aaf4ece6Schristos   } else {
34*aaf4ece6Schristos     mode = ios::out;
35*aaf4ece6Schristos     *p++ = 'w';
36*aaf4ece6Schristos   }
37*aaf4ece6Schristos 
38*aaf4ece6Schristos   if ( io_mode & ios::binary ) {
39*aaf4ece6Schristos     mode |= ios::binary;
40*aaf4ece6Schristos     *p++ = 'b';
41*aaf4ece6Schristos   }
42*aaf4ece6Schristos 
43*aaf4ece6Schristos   // Hard code the compression level
44*aaf4ece6Schristos   if ( io_mode & (ios::out|ios::app )) {
45*aaf4ece6Schristos     *p++ = '9';
46*aaf4ece6Schristos   }
47*aaf4ece6Schristos 
48*aaf4ece6Schristos   // Put the end-of-string indicator
49*aaf4ece6Schristos   *p = '\0';
50*aaf4ece6Schristos 
51*aaf4ece6Schristos   if ( (file = gzopen(name, char_mode)) == NULL )
52*aaf4ece6Schristos     return NULL;
53*aaf4ece6Schristos 
54*aaf4ece6Schristos   own_file_descriptor = 1;
55*aaf4ece6Schristos 
56*aaf4ece6Schristos   return this;
57*aaf4ece6Schristos 
58*aaf4ece6Schristos }
59*aaf4ece6Schristos 
attach(int file_descriptor,int io_mode)60*aaf4ece6Schristos gzfilebuf *gzfilebuf::attach( int file_descriptor,
61*aaf4ece6Schristos                               int io_mode ) {
62*aaf4ece6Schristos 
63*aaf4ece6Schristos   if ( is_open() )
64*aaf4ece6Schristos     return NULL;
65*aaf4ece6Schristos 
66*aaf4ece6Schristos   char char_mode[10];
67*aaf4ece6Schristos   char *p = char_mode;
68*aaf4ece6Schristos 
69*aaf4ece6Schristos   if ( io_mode & ios::in ) {
70*aaf4ece6Schristos     mode = ios::in;
71*aaf4ece6Schristos     *p++ = 'r';
72*aaf4ece6Schristos   } else if ( io_mode & ios::app ) {
73*aaf4ece6Schristos     mode = ios::app;
74*aaf4ece6Schristos     *p++ = 'a';
75*aaf4ece6Schristos   } else {
76*aaf4ece6Schristos     mode = ios::out;
77*aaf4ece6Schristos     *p++ = 'w';
78*aaf4ece6Schristos   }
79*aaf4ece6Schristos 
80*aaf4ece6Schristos   if ( io_mode & ios::binary ) {
81*aaf4ece6Schristos     mode |= ios::binary;
82*aaf4ece6Schristos     *p++ = 'b';
83*aaf4ece6Schristos   }
84*aaf4ece6Schristos 
85*aaf4ece6Schristos   // Hard code the compression level
86*aaf4ece6Schristos   if ( io_mode & (ios::out|ios::app )) {
87*aaf4ece6Schristos     *p++ = '9';
88*aaf4ece6Schristos   }
89*aaf4ece6Schristos 
90*aaf4ece6Schristos   // Put the end-of-string indicator
91*aaf4ece6Schristos   *p = '\0';
92*aaf4ece6Schristos 
93*aaf4ece6Schristos   if ( (file = gzdopen(file_descriptor, char_mode)) == NULL )
94*aaf4ece6Schristos     return NULL;
95*aaf4ece6Schristos 
96*aaf4ece6Schristos   own_file_descriptor = 0;
97*aaf4ece6Schristos 
98*aaf4ece6Schristos   return this;
99*aaf4ece6Schristos 
100*aaf4ece6Schristos }
101*aaf4ece6Schristos 
close()102*aaf4ece6Schristos gzfilebuf *gzfilebuf::close() {
103*aaf4ece6Schristos 
104*aaf4ece6Schristos   if ( is_open() ) {
105*aaf4ece6Schristos 
106*aaf4ece6Schristos     sync();
107*aaf4ece6Schristos     gzclose( file );
108*aaf4ece6Schristos     file = NULL;
109*aaf4ece6Schristos 
110*aaf4ece6Schristos   }
111*aaf4ece6Schristos 
112*aaf4ece6Schristos   return this;
113*aaf4ece6Schristos 
114*aaf4ece6Schristos }
115*aaf4ece6Schristos 
setcompressionlevel(int comp_level)116*aaf4ece6Schristos int gzfilebuf::setcompressionlevel( int comp_level ) {
117*aaf4ece6Schristos 
118*aaf4ece6Schristos   return gzsetparams(file, comp_level, -2);
119*aaf4ece6Schristos 
120*aaf4ece6Schristos }
121*aaf4ece6Schristos 
setcompressionstrategy(int comp_strategy)122*aaf4ece6Schristos int gzfilebuf::setcompressionstrategy( int comp_strategy ) {
123*aaf4ece6Schristos 
124*aaf4ece6Schristos   return gzsetparams(file, -2, comp_strategy);
125*aaf4ece6Schristos 
126*aaf4ece6Schristos }
127*aaf4ece6Schristos 
128*aaf4ece6Schristos 
seekoff(streamoff off,ios::seek_dir dir,int which)129*aaf4ece6Schristos streampos gzfilebuf::seekoff( streamoff off, ios::seek_dir dir, int which ) {
130*aaf4ece6Schristos 
131*aaf4ece6Schristos   return streampos(EOF);
132*aaf4ece6Schristos 
133*aaf4ece6Schristos }
134*aaf4ece6Schristos 
underflow()135*aaf4ece6Schristos int gzfilebuf::underflow() {
136*aaf4ece6Schristos 
137*aaf4ece6Schristos   // If the file hasn't been opened for reading, error.
138*aaf4ece6Schristos   if ( !is_open() || !(mode & ios::in) )
139*aaf4ece6Schristos     return EOF;
140*aaf4ece6Schristos 
141*aaf4ece6Schristos   // if a buffer doesn't exists, allocate one.
142*aaf4ece6Schristos   if ( !base() ) {
143*aaf4ece6Schristos 
144*aaf4ece6Schristos     if ( (allocate()) == EOF )
145*aaf4ece6Schristos       return EOF;
146*aaf4ece6Schristos     setp(0,0);
147*aaf4ece6Schristos 
148*aaf4ece6Schristos   } else {
149*aaf4ece6Schristos 
150*aaf4ece6Schristos     if ( in_avail() )
151*aaf4ece6Schristos       return (unsigned char) *gptr();
152*aaf4ece6Schristos 
153*aaf4ece6Schristos     if ( out_waiting() ) {
154*aaf4ece6Schristos       if ( flushbuf() == EOF )
155*aaf4ece6Schristos         return EOF;
156*aaf4ece6Schristos     }
157*aaf4ece6Schristos 
158*aaf4ece6Schristos   }
159*aaf4ece6Schristos 
160*aaf4ece6Schristos   // Attempt to fill the buffer.
161*aaf4ece6Schristos 
162*aaf4ece6Schristos   int result = fillbuf();
163*aaf4ece6Schristos   if ( result == EOF ) {
164*aaf4ece6Schristos     // disable get area
165*aaf4ece6Schristos     setg(0,0,0);
166*aaf4ece6Schristos     return EOF;
167*aaf4ece6Schristos   }
168*aaf4ece6Schristos 
169*aaf4ece6Schristos   return (unsigned char) *gptr();
170*aaf4ece6Schristos 
171*aaf4ece6Schristos }
172*aaf4ece6Schristos 
overflow(int c)173*aaf4ece6Schristos int gzfilebuf::overflow( int c ) {
174*aaf4ece6Schristos 
175*aaf4ece6Schristos   if ( !is_open() || !(mode & ios::out) )
176*aaf4ece6Schristos     return EOF;
177*aaf4ece6Schristos 
178*aaf4ece6Schristos   if ( !base() ) {
179*aaf4ece6Schristos     if ( allocate() == EOF )
180*aaf4ece6Schristos       return EOF;
181*aaf4ece6Schristos     setg(0,0,0);
182*aaf4ece6Schristos   } else {
183*aaf4ece6Schristos     if (in_avail()) {
184*aaf4ece6Schristos         return EOF;
185*aaf4ece6Schristos     }
186*aaf4ece6Schristos     if (out_waiting()) {
187*aaf4ece6Schristos       if (flushbuf() == EOF)
188*aaf4ece6Schristos         return EOF;
189*aaf4ece6Schristos     }
190*aaf4ece6Schristos   }
191*aaf4ece6Schristos 
192*aaf4ece6Schristos   int bl = blen();
193*aaf4ece6Schristos   setp( base(), base() + bl);
194*aaf4ece6Schristos 
195*aaf4ece6Schristos   if ( c != EOF ) {
196*aaf4ece6Schristos 
197*aaf4ece6Schristos     *pptr() = c;
198*aaf4ece6Schristos     pbump(1);
199*aaf4ece6Schristos 
200*aaf4ece6Schristos   }
201*aaf4ece6Schristos 
202*aaf4ece6Schristos   return 0;
203*aaf4ece6Schristos 
204*aaf4ece6Schristos }
205*aaf4ece6Schristos 
sync()206*aaf4ece6Schristos int gzfilebuf::sync() {
207*aaf4ece6Schristos 
208*aaf4ece6Schristos   if ( !is_open() )
209*aaf4ece6Schristos     return EOF;
210*aaf4ece6Schristos 
211*aaf4ece6Schristos   if ( out_waiting() )
212*aaf4ece6Schristos     return flushbuf();
213*aaf4ece6Schristos 
214*aaf4ece6Schristos   return 0;
215*aaf4ece6Schristos 
216*aaf4ece6Schristos }
217*aaf4ece6Schristos 
flushbuf()218*aaf4ece6Schristos int gzfilebuf::flushbuf() {
219*aaf4ece6Schristos 
220*aaf4ece6Schristos   int n;
221*aaf4ece6Schristos   char *q;
222*aaf4ece6Schristos 
223*aaf4ece6Schristos   q = pbase();
224*aaf4ece6Schristos   n = pptr() - q;
225*aaf4ece6Schristos 
226*aaf4ece6Schristos   if ( gzwrite( file, q, n) < n )
227*aaf4ece6Schristos     return EOF;
228*aaf4ece6Schristos 
229*aaf4ece6Schristos   setp(0,0);
230*aaf4ece6Schristos 
231*aaf4ece6Schristos   return 0;
232*aaf4ece6Schristos 
233*aaf4ece6Schristos }
234*aaf4ece6Schristos 
fillbuf()235*aaf4ece6Schristos int gzfilebuf::fillbuf() {
236*aaf4ece6Schristos 
237*aaf4ece6Schristos   int required;
238*aaf4ece6Schristos   char *p;
239*aaf4ece6Schristos 
240*aaf4ece6Schristos   p = base();
241*aaf4ece6Schristos 
242*aaf4ece6Schristos   required = blen();
243*aaf4ece6Schristos 
244*aaf4ece6Schristos   int t = gzread( file, p, required );
245*aaf4ece6Schristos 
246*aaf4ece6Schristos   if ( t <= 0) return EOF;
247*aaf4ece6Schristos 
248*aaf4ece6Schristos   setg( base(), base(), base()+t);
249*aaf4ece6Schristos 
250*aaf4ece6Schristos   return t;
251*aaf4ece6Schristos 
252*aaf4ece6Schristos }
253*aaf4ece6Schristos 
gzfilestream_common()254*aaf4ece6Schristos gzfilestream_common::gzfilestream_common() :
255*aaf4ece6Schristos   ios( gzfilestream_common::rdbuf() )
256*aaf4ece6Schristos { }
257*aaf4ece6Schristos 
~gzfilestream_common()258*aaf4ece6Schristos gzfilestream_common::~gzfilestream_common()
259*aaf4ece6Schristos { }
260*aaf4ece6Schristos 
attach(int fd,int io_mode)261*aaf4ece6Schristos void gzfilestream_common::attach( int fd, int io_mode ) {
262*aaf4ece6Schristos 
263*aaf4ece6Schristos   if ( !buffer.attach( fd, io_mode) )
264*aaf4ece6Schristos     clear( ios::failbit | ios::badbit );
265*aaf4ece6Schristos   else
266*aaf4ece6Schristos     clear();
267*aaf4ece6Schristos 
268*aaf4ece6Schristos }
269*aaf4ece6Schristos 
open(const char * name,int io_mode)270*aaf4ece6Schristos void gzfilestream_common::open( const char *name, int io_mode ) {
271*aaf4ece6Schristos 
272*aaf4ece6Schristos   if ( !buffer.open( name, io_mode ) )
273*aaf4ece6Schristos     clear( ios::failbit | ios::badbit );
274*aaf4ece6Schristos   else
275*aaf4ece6Schristos     clear();
276*aaf4ece6Schristos 
277*aaf4ece6Schristos }
278*aaf4ece6Schristos 
close()279*aaf4ece6Schristos void gzfilestream_common::close() {
280*aaf4ece6Schristos 
281*aaf4ece6Schristos   if ( !buffer.close() )
282*aaf4ece6Schristos     clear( ios::failbit | ios::badbit );
283*aaf4ece6Schristos 
284*aaf4ece6Schristos }
285*aaf4ece6Schristos 
rdbuf()286*aaf4ece6Schristos gzfilebuf *gzfilestream_common::rdbuf()
287*aaf4ece6Schristos {
288*aaf4ece6Schristos   return &buffer;
289*aaf4ece6Schristos }
290*aaf4ece6Schristos 
gzifstream()291*aaf4ece6Schristos gzifstream::gzifstream() :
292*aaf4ece6Schristos   ios( gzfilestream_common::rdbuf() )
293*aaf4ece6Schristos {
294*aaf4ece6Schristos   clear( ios::badbit );
295*aaf4ece6Schristos }
296*aaf4ece6Schristos 
gzifstream(const char * name,int io_mode)297*aaf4ece6Schristos gzifstream::gzifstream( const char *name, int io_mode ) :
298*aaf4ece6Schristos   ios( gzfilestream_common::rdbuf() )
299*aaf4ece6Schristos {
300*aaf4ece6Schristos   gzfilestream_common::open( name, io_mode );
301*aaf4ece6Schristos }
302*aaf4ece6Schristos 
gzifstream(int fd,int io_mode)303*aaf4ece6Schristos gzifstream::gzifstream( int fd, int io_mode ) :
304*aaf4ece6Schristos   ios( gzfilestream_common::rdbuf() )
305*aaf4ece6Schristos {
306*aaf4ece6Schristos   gzfilestream_common::attach( fd, io_mode );
307*aaf4ece6Schristos }
308*aaf4ece6Schristos 
~gzifstream()309*aaf4ece6Schristos gzifstream::~gzifstream() { }
310*aaf4ece6Schristos 
gzofstream()311*aaf4ece6Schristos gzofstream::gzofstream() :
312*aaf4ece6Schristos   ios( gzfilestream_common::rdbuf() )
313*aaf4ece6Schristos {
314*aaf4ece6Schristos   clear( ios::badbit );
315*aaf4ece6Schristos }
316*aaf4ece6Schristos 
gzofstream(const char * name,int io_mode)317*aaf4ece6Schristos gzofstream::gzofstream( const char *name, int io_mode ) :
318*aaf4ece6Schristos   ios( gzfilestream_common::rdbuf() )
319*aaf4ece6Schristos {
320*aaf4ece6Schristos   gzfilestream_common::open( name, io_mode );
321*aaf4ece6Schristos }
322*aaf4ece6Schristos 
gzofstream(int fd,int io_mode)323*aaf4ece6Schristos gzofstream::gzofstream( int fd, int io_mode ) :
324*aaf4ece6Schristos   ios( gzfilestream_common::rdbuf() )
325*aaf4ece6Schristos {
326*aaf4ece6Schristos   gzfilestream_common::attach( fd, io_mode );
327*aaf4ece6Schristos }
328*aaf4ece6Schristos 
~gzofstream()329*aaf4ece6Schristos gzofstream::~gzofstream() { }
330