zfstream.cc 13.1 KB
Newer Older
M
Mark Adler 已提交
1 2
/*
 * A C++ I/O streams interface to the zlib gz* functions
M
Mark Adler 已提交
3
 *
M
Mark Adler 已提交
4 5
 * by Ludwig Schwardt <schwardt@sun.ac.za>
 * original version by Kevin Ruland <kevin@rodin.wustl.edu>
M
Mark Adler 已提交
6
 *
M
Mark Adler 已提交
7 8 9 10 11 12 13 14 15 16 17 18 19 20
 * This version is standard-compliant and compatible with gcc 3.x.
 */

#include "zfstream.h"
#include <cstring>          // for strcpy, strcat, strlen (mode strings)
#include <cstdio>           // for BUFSIZ

// Internal buffer sizes (default and "unbuffered" versions)
#define BIGBUFSIZE BUFSIZ
#define SMALLBUFSIZE 1

/*****************************************************************************/

// Default constructor
M
Mark Adler 已提交
21 22
gzfilebuf::gzfilebuf()
: file(NULL), io_mode(std::ios_base::openmode(0)), own_fd(false),
M
Mark Adler 已提交
23 24 25 26 27 28 29
  buffer(NULL), buffer_size(BIGBUFSIZE), own_buffer(true)
{
  // No buffers to start with
  this->disable_buffer();
}

// Destructor
M
Mark Adler 已提交
30
gzfilebuf::~gzfilebuf()
M
Mark Adler 已提交
31 32 33 34
{
  // Sync output buffer and close only if responsible for file
  // (i.e. attached streams should be left open at this stage)
  this->sync();
M
Mark Adler 已提交
35
  if (own_fd)
M
Mark Adler 已提交
36 37 38 39 40 41 42 43
    this->close();
  // Make sure internal buffer is deallocated
  this->disable_buffer();
}

// Set compression level and strategy
int
gzfilebuf::setcompression(int comp_level,
M
Mark Adler 已提交
44
                          int comp_strategy)
M
Mark Adler 已提交
45 46 47 48 49
{
  return gzsetparams(file, comp_level, comp_strategy);
}

// Open gzipped file
M
Mark Adler 已提交
50 51 52
gzfilebuf*
gzfilebuf::open(const char *name,
                std::ios_base::openmode mode)
M
Mark Adler 已提交
53 54
{
  // Fail if file already open
M
Mark Adler 已提交
55
  if (this->is_open())
M
Mark Adler 已提交
56 57
    return NULL;
  // Don't support simultaneous read/write access (yet)
M
Mark Adler 已提交
58
  if ((mode & std::ios_base::in) && (mode & std::ios_base::out))
M
Mark Adler 已提交
59
    return NULL;
M
Mark Adler 已提交
60

M
Mark Adler 已提交
61 62 63 64
  // Build mode string for gzopen and check it [27.8.1.3.2]
  char char_mode[6] = "\0\0\0\0\0";
  if (!this->open_mode(mode, char_mode))
    return NULL;
M
Mark Adler 已提交
65

M
Mark Adler 已提交
66
  // Attempt to open file
M
Mark Adler 已提交
67
  if ((file = gzopen(name, char_mode)) == NULL)
M
Mark Adler 已提交
68 69 70 71 72 73 74 75 76 77 78 79
    return NULL;

  // On success, allocate internal buffer and set flags
  this->enable_buffer();
  io_mode = mode;
  own_fd = true;
  return this;
}

// Attach to gzipped file
gzfilebuf*
gzfilebuf::attach(int fd,
M
Mark Adler 已提交
80
                  std::ios_base::openmode mode)
M
Mark Adler 已提交
81 82
{
  // Fail if file already open
M
Mark Adler 已提交
83
  if (this->is_open())
M
Mark Adler 已提交
84 85
    return NULL;
  // Don't support simultaneous read/write access (yet)
M
Mark Adler 已提交
86
  if ((mode & std::ios_base::in) && (mode & std::ios_base::out))
M
Mark Adler 已提交
87
    return NULL;
M
Mark Adler 已提交
88

M
Mark Adler 已提交
89 90 91 92
  // Build mode string for gzdopen and check it [27.8.1.3.2]
  char char_mode[6] = "\0\0\0\0\0";
  if (!this->open_mode(mode, char_mode))
    return NULL;
M
Mark Adler 已提交
93

M
Mark Adler 已提交
94 95 96
  // Attempt to attach to file
  if ((file = gzdopen(fd, char_mode)) == NULL)
    return NULL;
M
Mark Adler 已提交
97

M
Mark Adler 已提交
98 99 100 101 102 103 104 105 106
  // On success, allocate internal buffer and set flags
  this->enable_buffer();
  io_mode = mode;
  own_fd = false;
  return this;
}

// Close gzipped file
gzfilebuf*
M
Mark Adler 已提交
107
gzfilebuf::close()
M
Mark Adler 已提交
108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129
{
  // Fail immediately if no file is open
  if (!this->is_open())
    return NULL;
  // Assume success
  gzfilebuf* retval = this;
  // Attempt to sync and close gzipped file
  if (this->sync() == -1)
    retval = NULL;
  if (gzclose(file) < 0)
    retval = NULL;
  // File is now gone anyway (postcondition [27.8.1.3.8])
  file = NULL;
  own_fd = false;
  // Destroy internal buffer if it exists
  this->disable_buffer();
  return retval;
}

/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */

// Convert int open mode to mode string
M
Mark Adler 已提交
130 131 132
bool
gzfilebuf::open_mode(std::ios_base::openmode mode,
                     char* c_mode) const
M
Mark Adler 已提交
133 134 135 136 137 138
{
  bool testb = mode & std::ios_base::binary;
  bool testi = mode & std::ios_base::in;
  bool testo = mode & std::ios_base::out;
  bool testt = mode & std::ios_base::trunc;
  bool testa = mode & std::ios_base::app;
M
Mark Adler 已提交
139

M
Mark Adler 已提交
140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158
  // Check for valid flag combinations - see [27.8.1.3.2] (Table 92)
  // Original zfstream hardcoded the compression level to maximum here...
  // Double the time for less than 1% size improvement seems
  // excessive though - keeping it at the default level
  // To change back, just append "9" to the next three mode strings
  if (!testi && testo && !testt && !testa)
    strcpy(c_mode, "w");
  if (!testi && testo && !testt && testa)
    strcpy(c_mode, "a");
  if (!testi && testo && testt && !testa)
    strcpy(c_mode, "w");
  if (testi && !testo && !testt && !testa)
    strcpy(c_mode, "r");
  // No read/write mode yet
//  if (testi && testo && !testt && !testa)
//    strcpy(c_mode, "r+");
//  if (testi && testo && testt && !testa)
//    strcpy(c_mode, "w+");

M
Mark Adler 已提交
159
  // Mode string should be empty for invalid combination of flags
M
Mark Adler 已提交
160 161 162 163 164 165 166 167
  if (strlen(c_mode) == 0)
    return false;
  if (testb)
    strcat(c_mode, "b");
  return true;
}

// Determine number of characters in internal get buffer
M
Mark Adler 已提交
168
std::streamsize
M
Mark Adler 已提交
169 170 171 172 173 174 175 176 177 178 179 180 181 182
gzfilebuf::showmanyc()
{
  // Calls to underflow will fail if file not opened for reading
  if (!this->is_open() || !(io_mode & std::ios_base::in))
    return -1;
  // Make sure get area is in use
  if (this->gptr() && (this->gptr() < this->egptr()))
    return std::streamsize(this->egptr() - this->gptr());
  else
    return 0;
}

// Fill get area from gzipped file
gzfilebuf::int_type
M
Mark Adler 已提交
183
gzfilebuf::underflow()
M
Mark Adler 已提交
184 185
{
  // If something is left in the get area by chance, return it
M
Mark Adler 已提交
186
  // (this shouldn't normally happen, as underflow is only supposed
M
Mark Adler 已提交
187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206
  // to be called when gptr >= egptr, but it serves as error check)
  if (this->gptr() && (this->gptr() < this->egptr()))
    return traits_type::to_int_type(*(this->gptr()));

  // If the file hasn't been opened for reading, produce error
  if (!this->is_open() || !(io_mode & std::ios_base::in))
    return traits_type::eof();

  // Attempt to fill internal buffer from gzipped file
  // (buffer must be guaranteed to exist...)
  int bytes_read = gzread(file, buffer, buffer_size);
  // Indicates error or EOF
  if (bytes_read <= 0)
  {
    // Reset get area
    this->setg(buffer, buffer, buffer);
    return traits_type::eof();
  }
  // Make all bytes read from file available as get area
  this->setg(buffer, buffer, buffer + bytes_read);
M
Mark Adler 已提交
207

M
Mark Adler 已提交
208 209 210 211 212 213
  // Return next character in get area
  return traits_type::to_int_type(*(this->gptr()));
}

// Write put area to gzipped file
gzfilebuf::int_type
M
Mark Adler 已提交
214
gzfilebuf::overflow(int_type c)
M
Mark Adler 已提交
215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234
{
  // Determine whether put area is in use
  if (this->pbase())
  {
    // Double-check pointer range
    if (this->pptr() > this->epptr() || this->pptr() < this->pbase())
      return traits_type::eof();
    // Add extra character to buffer if not EOF
    if (!traits_type::eq_int_type(c, traits_type::eof()))
    {
      *(this->pptr()) = traits_type::to_char_type(c);
      this->pbump(1);
    }
    // Number of characters to write to file
    int bytes_to_write = this->pptr() - this->pbase();
    // Overflow doesn't fail if nothing is to be written
    if (bytes_to_write > 0)
    {
      // If the file hasn't been opened for writing, produce error
      if (!this->is_open() || !(io_mode & std::ios_base::out))
M
Mark Adler 已提交
235
        return traits_type::eof();
M
Mark Adler 已提交
236 237
      // If gzipped file won't accept all bytes written to it, fail
      if (gzwrite(file, this->pbase(), bytes_to_write) != bytes_to_write)
M
Mark Adler 已提交
238
        return traits_type::eof();
M
Mark Adler 已提交
239 240 241 242 243 244 245 246 247 248 249 250 251 252
      // Reset next pointer to point to pbase on success
      this->pbump(-bytes_to_write);
    }
  }
  // Write extra character to file if not EOF
  else if (!traits_type::eq_int_type(c, traits_type::eof()))
  {
    // If the file hasn't been opened for writing, produce error
    if (!this->is_open() || !(io_mode & std::ios_base::out))
      return traits_type::eof();
    // Impromptu char buffer (allows "unbuffered" output)
    char_type last_char = traits_type::to_char_type(c);
    // If gzipped file won't accept this character, fail
    if (gzwrite(file, &last_char, 1) != 1)
M
Mark Adler 已提交
253
      return traits_type::eof();
M
Mark Adler 已提交
254 255 256 257 258 259 260 261 262 263 264
  }

  // If you got here, you have succeeded (even if c was EOF)
  // The return value should therefore be non-EOF
  if (traits_type::eq_int_type(c, traits_type::eof()))
    return traits_type::not_eof(c);
  else
    return c;
}

// Assign new buffer
M
Mark Adler 已提交
265
std::streambuf*
M
Mark Adler 已提交
266
gzfilebuf::setbuf(char_type* p,
M
Mark Adler 已提交
267
                  std::streamsize n)
M
Mark Adler 已提交
268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297
{
  // First make sure stuff is sync'ed, for safety
  if (this->sync() == -1)
    return NULL;
  // If buffering is turned off on purpose via setbuf(0,0), still allocate one...
  // "Unbuffered" only really refers to put [27.8.1.4.10], while get needs at
  // least a buffer of size 1 (very inefficient though, therefore make it bigger?)
  // This follows from [27.5.2.4.3]/12 (gptr needs to point at something, it seems)
  if (!p || !n)
  {
    // Replace existing buffer (if any) with small internal buffer
    this->disable_buffer();
    buffer = NULL;
    buffer_size = 0;
    own_buffer = true;
    this->enable_buffer();
  }
  else
  {
    // Replace existing buffer (if any) with external buffer
    this->disable_buffer();
    buffer = p;
    buffer_size = n;
    own_buffer = false;
    this->enable_buffer();
  }
  return this;
}

// Write put area to gzipped file (i.e. ensures that put area is empty)
M
Mark Adler 已提交
298 299
int
gzfilebuf::sync()
M
Mark Adler 已提交
300 301 302 303 304 305 306
{
  return traits_type::eq_int_type(this->overflow(), traits_type::eof()) ? -1 : 0;
}

/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */

// Allocate internal buffer
M
Mark Adler 已提交
307 308
void
gzfilebuf::enable_buffer()
M
Mark Adler 已提交
309 310
{
  // If internal buffer required, allocate one
M
Mark Adler 已提交
311
  if (own_buffer && !buffer)
M
Mark Adler 已提交
312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337
  {
    // Check for buffered vs. "unbuffered"
    if (buffer_size > 0)
    {
      // Allocate internal buffer
      buffer = new char_type[buffer_size];
      // Get area starts empty and will be expanded by underflow as need arises
      this->setg(buffer, buffer, buffer);
      // Setup entire internal buffer as put area.
      // The one-past-end pointer actually points to the last element of the buffer,
      // so that overflow(c) can safely add the extra character c to the sequence.
      // These pointers remain in place for the duration of the buffer
      this->setp(buffer, buffer + buffer_size - 1);
    }
    else
    {
      // Even in "unbuffered" case, (small?) get buffer is still required
      buffer_size = SMALLBUFSIZE;
      buffer = new char_type[buffer_size];
      this->setg(buffer, buffer, buffer);
      // "Unbuffered" means no put buffer
      this->setp(0, 0);
    }
  }
  else
  {
M
Mark Adler 已提交
338
    // If buffer already allocated, reset buffer pointers just to make sure no
M
Mark Adler 已提交
339 340 341
    // stale chars are lying around
    this->setg(buffer, buffer, buffer);
    this->setp(buffer, buffer + buffer_size - 1);
M
Mark Adler 已提交
342
  }
M
Mark Adler 已提交
343 344 345
}

// Destroy internal buffer
M
Mark Adler 已提交
346 347
void
gzfilebuf::disable_buffer()
M
Mark Adler 已提交
348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373
{
  // If internal buffer exists, deallocate it
  if (own_buffer && buffer)
  {
    // Preserve unbuffered status by zeroing size
    if (!this->pbase())
      buffer_size = 0;
    delete[] buffer;
    buffer = NULL;
    this->setg(0, 0, 0);
    this->setp(0, 0);
  }
  else
  {
    // Reset buffer pointers to initial state if external buffer exists
    this->setg(buffer, buffer, buffer);
    if (buffer)
      this->setp(buffer, buffer + buffer_size - 1);
    else
      this->setp(0, 0);
  }
}

/*****************************************************************************/

// Default constructor initializes stream buffer
M
Mark Adler 已提交
374
gzifstream::gzifstream()
M
Mark Adler 已提交
375 376 377 378 379
: std::istream(NULL), sb()
{ this->init(&sb); }

// Initialize stream buffer and open file
gzifstream::gzifstream(const char* name,
M
Mark Adler 已提交
380
                       std::ios_base::openmode mode)
M
Mark Adler 已提交
381 382 383 384 385 386 387 388
: std::istream(NULL), sb()
{
  this->init(&sb);
  this->open(name, mode);
}

// Initialize stream buffer and attach to file
gzifstream::gzifstream(int fd,
M
Mark Adler 已提交
389
                       std::ios_base::openmode mode)
M
Mark Adler 已提交
390 391 392 393 394 395 396
: std::istream(NULL), sb()
{
  this->init(&sb);
  this->attach(fd, mode);
}

// Open file and go into fail() state if unsuccessful
M
Mark Adler 已提交
397 398 399
void
gzifstream::open(const char* name,
                 std::ios_base::openmode mode)
M
Mark Adler 已提交
400 401 402 403 404 405 406 407
{
  if (!sb.open(name, mode | std::ios_base::in))
    this->setstate(std::ios_base::failbit);
  else
    this->clear();
}

// Attach to file and go into fail() state if unsuccessful
M
Mark Adler 已提交
408 409 410
void
gzifstream::attach(int fd,
                   std::ios_base::openmode mode)
M
Mark Adler 已提交
411 412 413 414 415 416 417 418
{
  if (!sb.attach(fd, mode | std::ios_base::in))
    this->setstate(std::ios_base::failbit);
  else
    this->clear();
}

// Close file
M
Mark Adler 已提交
419
void
M
Mark Adler 已提交
420 421 422 423 424 425 426 427 428
gzifstream::close()
{
  if (!sb.close())
    this->setstate(std::ios_base::failbit);
}

/*****************************************************************************/

// Default constructor initializes stream buffer
M
Mark Adler 已提交
429
gzofstream::gzofstream()
M
Mark Adler 已提交
430 431 432 433 434
: std::ostream(NULL), sb()
{ this->init(&sb); }

// Initialize stream buffer and open file
gzofstream::gzofstream(const char* name,
M
Mark Adler 已提交
435
                       std::ios_base::openmode mode)
M
Mark Adler 已提交
436 437 438 439 440 441 442 443
: std::ostream(NULL), sb()
{
  this->init(&sb);
  this->open(name, mode);
}

// Initialize stream buffer and attach to file
gzofstream::gzofstream(int fd,
M
Mark Adler 已提交
444
                       std::ios_base::openmode mode)
M
Mark Adler 已提交
445 446 447 448 449 450 451
: std::ostream(NULL), sb()
{
  this->init(&sb);
  this->attach(fd, mode);
}

// Open file and go into fail() state if unsuccessful
M
Mark Adler 已提交
452 453 454
void
gzofstream::open(const char* name,
                 std::ios_base::openmode mode)
M
Mark Adler 已提交
455 456 457 458 459 460 461 462
{
  if (!sb.open(name, mode | std::ios_base::out))
    this->setstate(std::ios_base::failbit);
  else
    this->clear();
}

// Attach to file and go into fail() state if unsuccessful
M
Mark Adler 已提交
463 464 465
void
gzofstream::attach(int fd,
                   std::ios_base::openmode mode)
M
Mark Adler 已提交
466 467 468 469 470 471 472 473
{
  if (!sb.attach(fd, mode | std::ios_base::out))
    this->setstate(std::ios_base::failbit);
  else
    this->clear();
}

// Close file
M
Mark Adler 已提交
474
void
M
Mark Adler 已提交
475 476 477 478 479
gzofstream::close()
{
  if (!sb.close())
    this->setstate(std::ios_base::failbit);
}