pg_backup_tar.c 27.4 KB
Newer Older
P
Philip Warner 已提交
1 2 3 4 5 6 7 8 9 10
/*-------------------------------------------------------------------------
 *
 * pg_backup_tar.c
 *
 *	This file is copied from the 'files' format file, but dumps data into
 *	one temp file then sends it to the output TAR archive.
 *
 *	See the headers to pg_backup_files & pg_restore for more details.
 *
 * Copyright (c) 2000, Philip Warner
B
Bruce Momjian 已提交
11 12
 *		Rights are granted to use this software in any way so long
 *		as this notice is not removed.
P
Philip Warner 已提交
13 14 15 16 17 18
 *
 *	The author is not responsible for loss or damages that may
 *	result from it's use.
 *
 *
 * IDENTIFICATION
19
 *		$Header: /cvsroot/pgsql/src/bin/pg_dump/pg_backup_tar.c,v 1.21 2002/04/24 02:21:04 momjian Exp $
P
Philip Warner 已提交
20 21 22
 *
 * Modifications - 28-Jun-2000 - pjw@rhyme.com.au
 *
B
Bruce Momjian 已提交
23
 *	Initial version.
P
Philip Warner 已提交
24
 *
25 26
 * Modifications - 04-Jan-2001 - pjw@rhyme.com.au
 *
B
Bruce Momjian 已提交
27
 *	  - Check results of IO routines more carefully.
28
 *
P
Philip Warner 已提交
29 30 31
 *-------------------------------------------------------------------------
 */

32 33 34 35
#include "pg_backup.h"
#include "pg_backup_archiver.h"
#include "pg_backup_tar.h"

P
Philip Warner 已提交
36 37 38 39 40
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
#include <unistd.h>

B
Bruce Momjian 已提交
41 42 43 44 45 46 47 48 49 50 51 52 53 54 55
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 _StartBlobs(ArchiveHandle *AH, TocEntry *te);
56 57
static void _StartBlob(ArchiveHandle *AH, TocEntry *te, Oid oid);
static void _EndBlob(ArchiveHandle *AH, TocEntry *te, Oid oid);
B
Bruce Momjian 已提交
58
static void _EndBlobs(ArchiveHandle *AH, TocEntry *te);
P
Philip Warner 已提交
59 60 61 62 63

#define K_STD_BUF_SIZE 1024


#ifdef HAVE_LIBZ
B
Bruce Momjian 已提交
64 65 66
 /* typedef gzFile	 ThingFile; */
typedef FILE ThingFile;

P
Philip Warner 已提交
67
#else
B
Bruce Momjian 已提交
68
typedef FILE ThingFile;
P
Philip Warner 已提交
69 70
#endif

B
Bruce Momjian 已提交
71 72 73 74 75 76 77 78 79 80 81
typedef struct
{
	ThingFile  *zFH;
	FILE	   *nFH;
	FILE	   *tarFH;
	FILE	   *tmpFH;
	char	   *targetFile;
	char		mode;
	int			pos;
	int			fileLen;
	ArchiveHandle *AH;
P
Philip Warner 已提交
82 83
} TAR_MEMBER;

B
Bruce Momjian 已提交
84 85
typedef struct
{
P
Philip Warner 已提交
86
	int			hasSeek;
B
Bruce Momjian 已提交
87 88 89
	int			filePos;
	TAR_MEMBER *blobToc;
	FILE	   *tarFH;
P
Philip Warner 已提交
90 91
	int			tarFHpos;
	int			tarNextMember;
B
Bruce Momjian 已提交
92
	TAR_MEMBER *FH;
P
Philip Warner 已提交
93
	int			isSpecialScript;
B
Bruce Momjian 已提交
94
	TAR_MEMBER *scriptTH;
P
Philip Warner 已提交
95 96
} lclContext;

B
Bruce Momjian 已提交
97 98 99 100
typedef struct
{
	TAR_MEMBER *TH;
	char	   *filename;
P
Philip Warner 已提交
101 102
} lclTocEntry;

103
static char *modulename = gettext_noop("tar archiver");
B
Bruce Momjian 已提交
104 105

static void _LoadBlobs(ArchiveHandle *AH, RestoreOptions *ropt);
P
Philip Warner 已提交
106

B
Bruce Momjian 已提交
107 108
static TAR_MEMBER *tarOpen(ArchiveHandle *AH, const char *filename, char mode);
static void tarClose(ArchiveHandle *AH, TAR_MEMBER *TH);
P
Philip Warner 已提交
109 110

#ifdef __NOT_USED__
B
Bruce Momjian 已提交
111
static char *tarGets(char *buf, int len, TAR_MEMBER *th);
P
Philip Warner 已提交
112
#endif
B
Bruce Momjian 已提交
113
static int	tarPrintf(ArchiveHandle *AH, TAR_MEMBER *th, const char *fmt,...);
P
Philip Warner 已提交
114

B
Bruce Momjian 已提交
115 116 117 118 119 120 121 122
static void _tarAddFile(ArchiveHandle *AH, TAR_MEMBER *th);
static int	_tarChecksum(char *th);
static TAR_MEMBER *_tarPositionTo(ArchiveHandle *AH, const char *filename);
static int	tarRead(void *buf, int len, TAR_MEMBER *th);
static int	tarWrite(const void *buf, int len, TAR_MEMBER *th);
static void _tarWriteHeader(TAR_MEMBER *th);
static int	_tarGetHeader(ArchiveHandle *AH, TAR_MEMBER *th);
static int	_tarReadRaw(ArchiveHandle *AH, void *buf, int len, TAR_MEMBER *th, FILE *fh);
P
Philip Warner 已提交
123

B
Bruce Momjian 已提交
124
static int	_scriptOut(ArchiveHandle *AH, const void *buf, int len);
P
Philip Warner 已提交
125 126

/*
B
Bruce Momjian 已提交
127
 *	Initializer
P
Philip Warner 已提交
128
 */
B
Bruce Momjian 已提交
129 130
void
InitArchiveFmt_Tar(ArchiveHandle *AH)
P
Philip Warner 已提交
131
{
B
Bruce Momjian 已提交
132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159
	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;

	AH->StartBlobsPtr = _StartBlobs;
	AH->StartBlobPtr = _StartBlob;
	AH->EndBlobPtr = _EndBlob;
	AH->EndBlobsPtr = _EndBlobs;

	/*
	 * Set up some special context used in compressing data.
	 */
	ctx = (lclContext *) malloc(sizeof(lclContext));
	AH->formatData = (void *) ctx;
	ctx->filePos = 0;
160 161 162 163 164 165
	
	/* Initialize LO buffering */
	AH->lo_buf_size = LOBBUFSIZE;
	AH->lo_buf = (void *)malloc(LOBBUFSIZE);
	if(AH->lo_buf == NULL)
                die_horribly(AH, modulename, "out of memory\n");
B
Bruce Momjian 已提交
166 167 168 169 170 171 172 173

	/*
	 * Now open the TOC file
	 */
	if (AH->mode == archModeWrite)
	{

		if (AH->fSpec && strcmp(AH->fSpec, "") != 0)
P
Philip Warner 已提交
174
			ctx->tarFH = fopen(AH->fSpec, PG_BINARY_W);
B
Bruce Momjian 已提交
175
		else
P
Philip Warner 已提交
176
			ctx->tarFH = stdout;
177

B
Bruce Momjian 已提交
178
		if (ctx->tarFH == NULL)
179
			die_horribly(NULL, modulename,
180
			"could not open TOC file for output: %s\n", strerror(errno));
181

P
Philip Warner 已提交
182 183
		ctx->tarFHpos = 0;

B
Bruce Momjian 已提交
184 185 186 187
		/*
		 * Make unbuffered since we will dup() it, and the buffers screw
		 * each other
		 */
188
		/* setvbuf(ctx->tarFH, NULL, _IONBF, 0); */
P
Philip Warner 已提交
189 190 191

		ctx->hasSeek = (fseek(ctx->tarFH, 0, SEEK_CUR) == 0);

B
Bruce Momjian 已提交
192
		if (AH->compression < 0 || AH->compression > 9)
P
Philip Warner 已提交
193 194 195 196 197 198
			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;

B
Bruce Momjian 已提交
199 200 201 202
		/*
		 * We don't support compression because reading the files back is
		 * not possible since gzdopen uses buffered IO which totally
		 * screws file positioning.
P
Philip Warner 已提交
203 204
		 */
		if (AH->compression != 0)
205
			die_horribly(NULL, modulename, "compression not supported by tar output format\n");
P
Philip Warner 已提交
206

B
Bruce Momjian 已提交
207 208 209
	}
	else
	{							/* Read Mode */
P
Philip Warner 已提交
210

B
Bruce Momjian 已提交
211
		if (AH->fSpec && strcmp(AH->fSpec, "") != 0)
P
Philip Warner 已提交
212
			ctx->tarFH = fopen(AH->fSpec, PG_BINARY_R);
B
Bruce Momjian 已提交
213
		else
P
Philip Warner 已提交
214 215
			ctx->tarFH = stdin;

216
		if (ctx->tarFH == NULL)
217
			die_horribly(NULL, modulename, "could not open TOC file for input: %s\n", strerror(errno));
218

B
Bruce Momjian 已提交
219 220 221 222
		/*
		 * Make unbuffered since we will dup() it, and the buffers screw
		 * each other
		 */
223
		/* setvbuf(ctx->tarFH, NULL, _IONBF, 0); */
P
Philip Warner 已提交
224 225 226 227 228

		ctx->tarFHpos = 0;

		ctx->hasSeek = (fseek(ctx->tarFH, 0, SEEK_CUR) == 0);

B
Bruce Momjian 已提交
229 230 231 232
		/*
		 * Forcibly unmark the header as read since we use the lookahead
		 * buffer
		 */
P
Philip Warner 已提交
233 234
		AH->readHeader = 0;

B
Bruce Momjian 已提交
235
		ctx->FH = (void *) tarOpen(AH, "toc.dat", 'r');
P
Philip Warner 已提交
236 237
		ReadHead(AH);
		ReadToc(AH);
B
Bruce Momjian 已提交
238 239
		tarClose(AH, ctx->FH);	/* Nothing else in the file... */
	}
P
Philip Warner 已提交
240 241 242 243 244

}

/*
 * - Start a new TOC entry
B
Bruce Momjian 已提交
245
 *	 Setup the output file name.
P
Philip Warner 已提交
246
 */
B
Bruce Momjian 已提交
247 248
static void
_ArchiveEntry(ArchiveHandle *AH, TocEntry *te)
P
Philip Warner 已提交
249
{
B
Bruce Momjian 已提交
250 251
	lclTocEntry *ctx;
	char		fn[K_STD_BUF_SIZE];
P
Philip Warner 已提交
252

B
Bruce Momjian 已提交
253
	ctx = (lclTocEntry *) malloc(sizeof(lclTocEntry));
254
	if (te->dataDumper != NULL)
B
Bruce Momjian 已提交
255
	{
P
Philip Warner 已提交
256
#ifdef HAVE_LIBZ
B
Bruce Momjian 已提交
257
		if (AH->compression == 0)
P
Philip Warner 已提交
258
			sprintf(fn, "%d.dat", te->id);
B
Bruce Momjian 已提交
259
		else
P
Philip Warner 已提交
260 261 262 263 264
			sprintf(fn, "%d.dat.gz", te->id);
#else
		sprintf(fn, "%d.dat", te->id);
#endif
		ctx->filename = strdup(fn);
B
Bruce Momjian 已提交
265 266 267
	}
	else
	{
P
Philip Warner 已提交
268 269
		ctx->filename = NULL;
		ctx->TH = NULL;
B
Bruce Momjian 已提交
270 271
	}
	te->formatData = (void *) ctx;
P
Philip Warner 已提交
272 273
}

B
Bruce Momjian 已提交
274 275
static void
_WriteExtraToc(ArchiveHandle *AH, TocEntry *te)
P
Philip Warner 已提交
276
{
B
Bruce Momjian 已提交
277
	lclTocEntry *ctx = (lclTocEntry *) te->formatData;
P
Philip Warner 已提交
278

B
Bruce Momjian 已提交
279
	if (ctx->filename)
P
Philip Warner 已提交
280
		WriteStr(AH, ctx->filename);
B
Bruce Momjian 已提交
281
	else
P
Philip Warner 已提交
282 283 284
		WriteStr(AH, "");
}

B
Bruce Momjian 已提交
285 286
static void
_ReadExtraToc(ArchiveHandle *AH, TocEntry *te)
P
Philip Warner 已提交
287
{
B
Bruce Momjian 已提交
288
	lclTocEntry *ctx = (lclTocEntry *) te->formatData;
P
Philip Warner 已提交
289

B
Bruce Momjian 已提交
290 291 292 293 294
	if (ctx == NULL)
	{
		ctx = (lclTocEntry *) malloc(sizeof(lclTocEntry));
		te->formatData = (void *) ctx;
	}
P
Philip Warner 已提交
295

B
Bruce Momjian 已提交
296 297 298
	ctx->filename = ReadStr(AH);
	if (strlen(ctx->filename) == 0)
	{
P
Philip Warner 已提交
299 300
		free(ctx->filename);
		ctx->filename = NULL;
B
Bruce Momjian 已提交
301 302
	}
	ctx->TH = NULL;
P
Philip Warner 已提交
303 304
}

B
Bruce Momjian 已提交
305 306
static void
_PrintExtraToc(ArchiveHandle *AH, TocEntry *te)
P
Philip Warner 已提交
307
{
B
Bruce Momjian 已提交
308
	lclTocEntry *ctx = (lclTocEntry *) te->formatData;
P
Philip Warner 已提交
309

310 311
	if (ctx->filename != NULL)
		ahprintf(AH, "-- File: %s\n", ctx->filename);
P
Philip Warner 已提交
312 313
}

B
Bruce Momjian 已提交
314 315
static void
_StartData(ArchiveHandle *AH, TocEntry *te)
P
Philip Warner 已提交
316
{
B
Bruce Momjian 已提交
317
	lclTocEntry *tctx = (lclTocEntry *) te->formatData;
P
Philip Warner 已提交
318 319 320 321

	tctx->TH = tarOpen(AH, tctx->filename, 'w');
}

B
Bruce Momjian 已提交
322 323
static TAR_MEMBER *
tarOpen(ArchiveHandle *AH, const char *filename, char mode)
P
Philip Warner 已提交
324
{
B
Bruce Momjian 已提交
325 326 327
	lclContext *ctx = (lclContext *) AH->formatData;
	TAR_MEMBER *tm;

P
Philip Warner 已提交
328
#ifdef HAVE_LIBZ
B
Bruce Momjian 已提交
329
	char		fmode[10];
P
Philip Warner 已提交
330 331 332 333 334
#endif

	if (mode == 'r')
	{
		tm = _tarPositionTo(AH, filename);
B
Bruce Momjian 已提交
335
		if (!tm)				/* Not found */
P
Philip Warner 已提交
336
		{
B
Bruce Momjian 已提交
337 338
			if (filename)		/* Couldn't find the requested file.
								 * Future: DO SEEK(0) and retry. */
339
				die_horribly(AH, modulename, "could not find file %s in archive\n", filename);
B
Bruce Momjian 已提交
340 341
			else
/* Any file OK, non left, so return NULL */
P
Philip Warner 已提交
342 343 344 345 346 347 348 349
				return NULL;
		}

#ifdef HAVE_LIBZ

		if (AH->compression == 0)
			tm->nFH = ctx->tarFH;
		else
350
			die_horribly(AH, modulename, "compression support is disabled in this format\n");
B
Bruce Momjian 已提交
351
		/* tm->zFH = gzdopen(dup(fileno(ctx->tarFH)), "rb"); */
P
Philip Warner 已提交
352 353 354 355 356 357

#else

		tm->nFH = ctx->tarFH;
#endif

B
Bruce Momjian 已提交
358 359 360
	}
	else
	{
P
Philip Warner 已提交
361 362 363 364
		tm = calloc(1, sizeof(TAR_MEMBER));

		tm->tmpFH = tmpfile();

B
Bruce Momjian 已提交
365
		if (tm->tmpFH == NULL)
366
			die_horribly(AH, modulename, "could not generate temporary file name: %s\n", strerror(errno));
367

P
Philip Warner 已提交
368 369 370 371 372 373
#ifdef HAVE_LIBZ

		if (AH->compression != 0)
		{
			sprintf(fmode, "wb%d", AH->compression);
			tm->zFH = gzdopen(dup(fileno(tm->tmpFH)), fmode);
374
			if (tm->zFH == NULL)
375
				die_horribly(AH, modulename, "could not gzdopen temporary file\n");
376

B
Bruce Momjian 已提交
377 378
		}
		else
P
Philip Warner 已提交
379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396
			tm->nFH = tm->tmpFH;

#else

		tm->nFH = tm->tmpFH;
#endif

		tm->AH = AH;
		tm->targetFile = strdup(filename);
	}

	tm->mode = mode;
	tm->tarFH = ctx->tarFH;

	return tm;

}

B
Bruce Momjian 已提交
397 398
static void
tarClose(ArchiveHandle *AH, TAR_MEMBER *th)
P
Philip Warner 已提交
399 400 401 402 403
{
	/*
	 * Close the GZ file since we dup'd. This will flush the buffers.
	 */
	if (AH->compression != 0)
404
		if (GZCLOSE(th->zFH) != 0)
405
			die_horribly(AH, modulename, "could not close tar member\n");
P
Philip Warner 已提交
406 407

	if (th->mode == 'w')
B
Bruce Momjian 已提交
408 409 410 411 412
		_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.
P
Philip Warner 已提交
413 414
	 */

B
Bruce Momjian 已提交
415
	if (th->targetFile)
P
Philip Warner 已提交
416 417 418 419 420 421 422
		free(th->targetFile);

	th->nFH = NULL;
	th->zFH = NULL;
}

#ifdef __NOT_USED__
B
Bruce Momjian 已提交
423 424
static char *
tarGets(char *buf, int len, TAR_MEMBER *th)
P
Philip Warner 已提交
425
{
B
Bruce Momjian 已提交
426 427 428 429
	char	   *s;
	int			cnt = 0;
	char		c = ' ';
	int			eof = 0;
P
Philip Warner 已提交
430 431 432 433 434 435 436

	/* Can't read past logical EOF */
	if (len > (th->fileLen - th->pos))
		len = th->fileLen - th->pos;

	while (cnt < len && c != '\n')
	{
B
Bruce Momjian 已提交
437 438
		if (_tarReadRaw(th->AH, &c, 1, th, NULL) <= 0)
		{
P
Philip Warner 已提交
439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454
			eof = 1;
			break;
		}
		buf[cnt++] = c;
	}

	if (eof && cnt == 0)
		s = NULL;
	else
	{
		buf[cnt++] = '\0';
		s = buf;
	}

	if (s)
	{
B
Bruce Momjian 已提交
455
		len = strlen(s);
P
Philip Warner 已提交
456 457 458 459 460 461 462
		th->pos += len;
	}

	return s;
}
#endif

B
Bruce Momjian 已提交
463
/*
P
Philip Warner 已提交
464 465 466
 * Just read bytes from the archive. This is the low level read routine
 * that is used for ALL reads on a tar file.
 */
B
Bruce Momjian 已提交
467 468
static int
_tarReadRaw(ArchiveHandle *AH, void *buf, int len, TAR_MEMBER *th, FILE *fh)
P
Philip Warner 已提交
469
{
B
Bruce Momjian 已提交
470 471 472 473
	lclContext *ctx = (lclContext *) AH->formatData;
	int			avail;
	int			used = 0;
	int			res = 0;
P
Philip Warner 已提交
474 475 476 477 478

	avail = AH->lookaheadLen - AH->lookaheadPos;
	if (avail > 0)
	{
		/* We have some lookahead bytes to use */
B
Bruce Momjian 已提交
479
		if (avail >= len)		/* Just use the lookahead buffer */
P
Philip Warner 已提交
480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495
			used = len;
		else
			used = avail;

		/* Copy, and adjust buffer pos */
		memcpy(buf, AH->lookahead, used);
		AH->lookaheadPos += used;

		/* Adjust required length */
		len -= used;
	}

	/* Read the file if len > 0 */
	if (len > 0)
	{
		if (fh)
B
Bruce Momjian 已提交
496
			res = fread(&((char *) buf)[used], 1, len, fh);
P
Philip Warner 已提交
497 498 499
		else if (th)
		{
			if (th->zFH)
B
Bruce Momjian 已提交
500
				res = GZREAD(&((char *) buf)[used], 1, len, th->zFH);
P
Philip Warner 已提交
501
			else
B
Bruce Momjian 已提交
502 503
				res = fread(&((char *) buf)[used], 1, len, th->nFH);
		}
P
Philip Warner 已提交
504
		else
505
			die_horribly(AH, modulename, "neither th nor fh specified in tarReadRaw() (internal error)\n");
P
Philip Warner 已提交
506 507
	}

508 509 510 511
#if 0
	write_msg(modulename, "requested %d bytes, got %d from lookahead and %d from file\n",
			  reqLen, used, res);
#endif
P
Philip Warner 已提交
512 513 514 515 516

	ctx->tarFHpos += res + used;

	return (res + used);
}
B
Bruce Momjian 已提交
517 518 519

static int
tarRead(void *buf, int len, TAR_MEMBER *th)
P
Philip Warner 已提交
520
{
B
Bruce Momjian 已提交
521
	int			res;
P
Philip Warner 已提交
522 523 524 525 526 527 528 529 530 531 532 533 534 535

	if (th->pos + len > th->fileLen)
		len = th->fileLen - th->pos;

	if (len <= 0)
		return 0;

	res = _tarReadRaw(th->AH, buf, len, th, NULL);

	th->pos += res;

	return res;
}

B
Bruce Momjian 已提交
536 537
static int
tarWrite(const void *buf, int len, TAR_MEMBER *th)
P
Philip Warner 已提交
538
{
B
Bruce Momjian 已提交
539
	int			res;
P
Philip Warner 已提交
540 541

	if (th->zFH != 0)
B
Bruce Momjian 已提交
542
		res = GZWRITE((void *) buf, 1, len, th->zFH);
P
Philip Warner 已提交
543 544 545
	else
		res = fwrite(buf, 1, len, th->nFH);

546
	if (res != len)
547 548
		die_horribly(th->AH, modulename,
					 "could not write to tar member (wrote %d, attempted %d)\n", res, len);
549

P
Philip Warner 已提交
550 551 552 553
	th->pos += res;
	return res;
}

B
Bruce Momjian 已提交
554 555
static int
_WriteData(ArchiveHandle *AH, const void *data, int dLen)
P
Philip Warner 已提交
556
{
B
Bruce Momjian 已提交
557
	lclTocEntry *tctx = (lclTocEntry *) AH->currToc->formatData;
P
Philip Warner 已提交
558

B
Bruce Momjian 已提交
559
	dLen = tarWrite((void *) data, dLen, tctx->TH);
P
Philip Warner 已提交
560

B
Bruce Momjian 已提交
561
	return dLen;
P
Philip Warner 已提交
562 563
}

B
Bruce Momjian 已提交
564 565
static void
_EndData(ArchiveHandle *AH, TocEntry *te)
P
Philip Warner 已提交
566
{
B
Bruce Momjian 已提交
567
	lclTocEntry *tctx = (lclTocEntry *) te->formatData;
P
Philip Warner 已提交
568

B
Bruce Momjian 已提交
569 570 571
	/* Close the file */
	tarClose(AH, tctx->TH);
	tctx->TH = NULL;
P
Philip Warner 已提交
572 573
}

B
Bruce Momjian 已提交
574 575
/*
 * Print data for a given file
P
Philip Warner 已提交
576
 */
B
Bruce Momjian 已提交
577 578
static void
_PrintFileData(ArchiveHandle *AH, char *filename, RestoreOptions *ropt)
P
Philip Warner 已提交
579
{
B
Bruce Momjian 已提交
580 581 582 583
	lclContext *ctx = (lclContext *) AH->formatData;
	char		buf[4096];
	int			cnt;
	TAR_MEMBER *th;
P
Philip Warner 已提交
584

B
Bruce Momjian 已提交
585
	if (!filename)
P
Philip Warner 已提交
586 587 588 589 590
		return;

	th = tarOpen(AH, filename, 'r');
	ctx->FH = th;

B
Bruce Momjian 已提交
591 592
	while ((cnt = tarRead(buf, 4095, th)) > 0)
	{
P
Philip Warner 已提交
593 594
		buf[cnt] = '\0';
		ahwrite(buf, 1, cnt, AH);
B
Bruce Momjian 已提交
595
	}
P
Philip Warner 已提交
596

B
Bruce Momjian 已提交
597
	tarClose(AH, th);
P
Philip Warner 已提交
598 599 600 601 602 603
}


/*
 * Print data for a given TOC entry
*/
B
Bruce Momjian 已提交
604 605
static void
_PrintTocData(ArchiveHandle *AH, TocEntry *te, RestoreOptions *ropt)
P
Philip Warner 已提交
606
{
B
Bruce Momjian 已提交
607 608 609 610 611 612 613 614
	lclContext *ctx = (lclContext *) AH->formatData;
	lclTocEntry *tctx = (lclTocEntry *) te->formatData;
	char	   *tmpCopy;
	int			i,
				pos1,
				pos2;

	if (!tctx->filename)
P
Philip Warner 已提交
615 616 617 618 619 620 621 622 623 624 625 626
		return;

	if (ctx->isSpecialScript)
	{
		if (!te->copyStmt)
			return;

		/* Abort the default COPY */
		ahprintf(AH, "\\.\n");

		/* Get a copy of the COPY statement and clean it up */
		tmpCopy = strdup(te->copyStmt);
B
Bruce Momjian 已提交
627
		for (i = 0; i < strlen(tmpCopy); i++)
628
			tmpCopy[i] = tolower((unsigned char) tmpCopy[i]);
P
Philip Warner 已提交
629 630

		/*
B
Bruce Momjian 已提交
631 632
		 * This is very nasty; we don't know if the archive used WITH
		 * OIDS, so we search the string for it in a paranoid sort of way.
P
Philip Warner 已提交
633 634
		 */
		if (strncmp(tmpCopy, "copy ", 5) != 0)
635 636
			die_horribly(AH, modulename,
						 "bad COPY statement - could not find \"copy\" in string \"%s\"\n", tmpCopy);
P
Philip Warner 已提交
637 638 639 640

		pos1 = 5;
		for (pos1 = 5; pos1 < strlen(tmpCopy); pos1++)
			if (tmpCopy[pos1] != ' ')
B
Bruce Momjian 已提交
641
				break;
P
Philip Warner 已提交
642 643 644

		if (tmpCopy[pos1] == '"')
			pos1 += 2;
B
Bruce Momjian 已提交
645

P
Philip Warner 已提交
646 647
		pos1 += strlen(te->name);

B
Bruce Momjian 已提交
648
		for (pos2 = pos1; pos2 < strlen(tmpCopy); pos2++)
P
Philip Warner 已提交
649 650 651 652
			if (strncmp(&tmpCopy[pos2], "from stdin", 10) == 0)
				break;

		if (pos2 >= strlen(tmpCopy))
653 654 655
			die_horribly(AH, modulename,
						 "bad COPY statement - could not find \"from stdin\" in string \"%s\" starting at position %d\n",
						 tmpCopy, pos1);
P
Philip Warner 已提交
656

B
Bruce Momjian 已提交
657 658
		ahwrite(tmpCopy, 1, pos2, AH);	/* 'copy "table" [with oids]' */
		ahprintf(AH, " from '$$PATH$$/%s' %s", tctx->filename, &tmpCopy[pos2 + 10]);
P
Philip Warner 已提交
659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696

		return;
	}

	if (strcmp(te->desc, "BLOBS") == 0)
		_LoadBlobs(AH, ropt);
	else
		_PrintFileData(AH, tctx->filename, ropt);
}

/* static void _getBlobTocEntry(ArchiveHandle* AH, int *oid, char fname[K_STD_BUF_SIZE])
 * {
 *	lclContext*		ctx = (lclContext*)AH->formatData;
 *	char			blobTe[K_STD_BUF_SIZE];
 *	int				fpos;
 *	int				eos;
 *
 *	if (tarGets(&blobTe[0], K_STD_BUF_SIZE - 1, ctx->blobToc) != NULL)
 *	{
 *		*oid = atoi(blobTe);
 *
 *		fpos = strcspn(blobTe, " ");
 *
 *		strncpy(fname, &blobTe[fpos+1], K_STD_BUF_SIZE - 1);
 *
 *		eos = strlen(fname)-1;
 *
 *		if (fname[eos] == '\n')
 *			fname[eos] = '\0';
 *
 *	} else {
 *
 *		*oid = 0;
 *		fname[0] = '\0';
 *	}
 *}
 */

B
Bruce Momjian 已提交
697 698
static void
_LoadBlobs(ArchiveHandle *AH, RestoreOptions *ropt)
P
Philip Warner 已提交
699
{
700
	Oid			oid;
B
Bruce Momjian 已提交
701 702 703 704
	lclContext *ctx = (lclContext *) AH->formatData;
	TAR_MEMBER *th;
	int			cnt;
	char		buf[4096];
P
Philip Warner 已提交
705

706 707
	StartRestoreBlobs(AH);

708
	th = tarOpen(AH, NULL, 'r');	/* Open next file */
P
Philip Warner 已提交
709 710 711 712
	while (th != NULL)
	{
		ctx->FH = th;

713
		oid = (Oid) strtoul(&th->targetFile[5], NULL, 10);
P
Philip Warner 已提交
714

B
Bruce Momjian 已提交
715
		if (strncmp(th->targetFile, "blob_", 5) == 0 && oid != 0)
P
Philip Warner 已提交
716
		{
717
			ahlog(AH, 1, "restoring large object OID %u\n", oid);
P
Philip Warner 已提交
718 719 720

			StartRestoreBlob(AH, oid);

B
Bruce Momjian 已提交
721 722
			while ((cnt = tarRead(buf, 4095, th)) > 0)
			{
P
Philip Warner 已提交
723 724 725 726 727 728 729 730 731 732 733
				buf[cnt] = '\0';
				ahwrite(buf, 1, cnt, AH);
			}
			EndRestoreBlob(AH, oid);
		}

		tarClose(AH, th);

		th = tarOpen(AH, NULL, 'r');
	}

734 735
	EndRestoreBlobs(AH);

P
Philip Warner 已提交
736 737 738
}


B
Bruce Momjian 已提交
739 740
static int
_WriteByte(ArchiveHandle *AH, const int i)
P
Philip Warner 已提交
741
{
B
Bruce Momjian 已提交
742 743 744
	lclContext *ctx = (lclContext *) AH->formatData;
	int			res;
	char		b = i;			/* Avoid endian problems */
P
Philip Warner 已提交
745

B
Bruce Momjian 已提交
746 747
	res = tarWrite(&b, 1, ctx->FH);
	if (res != EOF)
P
Philip Warner 已提交
748
		ctx->filePos += res;
B
Bruce Momjian 已提交
749
	return res;
P
Philip Warner 已提交
750 751
}

B
Bruce Momjian 已提交
752 753
static int
_ReadByte(ArchiveHandle *AH)
P
Philip Warner 已提交
754
{
B
Bruce Momjian 已提交
755 756 757
	lclContext *ctx = (lclContext *) AH->formatData;
	int			res;
	char		c = '\0';
P
Philip Warner 已提交
758

B
Bruce Momjian 已提交
759 760
	res = tarRead(&c, 1, ctx->FH);
	if (res != EOF)
P
Philip Warner 已提交
761
		ctx->filePos += res;
B
Bruce Momjian 已提交
762
	return c;
P
Philip Warner 已提交
763 764
}

B
Bruce Momjian 已提交
765 766
static int
_WriteBuf(ArchiveHandle *AH, const void *buf, int len)
P
Philip Warner 已提交
767
{
B
Bruce Momjian 已提交
768 769
	lclContext *ctx = (lclContext *) AH->formatData;
	int			res;
P
Philip Warner 已提交
770

B
Bruce Momjian 已提交
771 772 773
	res = tarWrite((void *) buf, len, ctx->FH);
	ctx->filePos += res;
	return res;
P
Philip Warner 已提交
774 775
}

B
Bruce Momjian 已提交
776 777
static int
_ReadBuf(ArchiveHandle *AH, void *buf, int len)
P
Philip Warner 已提交
778
{
B
Bruce Momjian 已提交
779 780
	lclContext *ctx = (lclContext *) AH->formatData;
	int			res;
P
Philip Warner 已提交
781

B
Bruce Momjian 已提交
782 783 784
	res = tarRead(buf, len, ctx->FH);
	ctx->filePos += res;
	return res;
P
Philip Warner 已提交
785 786
}

B
Bruce Momjian 已提交
787 788
static void
_CloseArchive(ArchiveHandle *AH)
P
Philip Warner 已提交
789
{
B
Bruce Momjian 已提交
790 791 792 793 794
	lclContext *ctx = (lclContext *) AH->formatData;
	TAR_MEMBER *th;
	RestoreOptions *ropt;
	int			savVerbose,
				i;
P
Philip Warner 已提交
795

B
Bruce Momjian 已提交
796 797
	if (AH->mode == archModeWrite)
	{
P
Philip Warner 已提交
798 799 800 801 802 803 804
		/*
		 * Write the Header & TOC to the archive FIRST
		 */
		th = tarOpen(AH, "toc.dat", 'w');
		ctx->FH = th;
		WriteHead(AH);
		WriteToc(AH);
B
Bruce Momjian 已提交
805
		tarClose(AH, th);		/* Not needed any more */
P
Philip Warner 已提交
806 807 808 809 810 811

		/*
		 * Now send the data (tables & blobs)
		 */
		WriteDataChunks(AH);

B
Bruce Momjian 已提交
812 813 814
		/*
		 * Now this format wants to append a script which does a full
		 * restore if the files have been extracted.
P
Philip Warner 已提交
815 816 817
		 */
		th = tarOpen(AH, "restore.sql", 'w');
		tarPrintf(AH, th, "create temporary table pgdump_restore_path(p text);\n");
B
Bruce Momjian 已提交
818 819 820 821 822 823 824 825 826 827
		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"
				  "-- Edit the following to match the path where the\n"
				  "-- tar archive has been extracted.\n"
				  "--\n");
P
Philip Warner 已提交
828 829 830
		tarPrintf(AH, th, "insert into pgdump_restore_path values('/tmp');\n\n");

		AH->CustomOutPtr = _scriptOut;
B
Bruce Momjian 已提交
831

P
Philip Warner 已提交
832 833 834 835 836 837
		ctx->isSpecialScript = 1;
		ctx->scriptTH = th;

		ropt = NewRestoreOptions();
		ropt->dropSchema = 1;
		ropt->compression = 0;
838
		ropt->superuser = PQuser(AH->connection);
839
		ropt->suppressDumpWarnings = true;
P
Philip Warner 已提交
840 841 842 843

		savVerbose = AH->public.verbose;
		AH->public.verbose = 0;

B
Bruce Momjian 已提交
844
		RestoreArchive((Archive *) AH, ropt);
P
Philip Warner 已提交
845 846 847 848

		AH->public.verbose = savVerbose;

		tarClose(AH, th);
849 850

		/* Add a block of NULLs since it's de-rigeur. */
B
Bruce Momjian 已提交
851
		for (i = 0; i < 512; i++)
852
		{
853
			if (fputc(0, ctx->tarFH) == EOF)
854
				die_horribly(AH, modulename,
855
				   "could not write null block at end of tar archive\n");
856 857
		}

B
Bruce Momjian 已提交
858
	}
P
Philip Warner 已提交
859

B
Bruce Momjian 已提交
860
	AH->FH = NULL;
P
Philip Warner 已提交
861 862
}

B
Bruce Momjian 已提交
863 864
static int
_scriptOut(ArchiveHandle *AH, const void *buf, int len)
P
Philip Warner 已提交
865
{
B
Bruce Momjian 已提交
866 867
	lclContext *ctx = (lclContext *) AH->formatData;

P
Philip Warner 已提交
868 869 870 871 872 873 874 875
	return tarWrite(buf, len, ctx->scriptTH);
}

/*
 * BLOB support
 */

/*
B
Bruce Momjian 已提交
876
 * Called by the archiver when starting to save all BLOB DATA (not schema).
P
Philip Warner 已提交
877
 * This routine should save whatever format-specific information is needed
B
Bruce Momjian 已提交
878
 * to read the BLOBs back into memory.
P
Philip Warner 已提交
879 880 881 882 883 884
 *
 * It is called just prior to the dumper's DataDumper routine.
 *
 * Optional, but strongly recommended.
 *
 */
B
Bruce Momjian 已提交
885 886
static void
_StartBlobs(ArchiveHandle *AH, TocEntry *te)
P
Philip Warner 已提交
887
{
B
Bruce Momjian 已提交
888 889
	lclContext *ctx = (lclContext *) AH->formatData;
	char		fname[K_STD_BUF_SIZE];
P
Philip Warner 已提交
890 891 892 893 894 895 896 897 898 899 900 901 902

	sprintf(fname, "blobs.toc");
	ctx->blobToc = tarOpen(AH, fname, 'w');

}

/*
 * Called by the archiver when the dumper calls StartBlob.
 *
 * Mandatory.
 *
 * Must save the passed OID for retrieval at restore-time.
 */
B
Bruce Momjian 已提交
903
static void
904
_StartBlob(ArchiveHandle *AH, TocEntry *te, Oid oid)
P
Philip Warner 已提交
905
{
B
Bruce Momjian 已提交
906 907 908 909
	lclContext *ctx = (lclContext *) AH->formatData;
	lclTocEntry *tctx = (lclTocEntry *) te->formatData;
	char		fname[255];
	char	   *sfx;
P
Philip Warner 已提交
910

B
Bruce Momjian 已提交
911
	if (oid == 0)
912
		die_horribly(AH, modulename, "invalid OID for large object (%u)\n", oid);
P
Philip Warner 已提交
913 914 915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930 931 932

	if (AH->compression != 0)
		sfx = ".gz";
	else
		sfx = "";

	sprintf(fname, "blob_%d.dat%s", oid, sfx);

	tarPrintf(AH, ctx->blobToc, "%d %s\n", oid, fname);

	tctx->TH = tarOpen(AH, fname, 'w');

}

/*
 * Called by the archiver when the dumper calls EndBlob.
 *
 * Optional.
 *
 */
B
Bruce Momjian 已提交
933
static void
934
_EndBlob(ArchiveHandle *AH, TocEntry *te, Oid oid)
P
Philip Warner 已提交
935
{
B
Bruce Momjian 已提交
936
	lclTocEntry *tctx = (lclTocEntry *) te->formatData;
P
Philip Warner 已提交
937 938 939 940 941

	tarClose(AH, tctx->TH);
}

/*
B
Bruce Momjian 已提交
942
 * Called by the archiver when finishing saving all BLOB DATA.
P
Philip Warner 已提交
943 944 945 946
 *
 * Optional.
 *
 */
B
Bruce Momjian 已提交
947 948
static void
_EndBlobs(ArchiveHandle *AH, TocEntry *te)
P
Philip Warner 已提交
949
{
B
Bruce Momjian 已提交
950 951
	lclContext *ctx = (lclContext *) AH->formatData;

P
Philip Warner 已提交
952
	/* Write out a fake zero OID to mark end-of-blobs. */
B
Bruce Momjian 已提交
953
	/* WriteInt(AH, 0); */
P
Philip Warner 已提交
954 955 956 957 958 959 960 961 962 963 964 965

	tarClose(AH, ctx->blobToc);

}



/*------------
 * TAR Support
 *------------
 */

B
Bruce Momjian 已提交
966 967
static int
tarPrintf(ArchiveHandle *AH, TAR_MEMBER *th, const char *fmt,...)
P
Philip Warner 已提交
968
{
B
Bruce Momjian 已提交
969
	char	   *p = NULL;
P
Philip Warner 已提交
970
	va_list		ap;
B
Bruce Momjian 已提交
971
	int			bSize = strlen(fmt) + 256;		/* Should be enough */
P
Philip Warner 已提交
972 973
	int			cnt = -1;

B
Bruce Momjian 已提交
974 975 976 977 978 979 980 981 982 983
	/*
	 * 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))
984
	{
B
Bruce Momjian 已提交
985 986
		if (p != NULL)
			free(p);
P
Philip Warner 已提交
987
		bSize *= 2;
B
Bruce Momjian 已提交
988
		p = (char *) malloc(bSize);
P
Philip Warner 已提交
989
		if (p == NULL)
990
			die_horribly(AH, modulename, "out of memory\n");
991
		va_start(ap, fmt);
P
Philip Warner 已提交
992
		cnt = vsnprintf(p, bSize, fmt, ap);
993
		va_end(ap);
P
Philip Warner 已提交
994 995 996 997 998 999
	}
	cnt = tarWrite(p, cnt, th);
	free(p);
	return cnt;
}

B
Bruce Momjian 已提交
1000 1001
static int
_tarChecksum(char *header)
P
Philip Warner 已提交
1002
{
B
Bruce Momjian 已提交
1003 1004 1005
	int			i,
				sum;

P
Philip Warner 已提交
1006
	sum = 0;
B
Bruce Momjian 已提交
1007
	for (i = 0; i < 512; i++)
P
Philip Warner 已提交
1008 1009
		if (i < 148 || i >= 156)
			sum += 0xFF & header[i];
B
Bruce Momjian 已提交
1010
	return sum + 256;			/* Assume 8 blanks in checksum field */
P
Philip Warner 已提交
1011 1012
}

B
Bruce Momjian 已提交
1013 1014
int
isValidTarHeader(char *header)
P
Philip Warner 已提交
1015
{
B
Bruce Momjian 已提交
1016 1017
	int			sum;
	int			chk = _tarChecksum(header);
P
Philip Warner 已提交
1018 1019 1020 1021 1022 1023 1024

	sscanf(&header[148], "%8o", &sum);

	return (sum == chk && strncmp(&header[257], "ustar  ", 7) == 0);
}

/* Given the member, write the TAR header & copy the file */
B
Bruce Momjian 已提交
1025 1026
static void
_tarAddFile(ArchiveHandle *AH, TAR_MEMBER *th)
P
Philip Warner 已提交
1027
{
B
Bruce Momjian 已提交
1028
	lclContext *ctx = (lclContext *) AH->formatData;
1029
	FILE	   *tmp = th->tmpFH;	/* Grab it for convenience */
P
Philip Warner 已提交
1030 1031 1032
	char		buf[32768];
	int			cnt;
	int			len = 0;
1033
	int			res;
B
Bruce Momjian 已提交
1034 1035
	int			i,
				pad;
P
Philip Warner 已提交
1036 1037 1038 1039 1040 1041 1042 1043 1044 1045

	/*
	 * Find file len & go back to start.
	 */
	fseek(tmp, 0, SEEK_END);
	th->fileLen = ftell(tmp);
	fseek(tmp, 0, SEEK_SET);

	_tarWriteHeader(th);

B
Bruce Momjian 已提交
1046
	while ((cnt = fread(&buf[0], 1, 32767, tmp)) > 0)
P
Philip Warner 已提交
1047
	{
1048
		res = fwrite(&buf[0], 1, cnt, th->tarFH);
B
Bruce Momjian 已提交
1049
		if (res != cnt)
1050
			die_horribly(AH, modulename, "write error appending to tar archive (wrote %d, attempted %d)\n", res, cnt);
1051
		len += res;
P
Philip Warner 已提交
1052 1053
	}

B
Bruce Momjian 已提交
1054
	if (fclose(tmp) != 0)		/* This *should* delete it... */
1055
		die_horribly(AH, modulename, "could not close tar member: %s\n", strerror(errno));
P
Philip Warner 已提交
1056 1057

	if (len != th->fileLen)
1058 1059
		die_horribly(AH, modulename, "actual file length (%d) does not match expected (%d)\n",
					 len, th->pos);
P
Philip Warner 已提交
1060 1061

	pad = ((len + 511) & ~511) - len;
B
Bruce Momjian 已提交
1062
	for (i = 0; i < pad; i++)
1063
	{
B
Bruce Momjian 已提交
1064
		if (fputc('\0', th->tarFH) == EOF)
1065
			die_horribly(AH, modulename, "could not output padding at end of tar member\n");
B
Bruce Momjian 已提交
1066
	}
P
Philip Warner 已提交
1067 1068 1069 1070 1071

	ctx->tarFHpos += len + pad;
}

/* Locate the file in the archive, read header and position to data */
B
Bruce Momjian 已提交
1072 1073
static TAR_MEMBER *
_tarPositionTo(ArchiveHandle *AH, const char *filename)
P
Philip Warner 已提交
1074
{
B
Bruce Momjian 已提交
1075 1076 1077 1078 1079 1080 1081 1082
	lclContext *ctx = (lclContext *) AH->formatData;
	TAR_MEMBER *th = calloc(1, sizeof(TAR_MEMBER));
	char		c;
	char		header[512];
	int			i,
				len,
				blks,
				id;
P
Philip Warner 已提交
1083 1084 1085 1086 1087 1088

	th->AH = AH;

	/* Go to end of current file, if any */
	if (ctx->tarFHpos != 0)
	{
1089
		ahlog(AH, 4, "moving from position %d (%x) to next member at file position %d (%x)\n",
B
Bruce Momjian 已提交
1090 1091
			  ctx->tarFHpos, ctx->tarFHpos,
			  ctx->tarNextMember, ctx->tarNextMember);
P
Philip Warner 已提交
1092 1093 1094 1095 1096

		while (ctx->tarFHpos < ctx->tarNextMember)
			_tarReadRaw(AH, &c, 1, NULL, ctx->tarFH);
	}

1097
	ahlog(AH, 4, "now at file position %d (%x)\n", ctx->tarFHpos, ctx->tarFHpos);
P
Philip Warner 已提交
1098 1099 1100 1101 1102 1103 1104

	/* We are at the start of the file. or at the next member */

	/* Get the header */
	if (!_tarGetHeader(AH, th))
	{
		if (filename)
1105
			die_horribly(AH, modulename, "could not find header for file %s in tar archive\n", filename);
B
Bruce Momjian 已提交
1106 1107
		else
/* We're just scanning the archibe for the next file, so return null */
P
Philip Warner 已提交
1108 1109 1110 1111 1112 1113
		{
			free(th);
			return NULL;
		}
	}

B
Bruce Momjian 已提交
1114
	while (filename != NULL && strcmp(th->targetFile, filename) != 0)
P
Philip Warner 已提交
1115
	{
1116
		ahlog(AH, 4, "skipping tar member %s\n", th->targetFile);
P
Philip Warner 已提交
1117 1118 1119

		id = atoi(th->targetFile);
		if ((TocIDRequired(AH, id, AH->ropt) & 2) != 0)
1120
			die_horribly(AH, modulename, "dumping data out of order is not supported in this archive format: "
1121
			"%s is required, but comes before %s in the archive file.\n",
1122
						 th->targetFile, filename);
P
Philip Warner 已提交
1123 1124

		/* Header doesn't match, so read to next header */
B
Bruce Momjian 已提交
1125 1126
		len = ((th->fileLen + 511) & ~511);		/* Padded length */
		blks = len >> 9;		/* # of 512 byte blocks */
P
Philip Warner 已提交
1127

B
Bruce Momjian 已提交
1128
		for (i = 0; i < blks; i++)
P
Philip Warner 已提交
1129 1130 1131
			_tarReadRaw(AH, &header[0], 512, NULL, ctx->tarFH);

		if (!_tarGetHeader(AH, th))
1132
			die_horribly(AH, modulename, "could not find header for file %s in tar archive\n", filename);
P
Philip Warner 已提交
1133 1134 1135 1136 1137 1138 1139 1140 1141 1142

	}

	ctx->tarNextMember = ctx->tarFHpos + ((th->fileLen + 511) & ~511);
	th->pos = 0;

	return th;
}

/* Read & verify a header */
B
Bruce Momjian 已提交
1143 1144
static int
_tarGetHeader(ArchiveHandle *AH, TAR_MEMBER *th)
P
Philip Warner 已提交
1145
{
B
Bruce Momjian 已提交
1146 1147 1148 1149 1150 1151 1152 1153 1154
	lclContext *ctx = (lclContext *) AH->formatData;
	char		h[512];
	char		name[100];
	int			sum,
				chk;
	int			len;
	int			hPos;
	int			i;
	bool		gotBlock = false;
P
Philip Warner 已提交
1155

1156 1157
	while (!gotBlock)
	{
1158 1159 1160 1161 1162 1163
#if 0
		if (ftell(ctx->tarFH) != ctx->tarFHpos)
			die_horribly(AH, modulename,
						 "mismatch in actual vs. predicted file position (%d vs. %d)\n",
						 ftell(ctx->tarFH), ctx->tarFHpos);
#endif
P
Philip Warner 已提交
1164

1165 1166
		/* Save the pos for reporting purposes */
		hPos = ctx->tarFHpos;
P
Philip Warner 已提交
1167

1168 1169
		/* Read a 512 byte block, return EOF, exit if short */
		len = _tarReadRaw(AH, &h[0], 512, NULL, ctx->tarFH);
B
Bruce Momjian 已提交
1170
		if (len == 0)			/* EOF */
1171 1172 1173
			return 0;

		if (len != 512)
1174
			die_horribly(AH, modulename, "incomplete tar header found (%d bytes)\n", len);
1175 1176 1177

		/* Calc checksum */
		chk = _tarChecksum(&h[0]);
P
Philip Warner 已提交
1178

1179
		/*
B
Bruce Momjian 已提交
1180 1181
		 * If the checksum failed, see if it is a null block. If so, then
		 * just try with next block...
1182 1183
		 */

B
Bruce Momjian 已提交
1184
		if (chk == sum)
1185
			gotBlock = true;
B
Bruce Momjian 已提交
1186 1187 1188
		else
		{
			for (i = 0; i < 512; i++)
1189
			{
B
Bruce Momjian 已提交
1190
				if (h[0] != 0)
1191
				{
B
Bruce Momjian 已提交
1192
					gotBlock = true;
1193 1194 1195 1196 1197
					break;
				}
			}
		}
	}
P
Philip Warner 已提交
1198 1199 1200 1201 1202

	sscanf(&h[0], "%99s", &name[0]);
	sscanf(&h[124], "%12o", &len);
	sscanf(&h[148], "%8o", &sum);

1203
	ahlog(AH, 3, "TOC Entry %s at %d (length %d, checksum %d)\n", &name[0], hPos, len, sum);
P
Philip Warner 已提交
1204 1205

	if (chk != sum)
1206 1207
		die_horribly(AH, modulename,
					 "corrupt tar header found in %s "
1208
		"(expected %d (%o), computed %d (%o)) file position %ld (%lx)\n",
1209
					 &name[0], sum, sum, chk, chk, ftell(ctx->tarFH), ftell(ctx->tarFH));
P
Philip Warner 已提交
1210 1211 1212 1213 1214 1215 1216

	th->targetFile = strdup(name);
	th->fileLen = len;

	return 1;
}

B
Bruce Momjian 已提交
1217 1218
static void
_tarWriteHeader(TAR_MEMBER *th)
P
Philip Warner 已提交
1219 1220 1221 1222 1223 1224
{
	char		h[512];
	int			i;
	int			lastSum = 0;
	int			sum;

B
Bruce Momjian 已提交
1225
	for (i = 0; i < 512; i++)
P
Philip Warner 已提交
1226 1227 1228 1229 1230 1231 1232 1233 1234
		h[i] = '\0';

	/* Name 100 */
	sprintf(&h[0], "%.99s", th->targetFile);

	/* Mode 8 */
	sprintf(&h[100], "100600 ");

	/* User ID 8 */
P
Philip Warner 已提交
1235
	sprintf(&h[108], " 04000 ");
P
Philip Warner 已提交
1236 1237

	/* Group 8 */
P
Philip Warner 已提交
1238
	sprintf(&h[116], " 02000 ");
P
Philip Warner 已提交
1239 1240

	/* File size 12 */
P
Philip Warner 已提交
1241
	sprintf(&h[124], "%10o ", th->fileLen);
P
Philip Warner 已提交
1242 1243

	/* Mod Time 12 */
B
Bruce Momjian 已提交
1244
	sprintf(&h[136], "%10o ", (int) time(NULL));
P
Philip Warner 已提交
1245 1246

	/* Checksum 8 */
P
Philip Warner 已提交
1247
	sprintf(&h[148], "%6o ", lastSum);
P
Philip Warner 已提交
1248

P
Philip Warner 已提交
1249 1250 1251
	/* Type 1 */
	/* sprintf(&h[156], "%c", LF_NORMAL); */
	sprintf(&h[156], "0");
P
Philip Warner 已提交
1252 1253 1254 1255 1256 1257

	/* Link name 100 (NULL) */

	/* Magic 8 */
	sprintf(&h[257], "ustar  ");

B
Bruce Momjian 已提交
1258 1259 1260
	/*
	 * GNU Version... sprintf(&h[257], "ustar"); sprintf(&h[263], "00");
	 */
P
Philip Warner 已提交
1261

P
Philip Warner 已提交
1262
	/* User 32 */
B
Bruce Momjian 已提交
1263 1264
	sprintf(&h[265], "%.31s", "");		/* How do I get username reliably?
										 * Do I need to? */
P
Philip Warner 已提交
1265 1266

	/* Group 32 */
B
Bruce Momjian 已提交
1267 1268
	sprintf(&h[297], "%.31s", "");		/* How do I get group reliably? Do
										 * I need to? */
P
Philip Warner 已提交
1269 1270

	/* Maj Dev 8 */
P
Philip Warner 已提交
1271
	/* sprintf(&h[329], "%6o ", 0); */
P
Philip Warner 已提交
1272 1273

	/* Min Dev */
P
Philip Warner 已提交
1274
	/* sprintf(&h[337], "%6o ", 0); */
P
Philip Warner 已提交
1275 1276


B
Bruce Momjian 已提交
1277
	while ((sum = _tarChecksum(h)) != lastSum)
P
Philip Warner 已提交
1278
	{
P
Philip Warner 已提交
1279
		sprintf(&h[148], "%6o ", sum);
P
Philip Warner 已提交
1280 1281 1282
		lastSum = sum;
	}

B
Bruce Momjian 已提交
1283
	if (fwrite(h, 1, 512, th->tarFH) != 512)
1284
		die_horribly(th->AH, modulename, "unable to write tar header\n");
1285

P
Philip Warner 已提交
1286
}