/*------------------------------------------------------------------------- * * pg_backup_custom.c * * Implements the custom output format. * * See the headers to pg_restore for more details. * * Copyright (c) 2000, Philip Warner * Rights are granted to use this software in any way so long * as this notice is not removed. * * The author is not responsible for loss or damages that may * result from it's use. * * * IDENTIFICATION * * Modifications - 28-Jun-2000 - pjw@rhyme.com.au * * Initial version. * *------------------------------------------------------------------------- */ #include #include "pg_backup.h" #include "pg_backup_archiver.h" extern int errno; static void _ArchiveEntry(ArchiveHandle* AH, TocEntry* te); static void _StartData(ArchiveHandle* AH, TocEntry* te); static int _WriteData(ArchiveHandle* AH, const void* data, int dLen); static void _EndData(ArchiveHandle* AH, TocEntry* te); static int _WriteByte(ArchiveHandle* AH, const int i); static int _ReadByte(ArchiveHandle* ); static int _WriteBuf(ArchiveHandle* AH, const void* buf, int len); static int _ReadBuf(ArchiveHandle* AH, void* buf, int len); static void _CloseArchive(ArchiveHandle* AH); static void _PrintTocData(ArchiveHandle* AH, TocEntry* te, RestoreOptions *ropt); static void _WriteExtraToc(ArchiveHandle* AH, TocEntry* te); static void _ReadExtraToc(ArchiveHandle* AH, TocEntry* te); static void _PrintExtraToc(ArchiveHandle* AH, TocEntry* te); static void _PrintData(ArchiveHandle* AH); static void _skipData(ArchiveHandle* AH); #define zlibOutSize 4096 #define zlibInSize 4096 typedef struct { z_streamp zp; char* zlibOut; char* zlibIn; int inSize; int hasSeek; int filePos; int dataStart; } lclContext; typedef struct { int dataPos; int dataLen; } lclTocEntry; static int _getFilePos(ArchiveHandle* AH, lclContext* ctx); static char* progname = "Archiver(custom)"; /* * Handler functions. */ void InitArchiveFmt_Custom(ArchiveHandle* AH) { lclContext* ctx; /* Assuming static functions, this can be copied for each format. */ AH->ArchiveEntryPtr = _ArchiveEntry; AH->StartDataPtr = _StartData; AH->WriteDataPtr = _WriteData; AH->EndDataPtr = _EndData; AH->WriteBytePtr = _WriteByte; AH->ReadBytePtr = _ReadByte; AH->WriteBufPtr = _WriteBuf; AH->ReadBufPtr = _ReadBuf; AH->ClosePtr = _CloseArchive; AH->PrintTocDataPtr = _PrintTocData; AH->ReadExtraTocPtr = _ReadExtraToc; AH->WriteExtraTocPtr = _WriteExtraToc; AH->PrintExtraTocPtr = _PrintExtraToc; /* * Set up some special context used in compressing data. */ ctx = (lclContext*)malloc(sizeof(lclContext)); if (ctx == NULL) die_horribly("%s: Unable to allocate archive context",progname); AH->formatData = (void*)ctx; ctx->zp = (z_streamp)malloc(sizeof(z_stream)); if (ctx->zp == NULL) die_horribly("%s: unable to allocate zlib stream archive context",progname); ctx->zlibOut = (char*)malloc(zlibOutSize); ctx->zlibIn = (char*)malloc(zlibInSize); ctx->inSize = zlibInSize; ctx->filePos = 0; if (ctx->zlibOut == NULL || ctx->zlibIn == NULL) die_horribly("%s: unable to allocate buffers in archive context",progname); /* * Now open the file */ if (AH->mode == archModeWrite) { if (AH->fSpec && strcmp(AH->fSpec,"") != 0) { AH->FH = fopen(AH->fSpec, PG_BINARY_W); } else { AH->FH = stdout; } if (!AH) die_horribly("%s: unable to open archive file %s",progname, AH->fSpec); ctx->hasSeek = (fseek(AH->FH, 0, SEEK_CUR) == 0); } else { if (AH->fSpec && strcmp(AH->fSpec,"") != 0) { AH->FH = fopen(AH->fSpec, PG_BINARY_R); } else { AH->FH = stdin; } if (!AH) die_horribly("%s: unable to open archive file %s",progname, AH->fSpec); ctx->hasSeek = (fseek(AH->FH, 0, SEEK_CUR) == 0); ReadHead(AH); ReadToc(AH); ctx->dataStart = _getFilePos(AH, ctx); } } /* * - Start a new TOC entry */ static void _ArchiveEntry(ArchiveHandle* AH, TocEntry* te) { lclTocEntry* ctx; ctx = (lclTocEntry*)malloc(sizeof(lclTocEntry)); if (te->dataDumper) { ctx->dataPos = -1; } else { ctx->dataPos = 0; } ctx->dataLen = 0; te->formatData = (void*)ctx; } static void _WriteExtraToc(ArchiveHandle* AH, TocEntry* te) { lclTocEntry* ctx = (lclTocEntry*)te->formatData; WriteInt(AH, ctx->dataPos); WriteInt(AH, ctx->dataLen); } static void _ReadExtraToc(ArchiveHandle* AH, TocEntry* te) { lclTocEntry* ctx = (lclTocEntry*)te->formatData; if (ctx == NULL) { ctx = (lclTocEntry*)malloc(sizeof(lclTocEntry)); te->formatData = (void*)ctx; } ctx->dataPos = ReadInt( AH ); ctx->dataLen = ReadInt( AH ); } static void _PrintExtraToc(ArchiveHandle* AH, TocEntry* te) { lclTocEntry* ctx = (lclTocEntry*)te->formatData; ahprintf(AH, "-- Data Pos: %d (Length %d)\n", ctx->dataPos, ctx->dataLen); } static void _StartData(ArchiveHandle* AH, TocEntry* te) { lclContext* ctx = (lclContext*)AH->formatData; z_streamp zp = ctx->zp; lclTocEntry* tctx = (lclTocEntry*)te->formatData; tctx->dataPos = _getFilePos(AH, ctx); WriteInt(AH, te->id); /* For sanity check */ #ifdef HAVE_LIBZ if (AH->compression < 0 || AH->compression > 9) { AH->compression = Z_DEFAULT_COMPRESSION; } if (AH->compression != 0) { zp->zalloc = Z_NULL; zp->zfree = Z_NULL; zp->opaque = Z_NULL; if (deflateInit(zp, AH->compression) != Z_OK) die_horribly("%s: could not initialize compression library - %s\n",progname, zp->msg); } #else AH->compression = 0; #endif /* Just be paranoid - maye End is called after Start, with no Write */ zp->next_out = ctx->zlibOut; zp->avail_out = zlibOutSize; } static int _DoDeflate(ArchiveHandle* AH, lclContext* ctx, int flush) { z_streamp zp = ctx->zp; #ifdef HAVE_LIBZ char* out = ctx->zlibOut; int res = Z_OK; if (AH->compression != 0) { res = deflate(zp, flush); if (res == Z_STREAM_ERROR) die_horribly("%s: could not compress data - %s\n",progname, zp->msg); if ( ( (flush == Z_FINISH) && (zp->avail_out < zlibOutSize) ) || (zp->avail_out == 0) || (zp->avail_in != 0) ) { /* * Extra paranoia: avoid zero-length chunks since a zero * length chunk is the EOF marker. This should never happen * but... */ if (zp->avail_out < zlibOutSize) { /* printf("Wrote %d byte deflated chunk\n", zlibOutSize - zp->avail_out); */ WriteInt(AH, zlibOutSize - zp->avail_out); fwrite(out, 1, zlibOutSize - zp->avail_out, AH->FH); ctx->filePos += zlibOutSize - zp->avail_out; } zp->next_out = out; zp->avail_out = zlibOutSize; } } else { #endif if (zp->avail_in > 0) { WriteInt(AH, zp->avail_in); fwrite(zp->next_in, 1, zp->avail_in, AH->FH); ctx->filePos += zp->avail_in; zp->avail_in = 0; } else { #ifdef HAVE_LIBZ if (flush == Z_FINISH) res = Z_STREAM_END; #endif } #ifdef HAVE_LIBZ } return res; #else return 1; #endif } static int _WriteData(ArchiveHandle* AH, const void* data, int dLen) { lclContext* ctx = (lclContext*)AH->formatData; z_streamp zp = ctx->zp; zp->next_in = (void*)data; zp->avail_in = dLen; while (zp->avail_in != 0) { /* printf("Deflating %d bytes\n", dLen); */ _DoDeflate(AH, ctx, 0); } return dLen; } static void _EndData(ArchiveHandle* AH, TocEntry* te) { lclContext* ctx = (lclContext*)AH->formatData; lclTocEntry* tctx = (lclTocEntry*) te->formatData; #ifdef HAVE_LIBZ z_streamp zp = ctx->zp; int res; if (AH->compression != 0) { zp->next_in = NULL; zp->avail_in = 0; do { /* printf("Ending data output\n"); */ res = _DoDeflate(AH, ctx, Z_FINISH); } while (res != Z_STREAM_END); if (deflateEnd(zp) != Z_OK) die_horribly("%s: error closing compression stream - %s\n", progname, zp->msg); } #endif /* Send the end marker */ WriteInt(AH, 0); tctx->dataLen = _getFilePos(AH, ctx) - tctx->dataPos; } /* * Print data for a gievn TOC entry */ static void _PrintTocData(ArchiveHandle* AH, TocEntry* te, RestoreOptions *ropt) { lclContext* ctx = (lclContext*)AH->formatData; int id; lclTocEntry* tctx = (lclTocEntry*) te->formatData; if (tctx->dataPos == 0) return; if (!ctx->hasSeek || tctx->dataPos < 0) { id = ReadInt(AH); while (id != te->id) { if (TocIDRequired(AH, id, ropt) & 2) die_horribly("%s: Dumping a specific TOC data block out of order is not supported" " without on this input stream (fseek required)\n", progname); _skipData(AH); id = ReadInt(AH); } } else { if (fseek(AH->FH, tctx->dataPos, SEEK_SET) != 0) die_horribly("%s: error %d in file seek\n",progname, errno); id = ReadInt(AH); } if (id != te->id) die_horribly("%s: Found unexpected block ID (%d) when reading data - expected %d\n", progname, id, te->id); ahprintf(AH, "--\n-- Data for TOC Entry ID %d (OID %s) %s %s\n--\n\n", te->id, te->oid, te->desc, te->name); _PrintData(AH); ahprintf(AH, "\n\n"); } /* * Print data from current file position. */ static void _PrintData(ArchiveHandle* AH) { lclContext* ctx = (lclContext*)AH->formatData; z_streamp zp = ctx->zp; int blkLen; char* in = ctx->zlibIn; int cnt; #ifdef HAVE_LIBZ int res; char* out = ctx->zlibOut; res = Z_OK; if (AH->compression != 0) { zp->zalloc = Z_NULL; zp->zfree = Z_NULL; zp->opaque = Z_NULL; if (inflateInit(zp) != Z_OK) die_horribly("%s: could not initialize compression library - %s\n", progname, zp->msg); } #endif blkLen = ReadInt(AH); while (blkLen != 0) { if (blkLen > ctx->inSize) { free(ctx->zlibIn); ctx->zlibIn = NULL; ctx->zlibIn = (char*)malloc(blkLen); if (!ctx->zlibIn) die_horribly("%s: failed to allocate decompression buffer\n", progname); ctx->inSize = blkLen; in = ctx->zlibIn; } cnt = fread(in, 1, blkLen, AH->FH); if (cnt != blkLen) die_horribly("%s: could not read data block - expected %d, got %d\n", progname, blkLen, cnt); ctx->filePos += blkLen; zp->next_in = in; zp->avail_in = blkLen; #ifdef HAVE_LIBZ if (AH->compression != 0) { while (zp->avail_in != 0) { zp->next_out = out; zp->avail_out = zlibOutSize; res = inflate(zp, 0); if (res != Z_OK && res != Z_STREAM_END) die_horribly("%s: unable to uncompress data - %s\n", progname, zp->msg); out[zlibOutSize - zp->avail_out] = '\0'; ahwrite(out, 1, zlibOutSize - zp->avail_out, AH); } } else { #endif ahwrite(in, 1, zp->avail_in, AH); zp->avail_in = 0; #ifdef HAVE_LIBZ } #endif blkLen = ReadInt(AH); } #ifdef HAVE_LIBZ if (AH->compression != 0) { zp->next_in = NULL; zp->avail_in = 0; while (res != Z_STREAM_END) { zp->next_out = out; zp->avail_out = zlibOutSize; res = inflate(zp, 0); if (res != Z_OK && res != Z_STREAM_END) die_horribly("%s: unable to uncompress data - %s\n", progname, zp->msg); out[zlibOutSize - zp->avail_out] = '\0'; ahwrite(out, 1, zlibOutSize - zp->avail_out, AH); } } #endif } /* * Skip data from current file position. */ static void _skipData(ArchiveHandle* AH) { lclContext* ctx = (lclContext*)AH->formatData; int blkLen; char* in = ctx->zlibIn; int cnt; blkLen = ReadInt(AH); while (blkLen != 0) { if (blkLen > ctx->inSize) { free(ctx->zlibIn); ctx->zlibIn = (char*)malloc(blkLen); ctx->inSize = blkLen; in = ctx->zlibIn; } cnt = fread(in, 1, blkLen, AH->FH); if (cnt != blkLen) die_horribly("%s: could not read data block - expected %d, got %d\n", progname, blkLen, cnt); ctx->filePos += blkLen; blkLen = ReadInt(AH); } } static int _WriteByte(ArchiveHandle* AH, const int i) { lclContext* ctx = (lclContext*)AH->formatData; int res; res = fputc(i, AH->FH); if (res != EOF) { ctx->filePos += 1; } return res; } static int _ReadByte(ArchiveHandle* AH) { lclContext* ctx = (lclContext*)AH->formatData; int res; res = fgetc(AH->FH); if (res != EOF) { ctx->filePos += 1; } return res; } static int _WriteBuf(ArchiveHandle* AH, const void* buf, int len) { lclContext* ctx = (lclContext*)AH->formatData; int res; res = fwrite(buf, 1, len, AH->FH); ctx->filePos += res; return res; } static int _ReadBuf(ArchiveHandle* AH, void* buf, int len) { lclContext* ctx = (lclContext*)AH->formatData; int res; res = fread(buf, 1, len, AH->FH); ctx->filePos += res; return res; } static void _CloseArchive(ArchiveHandle* AH) { lclContext* ctx = (lclContext*)AH->formatData; int tpos; if (AH->mode == archModeWrite) { WriteHead(AH); tpos = ftell(AH->FH); WriteToc(AH); ctx->dataStart = _getFilePos(AH, ctx); WriteDataChunks(AH); /* This is not an essential operation - it is really only * needed if we expect to be doing seeks to read the data back * - it may be ok to just use the existing self-consistent block * formatting. */ if (ctx->hasSeek) { fseek(AH->FH, tpos, SEEK_SET); WriteToc(AH); } } fclose(AH->FH); AH->FH = NULL; } static int _getFilePos(ArchiveHandle* AH, lclContext* ctx) { int pos; if (ctx->hasSeek) { pos = ftell(AH->FH); if (pos != ctx->filePos) { fprintf(stderr, "Warning: ftell mismatch with filePos\n"); } } else { pos = ctx->filePos; } return pos; }