LCOV - differential code coverage report
Current view: top level - src/bin/pg_dump - compress_gzip.c (source / functions) Coverage Total Hit UNC UBC CBC DUB
Current: Differential Code Coverage 16@8cea358b128 vs 17@8cea358b128 Lines: 81.4 % 172 140 1 31 140 1
Current Date: 2024-04-14 14:21:10 Functions: 88.2 % 17 15 1 1 15
Baseline: 16@8cea358b128 Branches: 57.1 % 70 40 30 40
Baseline Date: 2024-04-14 14:21:09 Line coverage date bins:
Legend: Lines: hit not hit | Branches: + taken - not taken # not executed [..60] days: 0.0 % 1 0 1
(240..) days: 81.9 % 171 140 31 140
Function coverage date bins:
(240..) days: 88.2 % 17 15 1 1 15
Branch coverage date bins:
(240..) days: 57.1 % 70 40 30 40

 Age         Owner                    Branch data    TLA  Line data    Source code
                                  1                 :                : /*-------------------------------------------------------------------------
                                  2                 :                :  *
                                  3                 :                :  * compress_gzip.c
                                  4                 :                :  *   Routines for archivers to read or write a gzip compressed data stream.
                                  5                 :                :  *
                                  6                 :                :  * Portions Copyright (c) 1996-2024, PostgreSQL Global Development Group
                                  7                 :                :  * Portions Copyright (c) 1994, Regents of the University of California
                                  8                 :                :  *
                                  9                 :                :  * IDENTIFICATION
                                 10                 :                :  *     src/bin/pg_dump/compress_gzip.c
                                 11                 :                :  *
                                 12                 :                :  *-------------------------------------------------------------------------
                                 13                 :                :  */
                                 14                 :                : #include "postgres_fe.h"
                                 15                 :                : #include <unistd.h>
                                 16                 :                : 
                                 17                 :                : #include "compress_gzip.h"
                                 18                 :                : #include "pg_backup_utils.h"
                                 19                 :                : 
                                 20                 :                : #ifdef HAVE_LIBZ
                                 21                 :                : #include "zlib.h"
                                 22                 :                : 
                                 23                 :                : /*----------------------
                                 24                 :                :  * Compressor API
                                 25                 :                :  *----------------------
                                 26                 :                :  */
                                 27                 :                : typedef struct GzipCompressorState
                                 28                 :                : {
                                 29                 :                :     z_streamp   zp;
                                 30                 :                : 
                                 31                 :                :     void       *outbuf;
                                 32                 :                :     size_t      outsize;
                                 33                 :                : } GzipCompressorState;
                                 34                 :                : 
                                 35                 :                : /* Private routines that support gzip compressed data I/O */
                                 36                 :                : static void DeflateCompressorInit(CompressorState *cs);
                                 37                 :                : static void DeflateCompressorEnd(ArchiveHandle *AH, CompressorState *cs);
                                 38                 :                : static void DeflateCompressorCommon(ArchiveHandle *AH, CompressorState *cs,
                                 39                 :                :                                     bool flush);
                                 40                 :                : static void EndCompressorGzip(ArchiveHandle *AH, CompressorState *cs);
                                 41                 :                : static void WriteDataToArchiveGzip(ArchiveHandle *AH, CompressorState *cs,
                                 42                 :                :                                    const void *data, size_t dLen);
                                 43                 :                : static void ReadDataFromArchiveGzip(ArchiveHandle *AH, CompressorState *cs);
                                 44                 :                : 
                                 45                 :                : static void
  382 tomas.vondra@postgre       46                 :CBC          79 : DeflateCompressorInit(CompressorState *cs)
                                 47                 :                : {
                                 48                 :                :     GzipCompressorState *gzipcs;
                                 49                 :                :     z_streamp   zp;
                                 50                 :                : 
                                 51                 :             79 :     gzipcs = (GzipCompressorState *) pg_malloc0(sizeof(GzipCompressorState));
                                 52                 :             79 :     zp = gzipcs->zp = (z_streamp) pg_malloc(sizeof(z_stream));
                                 53                 :             79 :     zp->zalloc = Z_NULL;
                                 54                 :             79 :     zp->zfree = Z_NULL;
                                 55                 :             79 :     zp->opaque = Z_NULL;
                                 56                 :                : 
                                 57                 :                :     /*
                                 58                 :                :      * outsize is the buffer size we tell zlib it can output to.  We actually
                                 59                 :                :      * allocate one extra byte because some routines want to append a trailing
                                 60                 :                :      * zero byte to the zlib output.
                                 61                 :                :      */
                                 62                 :             79 :     gzipcs->outsize = DEFAULT_IO_BUFFER_SIZE;
                                 63                 :             79 :     gzipcs->outbuf = pg_malloc(gzipcs->outsize + 1);
                                 64                 :                : 
                                 65                 :                :     /* -Z 0 uses the "None" compressor -- not zlib with no compression */
                                 66         [ -  + ]:             79 :     Assert(cs->compression_spec.level != 0);
                                 67                 :                : 
                                 68         [ -  + ]:             79 :     if (deflateInit(zp, cs->compression_spec.level) != Z_OK)
  382 tomas.vondra@postgre       69                 :UBC           0 :         pg_fatal("could not initialize compression library: %s", zp->msg);
                                 70                 :                : 
                                 71                 :                :     /* Just be paranoid - maybe End is called after Start, with no Write */
  382 tomas.vondra@postgre       72                 :CBC          79 :     zp->next_out = gzipcs->outbuf;
                                 73                 :             79 :     zp->avail_out = gzipcs->outsize;
                                 74                 :                : 
                                 75                 :                :     /* Keep track of gzipcs */
                                 76                 :             79 :     cs->private_data = gzipcs;
                                 77                 :             79 : }
                                 78                 :                : 
                                 79                 :                : static void
                                 80                 :             79 : DeflateCompressorEnd(ArchiveHandle *AH, CompressorState *cs)
                                 81                 :                : {
                                 82                 :             79 :     GzipCompressorState *gzipcs = (GzipCompressorState *) cs->private_data;
                                 83                 :                :     z_streamp   zp;
                                 84                 :                : 
                                 85                 :             79 :     zp = gzipcs->zp;
                                 86                 :             79 :     zp->next_in = NULL;
                                 87                 :             79 :     zp->avail_in = 0;
                                 88                 :                : 
                                 89                 :                :     /* Flush any remaining data from zlib buffer */
                                 90                 :             79 :     DeflateCompressorCommon(AH, cs, true);
                                 91                 :                : 
                                 92         [ -  + ]:             79 :     if (deflateEnd(zp) != Z_OK)
  382 tomas.vondra@postgre       93                 :UBC           0 :         pg_fatal("could not close compression stream: %s", zp->msg);
                                 94                 :                : 
  382 tomas.vondra@postgre       95                 :CBC          79 :     pg_free(gzipcs->outbuf);
                                 96                 :             79 :     pg_free(gzipcs->zp);
                                 97                 :             79 :     pg_free(gzipcs);
                                 98                 :             79 :     cs->private_data = NULL;
                                 99                 :             79 : }
                                100                 :                : 
                                101                 :                : static void
                                102                 :            221 : DeflateCompressorCommon(ArchiveHandle *AH, CompressorState *cs, bool flush)
                                103                 :                : {
  416                           104                 :            221 :     GzipCompressorState *gzipcs = (GzipCompressorState *) cs->private_data;
                                105                 :            221 :     z_streamp   zp = gzipcs->zp;
                                106                 :            221 :     void       *out = gzipcs->outbuf;
                                107                 :            221 :     int         res = Z_OK;
                                108                 :                : 
                                109   [ +  +  +  + ]:            365 :     while (gzipcs->zp->avail_in != 0 || flush)
                                110                 :                :     {
                                111         [ +  + ]:            223 :         res = deflate(zp, flush ? Z_FINISH : Z_NO_FLUSH);
                                112         [ -  + ]:            223 :         if (res == Z_STREAM_ERROR)
  416 tomas.vondra@postgre      113                 :UBC           0 :             pg_fatal("could not compress data: %s", zp->msg);
  416 tomas.vondra@postgre      114   [ +  +  -  + ]:CBC         223 :         if ((flush && (zp->avail_out < gzipcs->outsize))
                                115         [ +  - ]:            142 :             || (zp->avail_out == 0)
                                116         [ -  + ]:            142 :             || (zp->avail_in != 0)
                                117                 :                :             )
                                118                 :                :         {
                                119                 :                :             /*
                                120                 :                :              * Extra paranoia: avoid zero-length chunks, since a zero length
                                121                 :                :              * chunk is the EOF marker in the custom format. This should never
                                122                 :                :              * happen but ...
                                123                 :                :              */
                                124         [ +  - ]:             81 :             if (zp->avail_out < gzipcs->outsize)
                                125                 :                :             {
                                126                 :                :                 /*
                                127                 :                :                  * Any write function should do its own error checking but to
                                128                 :                :                  * make sure we do a check here as well ...
                                129                 :                :                  */
                                130                 :             81 :                 size_t      len = gzipcs->outsize - zp->avail_out;
                                131                 :                : 
                                132                 :             81 :                 cs->writeF(AH, (char *) out, len);
                                133                 :                :             }
                                134                 :             81 :             zp->next_out = out;
                                135                 :             81 :             zp->avail_out = gzipcs->outsize;
                                136                 :                :         }
                                137                 :                : 
                                138         [ +  + ]:            223 :         if (res == Z_STREAM_END)
                                139                 :             79 :             break;
                                140                 :                :     }
                                141                 :            221 : }
                                142                 :                : 
                                143                 :                : static void
                                144                 :            151 : EndCompressorGzip(ArchiveHandle *AH, CompressorState *cs)
                                145                 :                : {
                                146                 :                :     /* If deflation was initialized, finalize it */
  382                           147         [ +  + ]:            151 :     if (cs->private_data)
                                148                 :             79 :         DeflateCompressorEnd(AH, cs);
  416                           149                 :            151 : }
                                150                 :                : 
                                151                 :                : static void
                                152                 :            142 : WriteDataToArchiveGzip(ArchiveHandle *AH, CompressorState *cs,
                                153                 :                :                        const void *data, size_t dLen)
                                154                 :                : {
                                155                 :            142 :     GzipCompressorState *gzipcs = (GzipCompressorState *) cs->private_data;
                                156                 :                : 
                                157                 :            142 :     gzipcs->zp->next_in = (void *) unconstify(void *, data);
                                158                 :            142 :     gzipcs->zp->avail_in = dLen;
  382                           159                 :            142 :     DeflateCompressorCommon(AH, cs, false);
  416                           160                 :            142 : }
                                161                 :                : 
                                162                 :                : static void
                                163                 :             72 : ReadDataFromArchiveGzip(ArchiveHandle *AH, CompressorState *cs)
                                164                 :                : {
                                165                 :                :     z_streamp   zp;
                                166                 :                :     char       *out;
                                167                 :             72 :     int         res = Z_OK;
                                168                 :                :     size_t      cnt;
                                169                 :                :     char       *buf;
                                170                 :                :     size_t      buflen;
                                171                 :                : 
                                172                 :             72 :     zp = (z_streamp) pg_malloc(sizeof(z_stream));
                                173                 :             72 :     zp->zalloc = Z_NULL;
                                174                 :             72 :     zp->zfree = Z_NULL;
                                175                 :             72 :     zp->opaque = Z_NULL;
                                176                 :                : 
  388                           177                 :             72 :     buflen = DEFAULT_IO_BUFFER_SIZE;
                                178                 :             72 :     buf = pg_malloc(buflen);
                                179                 :                : 
                                180                 :             72 :     out = pg_malloc(DEFAULT_IO_BUFFER_SIZE + 1);
                                181                 :                : 
  416                           182         [ -  + ]:             72 :     if (inflateInit(zp) != Z_OK)
  416 tomas.vondra@postgre      183                 :UBC           0 :         pg_fatal("could not initialize compression library: %s",
                                184                 :                :                  zp->msg);
                                185                 :                : 
                                186                 :                :     /* no minimal chunk size for zlib */
  416 tomas.vondra@postgre      187         [ +  + ]:CBC         146 :     while ((cnt = cs->readF(AH, &buf, &buflen)))
                                188                 :                :     {
                                189                 :             74 :         zp->next_in = (void *) buf;
                                190                 :             74 :         zp->avail_in = cnt;
                                191                 :                : 
                                192         [ +  + ]:            154 :         while (zp->avail_in > 0)
                                193                 :                :         {
                                194                 :             80 :             zp->next_out = (void *) out;
  388                           195                 :             80 :             zp->avail_out = DEFAULT_IO_BUFFER_SIZE;
                                196                 :                : 
  416                           197                 :             80 :             res = inflate(zp, 0);
                                198   [ +  +  -  + ]:             80 :             if (res != Z_OK && res != Z_STREAM_END)
  416 tomas.vondra@postgre      199                 :UBC           0 :                 pg_fatal("could not uncompress data: %s", zp->msg);
                                200                 :                : 
  388 tomas.vondra@postgre      201                 :CBC          80 :             out[DEFAULT_IO_BUFFER_SIZE - zp->avail_out] = '\0';
                                202                 :             80 :             ahwrite(out, 1, DEFAULT_IO_BUFFER_SIZE - zp->avail_out, AH);
                                203                 :                :         }
                                204                 :                :     }
                                205                 :                : 
  416                           206                 :             72 :     zp->next_in = NULL;
                                207                 :             72 :     zp->avail_in = 0;
                                208         [ -  + ]:             72 :     while (res != Z_STREAM_END)
                                209                 :                :     {
  416 tomas.vondra@postgre      210                 :UBC           0 :         zp->next_out = (void *) out;
  388                           211                 :              0 :         zp->avail_out = DEFAULT_IO_BUFFER_SIZE;
  416                           212                 :              0 :         res = inflate(zp, 0);
                                213   [ #  #  #  # ]:              0 :         if (res != Z_OK && res != Z_STREAM_END)
                                214                 :              0 :             pg_fatal("could not uncompress data: %s", zp->msg);
                                215                 :                : 
  388                           216                 :              0 :         out[DEFAULT_IO_BUFFER_SIZE - zp->avail_out] = '\0';
                                217                 :              0 :         ahwrite(out, 1, DEFAULT_IO_BUFFER_SIZE - zp->avail_out, AH);
                                218                 :                :     }
                                219                 :                : 
  416 tomas.vondra@postgre      220         [ -  + ]:CBC          72 :     if (inflateEnd(zp) != Z_OK)
  416 tomas.vondra@postgre      221                 :UBC           0 :         pg_fatal("could not close compression library: %s", zp->msg);
                                222                 :                : 
  416 tomas.vondra@postgre      223                 :CBC          72 :     free(buf);
                                224                 :             72 :     free(out);
                                225                 :             72 :     free(zp);
                                226                 :             72 : }
                                227                 :                : 
                                228                 :                : /* Public routines that support gzip compressed data I/O */
                                229                 :                : void
                                230                 :            151 : InitCompressorGzip(CompressorState *cs,
                                231                 :                :                    const pg_compress_specification compression_spec)
                                232                 :                : {
                                233                 :            151 :     cs->readData = ReadDataFromArchiveGzip;
                                234                 :            151 :     cs->writeData = WriteDataToArchiveGzip;
                                235                 :            151 :     cs->end = EndCompressorGzip;
                                236                 :                : 
                                237                 :            151 :     cs->compression_spec = compression_spec;
                                238                 :                : 
                                239                 :                :     /*
                                240                 :                :      * If the caller has defined a write function, prepare the necessary
                                241                 :                :      * state.  Note that if the data is empty, End may be called immediately
                                242                 :                :      * after Init, without ever calling Write.
                                243                 :                :      */
  382                           244         [ +  + ]:            151 :     if (cs->writeF)
                                245                 :             79 :         DeflateCompressorInit(cs);
  416                           246                 :            151 : }
                                247                 :                : 
                                248                 :                : 
                                249                 :                : /*----------------------
                                250                 :                :  * Compress File API
                                251                 :                :  *----------------------
                                252                 :                :  */
                                253                 :                : 
                                254                 :                : static bool
  388                           255                 :            281 : Gzip_read(void *ptr, size_t size, size_t *rsize, CompressFileHandle *CFH)
                                256                 :                : {
  416                           257                 :            281 :     gzFile      gzfp = (gzFile) CFH->private_data;
                                258                 :                :     int         gzret;
                                259                 :                : 
  388                           260                 :            281 :     gzret = gzread(gzfp, ptr, size);
                                261   [ +  +  -  + ]:            281 :     if (gzret <= 0 && !gzeof(gzfp))
                                262                 :                :     {
                                263                 :                :         int         errnum;
  416 tomas.vondra@postgre      264                 :UBC           0 :         const char *errmsg = gzerror(gzfp, &errnum);
                                265                 :                : 
                                266         [ #  # ]:              0 :         pg_fatal("could not read from input file: %s",
                                267                 :                :                  errnum == Z_ERRNO ? strerror(errno) : errmsg);
                                268                 :                :     }
                                269                 :                : 
  388 tomas.vondra@postgre      270         [ +  - ]:CBC         281 :     if (rsize)
                                271                 :            281 :         *rsize = (size_t) gzret;
                                272                 :                : 
                                273                 :            281 :     return true;
                                274                 :                : }
                                275                 :                : 
                                276                 :                : static bool
  416                           277                 :          25794 : Gzip_write(const void *ptr, size_t size, CompressFileHandle *CFH)
                                278                 :                : {
                                279                 :          25794 :     gzFile      gzfp = (gzFile) CFH->private_data;
                                280                 :                : 
  388                           281                 :          25794 :     return gzwrite(gzfp, ptr, size) > 0;
                                282                 :                : }
                                283                 :                : 
                                284                 :                : static int
  416 tomas.vondra@postgre      285                 :UBC           0 : Gzip_getc(CompressFileHandle *CFH)
                                286                 :                : {
                                287                 :              0 :     gzFile      gzfp = (gzFile) CFH->private_data;
                                288                 :                :     int         ret;
                                289                 :                : 
                                290                 :              0 :     errno = 0;
                                291         [ #  # ]:              0 :     ret = gzgetc(gzfp);
                                292         [ #  # ]:              0 :     if (ret == EOF)
                                293                 :                :     {
                                294         [ #  # ]:              0 :         if (!gzeof(gzfp))
   33 michael@paquier.xyz       295                 :UNC           0 :             pg_fatal("could not read from input file: %m");
                                296                 :                :         else
  416 tomas.vondra@postgre      297                 :UBC           0 :             pg_fatal("could not read from input file: end of file");
                                298                 :                :     }
                                299                 :                : 
                                300                 :              0 :     return ret;
                                301                 :                : }
                                302                 :                : 
                                303                 :                : static char *
  416 tomas.vondra@postgre      304                 :CBC           4 : Gzip_gets(char *ptr, int size, CompressFileHandle *CFH)
                                305                 :                : {
                                306                 :              4 :     gzFile      gzfp = (gzFile) CFH->private_data;
                                307                 :                : 
                                308                 :              4 :     return gzgets(gzfp, ptr, size);
                                309                 :                : }
                                310                 :                : 
                                311                 :                : static bool
                                312                 :            246 : Gzip_close(CompressFileHandle *CFH)
                                313                 :                : {
                                314                 :            246 :     gzFile      gzfp = (gzFile) CFH->private_data;
                                315                 :                : 
                                316                 :            246 :     CFH->private_data = NULL;
                                317                 :                : 
  388                           318                 :            246 :     return gzclose(gzfp) == Z_OK;
                                319                 :                : }
                                320                 :                : 
                                321                 :                : static bool
  416                           322                 :              2 : Gzip_eof(CompressFileHandle *CFH)
                                323                 :                : {
                                324                 :              2 :     gzFile      gzfp = (gzFile) CFH->private_data;
                                325                 :                : 
  388                           326                 :              2 :     return gzeof(gzfp) == 1;
                                327                 :                : }
                                328                 :                : 
                                329                 :                : static const char *
  416 tomas.vondra@postgre      330                 :UBC           0 : Gzip_get_error(CompressFileHandle *CFH)
                                331                 :                : {
                                332                 :              0 :     gzFile      gzfp = (gzFile) CFH->private_data;
                                333                 :                :     const char *errmsg;
                                334                 :                :     int         errnum;
                                335                 :                : 
                                336                 :              0 :     errmsg = gzerror(gzfp, &errnum);
                                337         [ #  # ]:              0 :     if (errnum == Z_ERRNO)
                                338                 :              0 :         errmsg = strerror(errno);
                                339                 :                : 
                                340                 :              0 :     return errmsg;
                                341                 :                : }
                                342                 :                : 
                                343                 :                : static bool
  416 tomas.vondra@postgre      344                 :CBC         246 : Gzip_open(const char *path, int fd, const char *mode, CompressFileHandle *CFH)
                                345                 :                : {
                                346                 :                :     gzFile      gzfp;
                                347                 :                :     char        mode_compression[32];
                                348                 :                : 
                                349         [ +  + ]:            246 :     if (CFH->compression_spec.level != Z_DEFAULT_COMPRESSION)
                                350                 :                :     {
                                351                 :                :         /*
                                352                 :                :          * user has specified a compression level, so tell zlib to use it
                                353                 :                :          */
                                354                 :            157 :         snprintf(mode_compression, sizeof(mode_compression), "%s%d",
                                355                 :                :                  mode, CFH->compression_spec.level);
                                356                 :                :     }
                                357                 :                :     else
                                358                 :             89 :         strcpy(mode_compression, mode);
                                359                 :                : 
                                360         [ -  + ]:            246 :     if (fd >= 0)
  416 tomas.vondra@postgre      361                 :UBC           0 :         gzfp = gzdopen(dup(fd), mode_compression);
                                362                 :                :     else
  416 tomas.vondra@postgre      363                 :CBC         246 :         gzfp = gzopen(path, mode_compression);
                                364                 :                : 
                                365         [ -  + ]:            246 :     if (gzfp == NULL)
  388 tomas.vondra@postgre      366                 :UBC           0 :         return false;
                                367                 :                : 
  416 tomas.vondra@postgre      368                 :CBC         246 :     CFH->private_data = gzfp;
                                369                 :                : 
  388                           370                 :            246 :     return true;
                                371                 :                : }
                                372                 :                : 
                                373                 :                : static bool
  416                           374                 :            121 : Gzip_open_write(const char *path, const char *mode, CompressFileHandle *CFH)
                                375                 :                : {
                                376                 :                :     char       *fname;
                                377                 :                :     bool        ret;
                                378                 :                :     int         save_errno;
                                379                 :                : 
                                380                 :            121 :     fname = psprintf("%s.gz", path);
                                381                 :            121 :     ret = CFH->open_func(fname, -1, mode, CFH);
                                382                 :                : 
                                383                 :            121 :     save_errno = errno;
                                384                 :            121 :     pg_free(fname);
                                385                 :            121 :     errno = save_errno;
                                386                 :                : 
                                387                 :            121 :     return ret;
                                388                 :                : }
                                389                 :                : 
                                390                 :                : void
                                391                 :            246 : InitCompressFileHandleGzip(CompressFileHandle *CFH,
                                392                 :                :                            const pg_compress_specification compression_spec)
                                393                 :                : {
                                394                 :            246 :     CFH->open_func = Gzip_open;
                                395                 :            246 :     CFH->open_write_func = Gzip_open_write;
                                396                 :            246 :     CFH->read_func = Gzip_read;
                                397                 :            246 :     CFH->write_func = Gzip_write;
                                398                 :            246 :     CFH->gets_func = Gzip_gets;
                                399                 :            246 :     CFH->getc_func = Gzip_getc;
                                400                 :            246 :     CFH->close_func = Gzip_close;
                                401                 :            246 :     CFH->eof_func = Gzip_eof;
                                402                 :            246 :     CFH->get_error_func = Gzip_get_error;
                                403                 :                : 
                                404                 :            246 :     CFH->compression_spec = compression_spec;
                                405                 :                : 
                                406                 :            246 :     CFH->private_data = NULL;
                                407                 :            246 : }
                                408                 :                : #else                           /* HAVE_LIBZ */
                                409                 :                : void
                                410                 :                : InitCompressorGzip(CompressorState *cs,
                                411                 :                :                    const pg_compress_specification compression_spec)
                                412                 :                : {
                                413                 :                :     pg_fatal("this build does not support compression with %s", "gzip");
                                414                 :                : }
                                415                 :                : 
                                416                 :                : void
                                417                 :                : InitCompressFileHandleGzip(CompressFileHandle *CFH,
                                418                 :                :                            const pg_compress_specification compression_spec)
                                419                 :                : {
                                420                 :                :     pg_fatal("this build does not support compression with %s", "gzip");
                                421                 :                : }
                                422                 :                : #endif                          /* HAVE_LIBZ */
        

Generated by: LCOV version 2.1-beta2-3-g6141622