#include "pg_backup.h"
#include "pg_backup_archiver.h"
#include "pg_backup_tar.h"
#include "pg_backup_utils.h"
#include "parallel.h"
#include "pgtar.h"
#include <sys/stat.h>
#include <ctype.h>
#include <limits.h>
#include <unistd.h>
Go to the source code of this file.
Data Structures | |
struct | TAR_MEMBER |
struct | lclContext |
struct | lclTocEntry |
Defines | |
#define | K_STD_BUF_SIZE 1024 |
#define | MAX_TAR_MEMBER_FILELEN (((int64) 1 << Min(33, sizeof(pgoff_t)*8 - 1)) - 1) |
Functions | |
static void | _ArchiveEntry (ArchiveHandle *AH, TocEntry *te) |
static void | _StartData (ArchiveHandle *AH, TocEntry *te) |
static size_t | _WriteData (ArchiveHandle *AH, const void *data, size_t dLen) |
static void | _EndData (ArchiveHandle *AH, TocEntry *te) |
static int | _WriteByte (ArchiveHandle *AH, const int i) |
static int | _ReadByte (ArchiveHandle *) |
static size_t | _WriteBuf (ArchiveHandle *AH, const void *buf, size_t len) |
static size_t | _ReadBuf (ArchiveHandle *AH, void *buf, size_t 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 | _StartBlobs (ArchiveHandle *AH, TocEntry *te) |
static void | _StartBlob (ArchiveHandle *AH, TocEntry *te, Oid oid) |
static void | _EndBlob (ArchiveHandle *AH, TocEntry *te, Oid oid) |
static void | _EndBlobs (ArchiveHandle *AH, TocEntry *te) |
static void | _LoadBlobs (ArchiveHandle *AH, RestoreOptions *ropt) |
static TAR_MEMBER * | tarOpen (ArchiveHandle *AH, const char *filename, char mode) |
static void | tarClose (ArchiveHandle *AH, TAR_MEMBER *TH) |
static int | tarPrintf (ArchiveHandle *AH, TAR_MEMBER *th, const char *fmt,...) __attribute__((format(PG_PRINTF_ATTRIBUTE |
static int static void | _tarAddFile (ArchiveHandle *AH, TAR_MEMBER *th) |
static TAR_MEMBER * | _tarPositionTo (ArchiveHandle *AH, const char *filename) |
static size_t | tarRead (void *buf, size_t len, TAR_MEMBER *th) |
static size_t | tarWrite (const void *buf, size_t len, TAR_MEMBER *th) |
static void | _tarWriteHeader (TAR_MEMBER *th) |
static int | _tarGetHeader (ArchiveHandle *AH, TAR_MEMBER *th) |
static size_t | _tarReadRaw (ArchiveHandle *AH, void *buf, size_t len, TAR_MEMBER *th, FILE *fh) |
static size_t | _scriptOut (ArchiveHandle *AH, const void *buf, size_t len) |
void | InitArchiveFmt_Tar (ArchiveHandle *AH) |
static void | _PrintFileData (ArchiveHandle *AH, char *filename, RestoreOptions *ropt) |
bool | isValidTarHeader (char *header) |
Variables | |
static const char * | modulename = gettext_noop("tar archiver") |
#define K_STD_BUF_SIZE 1024 |
Definition at line 62 of file pg_backup_tar.c.
Referenced by _StartBlobs().
#define MAX_TAR_MEMBER_FILELEN (((int64) 1 << Min(33, sizeof(pgoff_t)*8 - 1)) - 1) |
Definition at line 87 of file pg_backup_tar.c.
Referenced by _tarAddFile().
static void _ArchiveEntry | ( | ArchiveHandle * | AH, | |
TocEntry * | te | |||
) | [static] |
Definition at line 273 of file pg_backup_tar.c.
References _archiveHandle::compression, _tocEntry::dataDumper, _tocEntry::dumpId, lclTocEntry::filename, fn(), _tocEntry::formatData, NULL, pg_malloc0(), pg_strdup(), and lclTocEntry::TH.
{ lclTocEntry *ctx; char fn[K_STD_BUF_SIZE]; ctx = (lclTocEntry *) pg_malloc0(sizeof(lclTocEntry)); if (te->dataDumper != NULL) { #ifdef HAVE_LIBZ if (AH->compression == 0) sprintf(fn, "%d.dat", te->dumpId); else sprintf(fn, "%d.dat.gz", te->dumpId); #else sprintf(fn, "%d.dat", te->dumpId); #endif ctx->filename = pg_strdup(fn); } else { ctx->filename = NULL; ctx->TH = NULL; } te->formatData = (void *) ctx; }
static void _CloseArchive | ( | ArchiveHandle * | AH | ) | [static] |
Definition at line 815 of file pg_backup_tar.c.
References archModeWrite, _archiveHandle::CustomOutPtr, exit_horribly(), _archiveHandle::FH, lclContext::FH, _archiveHandle::formatData, i, lclContext::isSpecialScript, _archiveHandle::mode, modulename, NewRestoreOptions(), NULL, _archiveHandle::public, RestoreArchive(), _archiveHandle::ropt, lclContext::scriptTH, tarClose(), lclContext::tarFH, tarOpen(), tarPrintf(), Archive::verbose, WriteDataChunks(), WriteHead(), and WriteToc().
{ lclContext *ctx = (lclContext *) AH->formatData; TAR_MEMBER *th; RestoreOptions *ropt; RestoreOptions *savRopt; int savVerbose, i; if (AH->mode == archModeWrite) { /* * Write the Header & TOC to the archive FIRST */ th = tarOpen(AH, "toc.dat", 'w'); ctx->FH = th; WriteHead(AH); WriteToc(AH); tarClose(AH, th); /* Not needed any more */ /* * Now send the data (tables & blobs) */ WriteDataChunks(AH, NULL); /* * Now this format wants to append a script which does a full restore * if the files have been extracted. */ th = tarOpen(AH, "restore.sql", 'w'); tarPrintf(AH, th, "--\n" "-- NOTE:\n" "--\n" "-- File paths need to be edited. Search for $$PATH$$ and\n" "-- replace it with the path to the directory containing\n" "-- the extracted data files.\n" "--\n"); AH->CustomOutPtr = _scriptOut; ctx->isSpecialScript = 1; ctx->scriptTH = th; ropt = NewRestoreOptions(); memcpy(ropt, AH->ropt, sizeof(RestoreOptions)); ropt->filename = NULL; ropt->dropSchema = 1; ropt->compression = 0; ropt->superuser = NULL; ropt->suppressDumpWarnings = true; savRopt = AH->ropt; AH->ropt = ropt; savVerbose = AH->public.verbose; AH->public.verbose = 0; RestoreArchive((Archive *) AH); AH->ropt = savRopt; AH->public.verbose = savVerbose; tarClose(AH, th); ctx->isSpecialScript = 0; /* * EOF marker for tar files is two blocks of NULLs. */ for (i = 0; i < 512 * 2; i++) { if (fputc(0, ctx->tarFH) == EOF) exit_horribly(modulename, "could not write null block at end of tar archive\n"); } } AH->FH = NULL; }
static void _EndBlob | ( | ArchiveHandle * | AH, | |
TocEntry * | te, | |||
Oid | oid | |||
) | [static] |
Definition at line 965 of file pg_backup_tar.c.
References _tocEntry::formatData, tarClose(), and lclTocEntry::TH.
{ lclTocEntry *tctx = (lclTocEntry *) te->formatData; tarClose(AH, tctx->TH); }
static void _EndBlobs | ( | ArchiveHandle * | AH, | |
TocEntry * | te | |||
) | [static] |
Definition at line 979 of file pg_backup_tar.c.
References lclContext::blobToc, _archiveHandle::formatData, and tarClose().
{ lclContext *ctx = (lclContext *) AH->formatData; /* Write out a fake zero OID to mark end-of-blobs. */ /* WriteInt(AH, 0); */ tarClose(AH, ctx->blobToc); }
static void _EndData | ( | ArchiveHandle * | AH, | |
TocEntry * | te | |||
) | [static] |
Definition at line 615 of file pg_backup_tar.c.
References _tocEntry::formatData, tarClose(), and lclTocEntry::TH.
{ lclTocEntry *tctx = (lclTocEntry *) te->formatData; /* Close the file */ tarClose(AH, tctx->TH); tctx->TH = NULL; }
static void _LoadBlobs | ( | ArchiveHandle * | AH, | |
RestoreOptions * | ropt | |||
) | [static] |
Definition at line 710 of file pg_backup_tar.c.
References ahlog(), ahwrite(), atooid, _restoreOptions::dropSchema, EndRestoreBlob(), EndRestoreBlobs(), lclContext::FH, _archiveHandle::formatData, NULL, StartRestoreBlob(), StartRestoreBlobs(), tarClose(), TAR_MEMBER::targetFile, tarOpen(), and tarRead().
Referenced by _PrintTocData().
{ Oid oid; lclContext *ctx = (lclContext *) AH->formatData; TAR_MEMBER *th; size_t cnt; bool foundBlob = false; char buf[4096]; StartRestoreBlobs(AH); th = tarOpen(AH, NULL, 'r'); /* Open next file */ while (th != NULL) { ctx->FH = th; if (strncmp(th->targetFile, "blob_", 5) == 0) { oid = atooid(&th->targetFile[5]); if (oid != 0) { ahlog(AH, 1, "restoring large object with OID %u\n", oid); StartRestoreBlob(AH, oid, ropt->dropSchema); while ((cnt = tarRead(buf, 4095, th)) > 0) { buf[cnt] = '\0'; ahwrite(buf, 1, cnt, AH); } EndRestoreBlob(AH, oid); foundBlob = true; } tarClose(AH, th); } else { tarClose(AH, th); /* * Once we have found the first blob, stop at the first non-blob * entry (which will be 'blobs.toc'). This coding would eat all * the rest of the archive if there are no blobs ... but this * function shouldn't be called at all in that case. */ if (foundBlob) break; } th = tarOpen(AH, NULL, 'r'); } EndRestoreBlobs(AH); }
static void _PrintExtraToc | ( | ArchiveHandle * | AH, | |
TocEntry * | te | |||
) | [static] |
Definition at line 331 of file pg_backup_tar.c.
References ahprintf(), lclTocEntry::filename, _tocEntry::formatData, NULL, _archiveHandle::public, and Archive::verbose.
{ lclTocEntry *ctx = (lclTocEntry *) te->formatData; if (AH->public.verbose && ctx->filename != NULL) ahprintf(AH, "-- File: %s\n", ctx->filename); }
static void _PrintFileData | ( | ArchiveHandle * | AH, | |
char * | filename, | |||
RestoreOptions * | ropt | |||
) | [static] |
Definition at line 628 of file pg_backup_tar.c.
References ahwrite(), lclContext::FH, _archiveHandle::formatData, tarClose(), tarOpen(), and tarRead().
Referenced by _PrintTocData().
{ lclContext *ctx = (lclContext *) AH->formatData; char buf[4096]; size_t cnt; TAR_MEMBER *th; if (!filename) return; th = tarOpen(AH, filename, 'r'); ctx->FH = th; while ((cnt = tarRead(buf, 4095, th)) > 0) { buf[cnt] = '\0'; ahwrite(buf, 1, cnt, AH); } tarClose(AH, th); }
static void _PrintTocData | ( | ArchiveHandle * | AH, | |
TocEntry * | te, | |||
RestoreOptions * | ropt | |||
) | [static] |
Definition at line 655 of file pg_backup_tar.c.
References _LoadBlobs(), _PrintFileData(), ahprintf(), ahwrite(), _tocEntry::copyStmt, _tocEntry::desc, exit_horribly(), _tocEntry::formatData, _archiveHandle::formatData, lclContext::isSpecialScript, and modulename.
{ lclContext *ctx = (lclContext *) AH->formatData; lclTocEntry *tctx = (lclTocEntry *) te->formatData; int pos1; if (!tctx->filename) return; /* * If we're writing the special restore.sql script, emit a suitable * command to include each table's data from the corresponding file. * * In the COPY case this is a bit klugy because the regular COPY command * was already printed before we get control. */ if (ctx->isSpecialScript) { if (te->copyStmt) { /* Abort the COPY FROM stdin */ ahprintf(AH, "\\.\n"); /* * The COPY statement should look like "COPY ... FROM stdin;\n", * see dumpTableData(). */ pos1 = (int) strlen(te->copyStmt) - 13; if (pos1 < 6 || strncmp(te->copyStmt, "COPY ", 5) != 0 || strcmp(te->copyStmt + pos1, " FROM stdin;\n") != 0) exit_horribly(modulename, "unexpected COPY statement syntax: \"%s\"\n", te->copyStmt); /* Emit all but the FROM part ... */ ahwrite(te->copyStmt, 1, pos1, AH); /* ... and insert modified FROM */ ahprintf(AH, " FROM '$$PATH$$/%s';\n\n", tctx->filename); } else { /* --inserts mode, no worries, just include the data file */ ahprintf(AH, "\\i $$PATH$$/%s\n\n", tctx->filename); } return; } if (strcmp(te->desc, "BLOBS") == 0) _LoadBlobs(AH, ropt); else _PrintFileData(AH, tctx->filename, ropt); }
static size_t _ReadBuf | ( | ArchiveHandle * | AH, | |
void * | buf, | |||
size_t | len | |||
) | [static] |
Definition at line 804 of file pg_backup_tar.c.
References lclContext::FH, lclContext::filePos, _archiveHandle::formatData, and tarRead().
{ lclContext *ctx = (lclContext *) AH->formatData; size_t res; res = tarRead(buf, len, ctx->FH); ctx->filePos += res; return res; }
static int _ReadByte | ( | ArchiveHandle * | AH | ) | [static] |
Definition at line 779 of file pg_backup_tar.c.
References exit_horribly(), lclContext::FH, lclContext::filePos, _archiveHandle::formatData, modulename, and tarRead().
{ lclContext *ctx = (lclContext *) AH->formatData; size_t res; unsigned char c; res = tarRead(&c, 1, ctx->FH); if (res != 1) exit_horribly(modulename, "unexpected end of file\n"); ctx->filePos += 1; return c; }
static void _ReadExtraToc | ( | ArchiveHandle * | AH, | |
TocEntry * | te | |||
) | [static] |
Definition at line 311 of file pg_backup_tar.c.
References lclTocEntry::filename, _tocEntry::formatData, free, NULL, pg_malloc0(), ReadStr(), and lclTocEntry::TH.
{ lclTocEntry *ctx = (lclTocEntry *) te->formatData; if (ctx == NULL) { ctx = (lclTocEntry *) pg_malloc0(sizeof(lclTocEntry)); te->formatData = (void *) ctx; } ctx->filename = ReadStr(AH); if (strlen(ctx->filename) == 0) { free(ctx->filename); ctx->filename = NULL; } ctx->TH = NULL; }
static size_t _scriptOut | ( | ArchiveHandle * | AH, | |
const void * | buf, | |||
size_t | len | |||
) | [static] |
Definition at line 897 of file pg_backup_tar.c.
References _archiveHandle::formatData, lclContext::scriptTH, and tarWrite().
{ lclContext *ctx = (lclContext *) AH->formatData; return tarWrite(buf, len, ctx->scriptTH); }
static void _StartBlob | ( | ArchiveHandle * | AH, | |
TocEntry * | te, | |||
Oid | oid | |||
) | [static] |
Definition at line 936 of file pg_backup_tar.c.
References lclContext::blobToc, _archiveHandle::compression, exit_horribly(), _tocEntry::formatData, _archiveHandle::formatData, modulename, tarOpen(), and tarPrintf().
{ lclContext *ctx = (lclContext *) AH->formatData; lclTocEntry *tctx = (lclTocEntry *) te->formatData; char fname[255]; char *sfx; if (oid == 0) exit_horribly(modulename, "invalid OID for large object (%u)\n", oid); if (AH->compression != 0) sfx = ".gz"; else sfx = ""; sprintf(fname, "blob_%u.dat%s", oid, sfx); tarPrintf(AH, ctx->blobToc, "%u %s\n", oid, fname); tctx->TH = tarOpen(AH, fname, 'w'); }
static void _StartBlobs | ( | ArchiveHandle * | AH, | |
TocEntry * | te | |||
) | [static] |
Definition at line 919 of file pg_backup_tar.c.
References lclContext::blobToc, _archiveHandle::formatData, K_STD_BUF_SIZE, and tarOpen().
{ lclContext *ctx = (lclContext *) AH->formatData; char fname[K_STD_BUF_SIZE]; sprintf(fname, "blobs.toc"); ctx->blobToc = tarOpen(AH, fname, 'w'); }
static void _StartData | ( | ArchiveHandle * | AH, | |
TocEntry * | te | |||
) | [static] |
Definition at line 340 of file pg_backup_tar.c.
References lclTocEntry::filename, _tocEntry::formatData, tarOpen(), and lclTocEntry::TH.
{ lclTocEntry *tctx = (lclTocEntry *) te->formatData; tctx->TH = tarOpen(AH, tctx->filename, 'w'); }
static void _tarAddFile | ( | ArchiveHandle * | AH, | |
TAR_MEMBER * | th | |||
) | [static] |
Definition at line 1055 of file pg_backup_tar.c.
References _tarWriteHeader(), exit_horribly(), TAR_MEMBER::fileLen, _archiveHandle::formatData, fseeko, ftello, i, MAX_TAR_MEMBER_FILELEN, modulename, pgoff_t, snprintf(), strerror(), TAR_MEMBER::tarFH, lclContext::tarFHpos, and TAR_MEMBER::tmpFH.
Referenced by tarClose().
{ lclContext *ctx = (lclContext *) AH->formatData; FILE *tmp = th->tmpFH; /* Grab it for convenience */ char buf[32768]; size_t cnt; pgoff_t len = 0; size_t res; size_t i, pad; /* * Find file len & go back to start. */ fseeko(tmp, 0, SEEK_END); th->fileLen = ftello(tmp); fseeko(tmp, 0, SEEK_SET); /* * Some compilers will throw a warning knowing this test can never be true * because pgoff_t can't exceed the compared maximum on their platform. */ if (th->fileLen > MAX_TAR_MEMBER_FILELEN) exit_horribly(modulename, "archive member too large for tar format\n"); _tarWriteHeader(th); while ((cnt = fread(buf, 1, sizeof(buf), tmp)) > 0) { res = fwrite(buf, 1, cnt, th->tarFH); if (res != cnt) exit_horribly(modulename, "could not write to output file: %s\n", strerror(errno)); len += res; } if (fclose(tmp) != 0) /* This *should* delete it... */ exit_horribly(modulename, "could not close temporary file: %s\n", strerror(errno)); if (len != th->fileLen) { char buf1[32], buf2[32]; snprintf(buf1, sizeof(buf1), INT64_FORMAT, (int64) len); snprintf(buf2, sizeof(buf2), INT64_FORMAT, (int64) th->fileLen); exit_horribly(modulename, "actual file length (%s) does not match expected (%s)\n", buf1, buf2); } pad = ((len + 511) & ~511) - len; for (i = 0; i < pad; i++) { if (fputc('\0', th->tarFH) == EOF) exit_horribly(modulename, "could not output padding at end of tar member\n"); } ctx->tarFHpos += len + pad; }
static int _tarGetHeader | ( | ArchiveHandle * | AH, | |
TAR_MEMBER * | th | |||
) | [static] |
Definition at line 1201 of file pg_backup_tar.c.
References _tarReadRaw(), ahlog(), exit_horribly(), TAR_MEMBER::fileLen, _archiveHandle::formatData, ftello, i, modulename, ngettext, NULL, pg_strdup(), pgoff_t, snprintf(), tarChecksum(), lclContext::tarFH, lclContext::tarFHpos, and TAR_MEMBER::targetFile.
Referenced by _tarPositionTo().
{ lclContext *ctx = (lclContext *) AH->formatData; char h[512]; char tag[100]; int sum, chk; size_t len; unsigned long ullen; pgoff_t hPos; bool gotBlock = false; while (!gotBlock) { #if 0 if (ftello(ctx->tarFH) != ctx->tarFHpos) { char buf1[100], buf2[100]; snprintf(buf1, sizeof(buf1), INT64_FORMAT, (int64) ftello(ctx->tarFH)); snprintf(buf2, sizeof(buf2), INT64_FORMAT, (int64) ftello(ctx->tarFHpos)); exit_horribly(modulename, "mismatch in actual vs. predicted file position (%s vs. %s)\n", buf1, buf2); } #endif /* Save the pos for reporting purposes */ hPos = ctx->tarFHpos; /* Read a 512 byte block, return EOF, exit if short */ len = _tarReadRaw(AH, h, 512, NULL, ctx->tarFH); if (len == 0) /* EOF */ return 0; if (len != 512) exit_horribly(modulename, ngettext("incomplete tar header found (%lu byte)\n", "incomplete tar header found (%lu bytes)\n", len), (unsigned long) len); /* Calc checksum */ chk = tarChecksum(h); sscanf(&h[148], "%8o", &sum); /* * If the checksum failed, see if it is a null block. If so, silently * continue to the next block. */ if (chk == sum) gotBlock = true; else { int i; for (i = 0; i < 512; i++) { if (h[i] != 0) { gotBlock = true; break; } } } } sscanf(&h[0], "%99s", tag); sscanf(&h[124], "%12lo", &ullen); len = (size_t) ullen; { char buf[100]; snprintf(buf, sizeof(buf), INT64_FORMAT, (int64) hPos); ahlog(AH, 3, "TOC Entry %s at %s (length %lu, checksum %d)\n", tag, buf, (unsigned long) len, sum); } if (chk != sum) { char buf[100]; snprintf(buf, sizeof(buf), INT64_FORMAT, (int64) ftello(ctx->tarFH)); exit_horribly(modulename, "corrupt tar header found in %s " "(expected %d, computed %d) file position %s\n", tag, sum, chk, buf); } th->targetFile = pg_strdup(tag); th->fileLen = len; return 1; }
static TAR_MEMBER * _tarPositionTo | ( | ArchiveHandle * | AH, | |
const char * | filename | |||
) | [static] |
Definition at line 1119 of file pg_backup_tar.c.
References _tarGetHeader(), _tarReadRaw(), TAR_MEMBER::AH, ahlog(), exit_horribly(), TAR_MEMBER::fileLen, _archiveHandle::formatData, free, header(), i, modulename, NULL, pg_malloc0(), TAR_MEMBER::pos, REQ_DATA, snprintf(), lclContext::tarFH, lclContext::tarFHpos, TAR_MEMBER::targetFile, lclContext::tarNextMember, and TocIDRequired().
Referenced by tarOpen().
{ lclContext *ctx = (lclContext *) AH->formatData; TAR_MEMBER *th = pg_malloc0(sizeof(TAR_MEMBER)); char c; char header[512]; size_t i, len, blks; int id; th->AH = AH; /* Go to end of current file, if any */ if (ctx->tarFHpos != 0) { char buf1[100], buf2[100]; snprintf(buf1, sizeof(buf1), INT64_FORMAT, (int64) ctx->tarFHpos); snprintf(buf2, sizeof(buf2), INT64_FORMAT, (int64) ctx->tarNextMember); ahlog(AH, 4, "moving from position %s to next member at file position %s\n", buf1, buf2); while (ctx->tarFHpos < ctx->tarNextMember) _tarReadRaw(AH, &c, 1, NULL, ctx->tarFH); } { char buf[100]; snprintf(buf, sizeof(buf), INT64_FORMAT, (int64) ctx->tarFHpos); ahlog(AH, 4, "now at file position %s\n", buf); } /* We are at the start of the file, or at the next member */ /* Get the header */ if (!_tarGetHeader(AH, th)) { if (filename) exit_horribly(modulename, "could not find header for file \"%s\" in tar archive\n", filename); else { /* * We're just scanning the archive for the next file, so return * null */ free(th); return NULL; } } while (filename != NULL && strcmp(th->targetFile, filename) != 0) { ahlog(AH, 4, "skipping tar member %s\n", th->targetFile); id = atoi(th->targetFile); if ((TocIDRequired(AH, id) & REQ_DATA) != 0) exit_horribly(modulename, "restoring data out of order is not supported in this archive format: " "\"%s\" is required, but comes before \"%s\" in the archive file.\n", th->targetFile, filename); /* Header doesn't match, so read to next header */ len = ((th->fileLen + 511) & ~511); /* Padded length */ blks = len >> 9; /* # of 512 byte blocks */ for (i = 0; i < blks; i++) _tarReadRaw(AH, &header[0], 512, NULL, ctx->tarFH); if (!_tarGetHeader(AH, th)) exit_horribly(modulename, "could not find header for file \"%s\" in tar archive\n", filename); } ctx->tarNextMember = ctx->tarFHpos + ((th->fileLen + 511) & ~511); th->pos = 0; return th; }
static size_t _tarReadRaw | ( | ArchiveHandle * | AH, | |
void * | buf, | |||
size_t | len, | |||
TAR_MEMBER * | th, | |||
FILE * | fh | |||
) | [static] |
Definition at line 523 of file pg_backup_tar.c.
References exit_horribly(), _archiveHandle::formatData, GZREAD, _archiveHandle::lookahead, _archiveHandle::lookaheadLen, _archiveHandle::lookaheadPos, modulename, TAR_MEMBER::nFH, lclContext::tarFHpos, and TAR_MEMBER::zFH.
Referenced by _tarGetHeader(), _tarPositionTo(), and tarRead().
{ lclContext *ctx = (lclContext *) AH->formatData; size_t avail; size_t used = 0; size_t res = 0; avail = AH->lookaheadLen - AH->lookaheadPos; if (avail > 0) { /* We have some lookahead bytes to use */ if (avail >= len) /* Just use the lookahead buffer */ used = len; else used = avail; /* Copy, and adjust buffer pos */ memcpy(buf, AH->lookahead + AH->lookaheadPos, used); AH->lookaheadPos += used; /* Adjust required length */ len -= used; } /* Read the file if len > 0 */ if (len > 0) { if (fh) res = fread(&((char *) buf)[used], 1, len, fh); else if (th) { if (th->zFH) res = GZREAD(&((char *) buf)[used], 1, len, th->zFH); else res = fread(&((char *) buf)[used], 1, len, th->nFH); } else exit_horribly(modulename, "internal error -- neither th nor fh specified in tarReadRaw()\n"); } ctx->tarFHpos += res + used; return (res + used); }
static void _tarWriteHeader | ( | TAR_MEMBER * | th | ) | [static] |
Definition at line 1300 of file pg_backup_tar.c.
References exit_horribly(), TAR_MEMBER::fileLen, modulename, NULL, strerror(), tarCreateHeader(), TAR_MEMBER::tarFH, and TAR_MEMBER::targetFile.
Referenced by _tarAddFile().
{ char h[512]; tarCreateHeader(h, th->targetFile, NULL, th->fileLen, 0600, 04000, 02000, time(NULL)); /* Now write the completed header. */ if (fwrite(h, 1, 512, th->tarFH) != 512) exit_horribly(modulename, "could not write to output file: %s\n", strerror(errno)); }
static size_t _WriteBuf | ( | ArchiveHandle * | AH, | |
const void * | buf, | |||
size_t | len | |||
) | [static] |
Definition at line 793 of file pg_backup_tar.c.
References lclContext::FH, lclContext::filePos, _archiveHandle::formatData, and tarWrite().
{ lclContext *ctx = (lclContext *) AH->formatData; size_t res; res = tarWrite(buf, len, ctx->FH); ctx->filePos += res; return res; }
static int _WriteByte | ( | ArchiveHandle * | AH, | |
const int | i | |||
) | [static] |
Definition at line 766 of file pg_backup_tar.c.
References lclContext::FH, lclContext::filePos, _archiveHandle::formatData, and tarWrite().
{ lclContext *ctx = (lclContext *) AH->formatData; int res; char b = i; /* Avoid endian problems */ res = tarWrite(&b, 1, ctx->FH); if (res != EOF) ctx->filePos += res; return res; }
static size_t _WriteData | ( | ArchiveHandle * | AH, | |
const void * | data, | |||
size_t | dLen | |||
) | [static] |
Definition at line 605 of file pg_backup_tar.c.
References _archiveHandle::currToc, _tocEntry::formatData, tarWrite(), and lclTocEntry::TH.
{ lclTocEntry *tctx = (lclTocEntry *) AH->currToc->formatData; dLen = tarWrite(data, dLen, tctx->TH); return dLen; }
static void _WriteExtraToc | ( | ArchiveHandle * | AH, | |
TocEntry * | te | |||
) | [static] |
Definition at line 300 of file pg_backup_tar.c.
References lclTocEntry::filename, _tocEntry::formatData, and WriteStr().
{ lclTocEntry *ctx = (lclTocEntry *) te->formatData; if (ctx->filename) WriteStr(AH, ctx->filename); else WriteStr(AH, ""); }
void InitArchiveFmt_Tar | ( | ArchiveHandle * | AH | ) |
Definition at line 135 of file pg_backup_tar.c.
References archModeWrite, checkSeek(), exit_horribly(), lclContext::FH, lclContext::filePos, lclContext::hasSeek, lclContext::isSpecialScript, LOBBUFSIZE, modulename, NULL, PG_BINARY_R, PG_BINARY_W, pg_malloc(), pg_malloc0(), ReadHead(), ReadToc(), strerror(), tarClose(), lclContext::tarFH, lclContext::tarFHpos, tarOpen(), and Z_DEFAULT_COMPRESSION.
Referenced by _allocAH().
{ 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->ReopenPtr = NULL; AH->PrintTocDataPtr = _PrintTocData; AH->ReadExtraTocPtr = _ReadExtraToc; AH->WriteExtraTocPtr = _WriteExtraToc; AH->PrintExtraTocPtr = _PrintExtraToc; AH->StartBlobsPtr = _StartBlobs; AH->StartBlobPtr = _StartBlob; AH->EndBlobPtr = _EndBlob; AH->EndBlobsPtr = _EndBlobs; AH->ClonePtr = NULL; AH->DeClonePtr = NULL; AH->MasterStartParallelItemPtr = NULL; AH->MasterEndParallelItemPtr = NULL; AH->WorkerJobDumpPtr = NULL; AH->WorkerJobRestorePtr = NULL; /* * Set up some special context used in compressing data. */ ctx = (lclContext *) pg_malloc0(sizeof(lclContext)); AH->formatData = (void *) ctx; ctx->filePos = 0; ctx->isSpecialScript = 0; /* Initialize LO buffering */ AH->lo_buf_size = LOBBUFSIZE; AH->lo_buf = (void *) pg_malloc(LOBBUFSIZE); /* * Now open the tar file, and load the TOC if we're in read mode. */ if (AH->mode == archModeWrite) { if (AH->fSpec && strcmp(AH->fSpec, "") != 0) { ctx->tarFH = fopen(AH->fSpec, PG_BINARY_W); if (ctx->tarFH == NULL) exit_horribly(modulename, "could not open TOC file \"%s\" for output: %s\n", AH->fSpec, strerror(errno)); } else { ctx->tarFH = stdout; if (ctx->tarFH == NULL) exit_horribly(modulename, "could not open TOC file for output: %s\n", strerror(errno)); } ctx->tarFHpos = 0; /* * Make unbuffered since we will dup() it, and the buffers screw each * other */ /* setvbuf(ctx->tarFH, NULL, _IONBF, 0); */ ctx->hasSeek = checkSeek(ctx->tarFH); if (AH->compression < 0 || AH->compression > 9) AH->compression = Z_DEFAULT_COMPRESSION; /* Don't compress into tar files unless asked to do so */ if (AH->compression == Z_DEFAULT_COMPRESSION) AH->compression = 0; /* * We don't support compression because reading the files back is not * possible since gzdopen uses buffered IO which totally screws file * positioning. */ if (AH->compression != 0) exit_horribly(modulename, "compression is not supported by tar archive format\n"); } else { /* Read Mode */ if (AH->fSpec && strcmp(AH->fSpec, "") != 0) { ctx->tarFH = fopen(AH->fSpec, PG_BINARY_R); if (ctx->tarFH == NULL) exit_horribly(modulename, "could not open TOC file \"%s\" for input: %s\n", AH->fSpec, strerror(errno)); } else { ctx->tarFH = stdin; if (ctx->tarFH == NULL) exit_horribly(modulename, "could not open TOC file for input: %s\n", strerror(errno)); } /* * Make unbuffered since we will dup() it, and the buffers screw each * other */ /* setvbuf(ctx->tarFH, NULL, _IONBF, 0); */ ctx->tarFHpos = 0; ctx->hasSeek = checkSeek(ctx->tarFH); /* * Forcibly unmark the header as read since we use the lookahead * buffer */ AH->readHeader = 0; ctx->FH = (void *) tarOpen(AH, "toc.dat", 'r'); ReadHead(AH); ReadToc(AH); tarClose(AH, ctx->FH); /* Nothing else in the file... */ } }
bool isValidTarHeader | ( | char * | header | ) |
Definition at line 1029 of file pg_backup_tar.c.
References memcmp(), and tarChecksum().
Referenced by _discoverArchiveFormat().
{ int sum; int chk = tarChecksum(header); sscanf(&header[148], "%8o", &sum); if (sum != chk) return false; /* POSIX tar format */ if (memcmp(&header[257], "ustar\0", 6) == 0 && memcmp(&header[263], "00", 2) == 0) return true; /* GNU tar format */ if (memcmp(&header[257], "ustar \0", 8) == 0) return true; /* not-quite-POSIX format written by pre-9.3 pg_dump */ if (memcmp(&header[257], "ustar00\0", 8) == 0) return true; return false; }
static void tarClose | ( | ArchiveHandle * | AH, | |
TAR_MEMBER * | TH | |||
) | [static] |
Definition at line 453 of file pg_backup_tar.c.
References _tarAddFile(), _archiveHandle::compression, exit_horribly(), free, GZCLOSE, TAR_MEMBER::mode, modulename, TAR_MEMBER::nFH, TAR_MEMBER::targetFile, and TAR_MEMBER::zFH.
Referenced by _CloseArchive(), _EndBlob(), _EndBlobs(), _EndData(), _LoadBlobs(), _PrintFileData(), and InitArchiveFmt_Tar().
{ /* * Close the GZ file since we dup'd. This will flush the buffers. */ if (AH->compression != 0) if (GZCLOSE(th->zFH) != 0) exit_horribly(modulename, "could not close tar member\n"); if (th->mode == 'w') _tarAddFile(AH, th); /* This will close the temp file */ /* * else Nothing to do for normal read since we don't dup() normal file * handle, and we don't use temp files. */ if (th->targetFile) free(th->targetFile); th->nFH = NULL; th->zFH = NULL; }
static TAR_MEMBER * tarOpen | ( | ArchiveHandle * | AH, | |
const char * | filename, | |||
char | mode | |||
) | [static] |
Definition at line 348 of file pg_backup_tar.c.
References _tarPositionTo(), _archiveHandle::compression, exit_horribly(), _archiveHandle::formatData, free, modulename, name, NULL, pg_malloc0(), pg_strdup(), strerror(), lclContext::tarFH, and tm.
Referenced by _CloseArchive(), _LoadBlobs(), _PrintFileData(), _StartBlob(), _StartBlobs(), _StartData(), and InitArchiveFmt_Tar().
{ lclContext *ctx = (lclContext *) AH->formatData; TAR_MEMBER *tm; #ifdef HAVE_LIBZ char fmode[10]; #endif if (mode == 'r') { tm = _tarPositionTo(AH, filename); if (!tm) /* Not found */ { if (filename) { /* * Couldn't find the requested file. Future: do SEEK(0) and * retry. */ exit_horribly(modulename, "could not find file \"%s\" in archive\n", filename); } else { /* Any file OK, none left, so return NULL */ return NULL; } } #ifdef HAVE_LIBZ if (AH->compression == 0) tm->nFH = ctx->tarFH; else exit_horribly(modulename, "compression is not supported by tar archive format\n"); /* tm->zFH = gzdopen(dup(fileno(ctx->tarFH)), "rb"); */ #else tm->nFH = ctx->tarFH; #endif } else { tm = pg_malloc0(sizeof(TAR_MEMBER)); #ifndef WIN32 tm->tmpFH = tmpfile(); #else /* * On WIN32, tmpfile() generates a filename in the root directory, * which requires administrative permissions on certain systems. Loop * until we find a unique file name we can create. */ while (1) { char *name; int fd; name = _tempnam(NULL, "pg_temp_"); if (name == NULL) break; fd = open(name, O_RDWR | O_CREAT | O_EXCL | O_BINARY | O_TEMPORARY, S_IRUSR | S_IWUSR); free(name); if (fd != -1) /* created a file */ { tm->tmpFH = fdopen(fd, "w+b"); break; } else if (errno != EEXIST) /* failure other than file exists */ break; } #endif if (tm->tmpFH == NULL) exit_horribly(modulename, "could not generate temporary file name: %s\n", strerror(errno)); #ifdef HAVE_LIBZ if (AH->compression != 0) { sprintf(fmode, "wb%d", AH->compression); tm->zFH = gzdopen(dup(fileno(tm->tmpFH)), fmode); if (tm->zFH == NULL) exit_horribly(modulename, "could not open temporary file\n"); } else tm->nFH = tm->tmpFH; #else tm->nFH = tm->tmpFH; #endif tm->AH = AH; tm->targetFile = pg_strdup(filename); } tm->mode = mode; tm->tarFH = ctx->tarFH; return tm; }
static int tarPrintf | ( | ArchiveHandle * | AH, | |
TAR_MEMBER * | th, | |||
const char * | fmt, | |||
... | ||||
) | [static] |
Definition at line 997 of file pg_backup_tar.c.
References free, NULL, pg_malloc(), tarWrite(), and vsnprintf().
Referenced by _CloseArchive(), and _StartBlob().
{ char *p = NULL; va_list ap; size_t bSize = strlen(fmt) + 256; /* Should be enough */ int cnt = -1; /* * This is paranoid: deal with the possibility that vsnprintf is willing * to ignore trailing null */ /* * or returns > 0 even if string does not fit. It may be the case that it * returns cnt = bufsize */ while (cnt < 0 || cnt >= (bSize - 1)) { if (p != NULL) free(p); bSize *= 2; p = (char *) pg_malloc(bSize); va_start(ap, fmt); cnt = vsnprintf(p, bSize, fmt, ap); va_end(ap); } cnt = tarWrite(p, cnt, th); free(p); return cnt; }
static size_t tarRead | ( | void * | buf, | |
size_t | len, | |||
TAR_MEMBER * | th | |||
) | [static] |
Definition at line 569 of file pg_backup_tar.c.
References _tarReadRaw(), TAR_MEMBER::AH, TAR_MEMBER::fileLen, NULL, and TAR_MEMBER::pos.
Referenced by _LoadBlobs(), _PrintFileData(), _ReadBuf(), and _ReadByte().
static size_t tarWrite | ( | const void * | buf, | |
size_t | len, | |||
TAR_MEMBER * | th | |||
) | [static] |
Definition at line 587 of file pg_backup_tar.c.
References exit_horribly(), GZWRITE, modulename, TAR_MEMBER::nFH, NULL, TAR_MEMBER::pos, strerror(), and TAR_MEMBER::zFH.
Referenced by _scriptOut(), _WriteBuf(), _WriteByte(), _WriteData(), and tarPrintf().
const char* modulename = gettext_noop("tar archiver") [static] |
Definition at line 109 of file pg_backup_tar.c.
Referenced by _CloseArchive(), _PrintTocData(), _ReadByte(), _StartBlob(), _tarAddFile(), _tarGetHeader(), _tarPositionTo(), _tarReadRaw(), _tarWriteHeader(), InitArchiveFmt_Tar(), tarClose(), tarOpen(), and tarWrite().