pg_backup_tar.c 26.8 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.23 2002/05/29 01:38:56 tgl 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

		return;
	}

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

B
Bruce Momjian 已提交
669 670
static void
_LoadBlobs(ArchiveHandle *AH, RestoreOptions *ropt)
P
Philip Warner 已提交
671
{
672
	Oid			oid;
B
Bruce Momjian 已提交
673 674 675 676
	lclContext *ctx = (lclContext *) AH->formatData;
	TAR_MEMBER *th;
	int			cnt;
	char		buf[4096];
P
Philip Warner 已提交
677

678 679
	StartRestoreBlobs(AH);

680
	th = tarOpen(AH, NULL, 'r');	/* Open next file */
P
Philip Warner 已提交
681 682 683 684
	while (th != NULL)
	{
		ctx->FH = th;

685
		if (strncmp(th->targetFile, "blob_", 5) == 0)
P
Philip Warner 已提交
686
		{
687 688 689 690
			oid = atooid(&th->targetFile[5]);
			if (oid != 0)
			{
				ahlog(AH, 1, "restoring large object OID %u\n", oid);
P
Philip Warner 已提交
691

692
				StartRestoreBlob(AH, oid);
P
Philip Warner 已提交
693

694 695 696 697 698 699
				while ((cnt = tarRead(buf, 4095, th)) > 0)
				{
					buf[cnt] = '\0';
					ahwrite(buf, 1, cnt, AH);
				}
				EndRestoreBlob(AH, oid);
P
Philip Warner 已提交
700 701 702 703 704 705 706 707
			}
		}

		tarClose(AH, th);

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

708 709
	EndRestoreBlobs(AH);

P
Philip Warner 已提交
710 711 712
}


B
Bruce Momjian 已提交
713 714
static int
_WriteByte(ArchiveHandle *AH, const int i)
P
Philip Warner 已提交
715
{
B
Bruce Momjian 已提交
716 717 718
	lclContext *ctx = (lclContext *) AH->formatData;
	int			res;
	char		b = i;			/* Avoid endian problems */
P
Philip Warner 已提交
719

B
Bruce Momjian 已提交
720 721
	res = tarWrite(&b, 1, ctx->FH);
	if (res != EOF)
P
Philip Warner 已提交
722
		ctx->filePos += res;
B
Bruce Momjian 已提交
723
	return res;
P
Philip Warner 已提交
724 725
}

B
Bruce Momjian 已提交
726 727
static int
_ReadByte(ArchiveHandle *AH)
P
Philip Warner 已提交
728
{
B
Bruce Momjian 已提交
729 730 731
	lclContext *ctx = (lclContext *) AH->formatData;
	int			res;
	char		c = '\0';
P
Philip Warner 已提交
732

B
Bruce Momjian 已提交
733 734
	res = tarRead(&c, 1, ctx->FH);
	if (res != EOF)
P
Philip Warner 已提交
735
		ctx->filePos += res;
B
Bruce Momjian 已提交
736
	return c;
P
Philip Warner 已提交
737 738
}

B
Bruce Momjian 已提交
739 740
static int
_WriteBuf(ArchiveHandle *AH, const void *buf, int len)
P
Philip Warner 已提交
741
{
B
Bruce Momjian 已提交
742 743
	lclContext *ctx = (lclContext *) AH->formatData;
	int			res;
P
Philip Warner 已提交
744

B
Bruce Momjian 已提交
745 746 747
	res = tarWrite((void *) buf, len, ctx->FH);
	ctx->filePos += res;
	return res;
P
Philip Warner 已提交
748 749
}

B
Bruce Momjian 已提交
750 751
static int
_ReadBuf(ArchiveHandle *AH, void *buf, int len)
P
Philip Warner 已提交
752
{
B
Bruce Momjian 已提交
753 754
	lclContext *ctx = (lclContext *) AH->formatData;
	int			res;
P
Philip Warner 已提交
755

B
Bruce Momjian 已提交
756 757 758
	res = tarRead(buf, len, ctx->FH);
	ctx->filePos += res;
	return res;
P
Philip Warner 已提交
759 760
}

B
Bruce Momjian 已提交
761 762
static void
_CloseArchive(ArchiveHandle *AH)
P
Philip Warner 已提交
763
{
B
Bruce Momjian 已提交
764 765 766 767 768
	lclContext *ctx = (lclContext *) AH->formatData;
	TAR_MEMBER *th;
	RestoreOptions *ropt;
	int			savVerbose,
				i;
P
Philip Warner 已提交
769

B
Bruce Momjian 已提交
770 771
	if (AH->mode == archModeWrite)
	{
P
Philip Warner 已提交
772 773 774 775 776 777 778
		/*
		 * Write the Header & TOC to the archive FIRST
		 */
		th = tarOpen(AH, "toc.dat", 'w');
		ctx->FH = th;
		WriteHead(AH);
		WriteToc(AH);
B
Bruce Momjian 已提交
779
		tarClose(AH, th);		/* Not needed any more */
P
Philip Warner 已提交
780 781 782 783 784 785

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

B
Bruce Momjian 已提交
786 787 788
		/*
		 * Now this format wants to append a script which does a full
		 * restore if the files have been extracted.
P
Philip Warner 已提交
789 790 791
		 */
		th = tarOpen(AH, "restore.sql", 'w');
		tarPrintf(AH, th, "create temporary table pgdump_restore_path(p text);\n");
B
Bruce Momjian 已提交
792 793 794 795 796 797 798 799 800 801
		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 已提交
802 803 804
		tarPrintf(AH, th, "insert into pgdump_restore_path values('/tmp');\n\n");

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

P
Philip Warner 已提交
806 807 808 809 810 811
		ctx->isSpecialScript = 1;
		ctx->scriptTH = th;

		ropt = NewRestoreOptions();
		ropt->dropSchema = 1;
		ropt->compression = 0;
812
		ropt->superuser = NULL;
813
		ropt->suppressDumpWarnings = true;
P
Philip Warner 已提交
814 815 816 817

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

B
Bruce Momjian 已提交
818
		RestoreArchive((Archive *) AH, ropt);
P
Philip Warner 已提交
819 820 821 822

		AH->public.verbose = savVerbose;

		tarClose(AH, th);
823 824

		/* Add a block of NULLs since it's de-rigeur. */
B
Bruce Momjian 已提交
825
		for (i = 0; i < 512; i++)
826
		{
827
			if (fputc(0, ctx->tarFH) == EOF)
828
				die_horribly(AH, modulename,
829
				   "could not write null block at end of tar archive\n");
830 831
		}

B
Bruce Momjian 已提交
832
	}
P
Philip Warner 已提交
833

B
Bruce Momjian 已提交
834
	AH->FH = NULL;
P
Philip Warner 已提交
835 836
}

B
Bruce Momjian 已提交
837 838
static int
_scriptOut(ArchiveHandle *AH, const void *buf, int len)
P
Philip Warner 已提交
839
{
B
Bruce Momjian 已提交
840 841
	lclContext *ctx = (lclContext *) AH->formatData;

P
Philip Warner 已提交
842 843 844 845 846 847 848 849
	return tarWrite(buf, len, ctx->scriptTH);
}

/*
 * BLOB support
 */

/*
B
Bruce Momjian 已提交
850
 * Called by the archiver when starting to save all BLOB DATA (not schema).
P
Philip Warner 已提交
851
 * This routine should save whatever format-specific information is needed
B
Bruce Momjian 已提交
852
 * to read the BLOBs back into memory.
P
Philip Warner 已提交
853 854 855 856 857 858
 *
 * It is called just prior to the dumper's DataDumper routine.
 *
 * Optional, but strongly recommended.
 *
 */
B
Bruce Momjian 已提交
859 860
static void
_StartBlobs(ArchiveHandle *AH, TocEntry *te)
P
Philip Warner 已提交
861
{
B
Bruce Momjian 已提交
862 863
	lclContext *ctx = (lclContext *) AH->formatData;
	char		fname[K_STD_BUF_SIZE];
P
Philip Warner 已提交
864 865 866 867 868 869 870 871 872 873 874 875 876

	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 已提交
877
static void
878
_StartBlob(ArchiveHandle *AH, TocEntry *te, Oid oid)
P
Philip Warner 已提交
879
{
B
Bruce Momjian 已提交
880 881 882 883
	lclContext *ctx = (lclContext *) AH->formatData;
	lclTocEntry *tctx = (lclTocEntry *) te->formatData;
	char		fname[255];
	char	   *sfx;
P
Philip Warner 已提交
884

B
Bruce Momjian 已提交
885
	if (oid == 0)
886
		die_horribly(AH, modulename, "invalid OID for large object (%u)\n", oid);
P
Philip Warner 已提交
887 888 889 890 891 892

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

893
	sprintf(fname, "blob_%u.dat%s", oid, sfx);
P
Philip Warner 已提交
894

895
	tarPrintf(AH, ctx->blobToc, "%u %s\n", oid, fname);
P
Philip Warner 已提交
896 897 898 899 900 901 902 903 904 905 906

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

}

/*
 * Called by the archiver when the dumper calls EndBlob.
 *
 * Optional.
 *
 */
B
Bruce Momjian 已提交
907
static void
908
_EndBlob(ArchiveHandle *AH, TocEntry *te, Oid oid)
P
Philip Warner 已提交
909
{
B
Bruce Momjian 已提交
910
	lclTocEntry *tctx = (lclTocEntry *) te->formatData;
P
Philip Warner 已提交
911 912 913 914 915

	tarClose(AH, tctx->TH);
}

/*
B
Bruce Momjian 已提交
916
 * Called by the archiver when finishing saving all BLOB DATA.
P
Philip Warner 已提交
917 918 919 920
 *
 * Optional.
 *
 */
B
Bruce Momjian 已提交
921 922
static void
_EndBlobs(ArchiveHandle *AH, TocEntry *te)
P
Philip Warner 已提交
923
{
B
Bruce Momjian 已提交
924 925
	lclContext *ctx = (lclContext *) AH->formatData;

P
Philip Warner 已提交
926
	/* Write out a fake zero OID to mark end-of-blobs. */
B
Bruce Momjian 已提交
927
	/* WriteInt(AH, 0); */
P
Philip Warner 已提交
928 929 930 931 932 933 934 935 936 937 938 939

	tarClose(AH, ctx->blobToc);

}



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

B
Bruce Momjian 已提交
940 941
static int
tarPrintf(ArchiveHandle *AH, TAR_MEMBER *th, const char *fmt,...)
P
Philip Warner 已提交
942
{
B
Bruce Momjian 已提交
943
	char	   *p = NULL;
P
Philip Warner 已提交
944
	va_list		ap;
B
Bruce Momjian 已提交
945
	int			bSize = strlen(fmt) + 256;		/* Should be enough */
P
Philip Warner 已提交
946 947
	int			cnt = -1;

B
Bruce Momjian 已提交
948 949 950 951 952 953 954 955 956 957
	/*
	 * 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))
958
	{
B
Bruce Momjian 已提交
959 960
		if (p != NULL)
			free(p);
P
Philip Warner 已提交
961
		bSize *= 2;
B
Bruce Momjian 已提交
962
		p = (char *) malloc(bSize);
P
Philip Warner 已提交
963
		if (p == NULL)
964
			die_horribly(AH, modulename, "out of memory\n");
965
		va_start(ap, fmt);
P
Philip Warner 已提交
966
		cnt = vsnprintf(p, bSize, fmt, ap);
967
		va_end(ap);
P
Philip Warner 已提交
968 969 970 971 972 973
	}
	cnt = tarWrite(p, cnt, th);
	free(p);
	return cnt;
}

B
Bruce Momjian 已提交
974 975
static int
_tarChecksum(char *header)
P
Philip Warner 已提交
976
{
B
Bruce Momjian 已提交
977 978 979
	int			i,
				sum;

P
Philip Warner 已提交
980
	sum = 0;
B
Bruce Momjian 已提交
981
	for (i = 0; i < 512; i++)
P
Philip Warner 已提交
982 983
		if (i < 148 || i >= 156)
			sum += 0xFF & header[i];
B
Bruce Momjian 已提交
984
	return sum + 256;			/* Assume 8 blanks in checksum field */
P
Philip Warner 已提交
985 986
}

B
Bruce Momjian 已提交
987 988
int
isValidTarHeader(char *header)
P
Philip Warner 已提交
989
{
B
Bruce Momjian 已提交
990 991
	int			sum;
	int			chk = _tarChecksum(header);
P
Philip Warner 已提交
992 993 994 995 996 997 998

	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 已提交
999 1000
static void
_tarAddFile(ArchiveHandle *AH, TAR_MEMBER *th)
P
Philip Warner 已提交
1001
{
B
Bruce Momjian 已提交
1002
	lclContext *ctx = (lclContext *) AH->formatData;
1003
	FILE	   *tmp = th->tmpFH;	/* Grab it for convenience */
P
Philip Warner 已提交
1004 1005 1006
	char		buf[32768];
	int			cnt;
	int			len = 0;
1007
	int			res;
B
Bruce Momjian 已提交
1008 1009
	int			i,
				pad;
P
Philip Warner 已提交
1010 1011 1012 1013 1014 1015 1016 1017 1018 1019

	/*
	 * 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 已提交
1020
	while ((cnt = fread(&buf[0], 1, 32767, tmp)) > 0)
P
Philip Warner 已提交
1021
	{
1022
		res = fwrite(&buf[0], 1, cnt, th->tarFH);
B
Bruce Momjian 已提交
1023
		if (res != cnt)
1024
			die_horribly(AH, modulename, "write error appending to tar archive (wrote %d, attempted %d)\n", res, cnt);
1025
		len += res;
P
Philip Warner 已提交
1026 1027
	}

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

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

	pad = ((len + 511) & ~511) - len;
B
Bruce Momjian 已提交
1036
	for (i = 0; i < pad; i++)
1037
	{
B
Bruce Momjian 已提交
1038
		if (fputc('\0', th->tarFH) == EOF)
1039
			die_horribly(AH, modulename, "could not output padding at end of tar member\n");
B
Bruce Momjian 已提交
1040
	}
P
Philip Warner 已提交
1041 1042 1043 1044 1045

	ctx->tarFHpos += len + pad;
}

/* Locate the file in the archive, read header and position to data */
B
Bruce Momjian 已提交
1046 1047
static TAR_MEMBER *
_tarPositionTo(ArchiveHandle *AH, const char *filename)
P
Philip Warner 已提交
1048
{
B
Bruce Momjian 已提交
1049 1050 1051 1052 1053 1054 1055 1056
	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 已提交
1057 1058 1059 1060 1061 1062

	th->AH = AH;

	/* Go to end of current file, if any */
	if (ctx->tarFHpos != 0)
	{
1063
		ahlog(AH, 4, "moving from position %d (%x) to next member at file position %d (%x)\n",
B
Bruce Momjian 已提交
1064 1065
			  ctx->tarFHpos, ctx->tarFHpos,
			  ctx->tarNextMember, ctx->tarNextMember);
P
Philip Warner 已提交
1066 1067 1068 1069 1070

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

1071
	ahlog(AH, 4, "now at file position %d (%x)\n", ctx->tarFHpos, ctx->tarFHpos);
P
Philip Warner 已提交
1072 1073 1074 1075 1076 1077 1078

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

	/* Get the header */
	if (!_tarGetHeader(AH, th))
	{
		if (filename)
1079
			die_horribly(AH, modulename, "could not find header for file %s in tar archive\n", filename);
B
Bruce Momjian 已提交
1080 1081
		else
/* We're just scanning the archibe for the next file, so return null */
P
Philip Warner 已提交
1082 1083 1084 1085 1086 1087
		{
			free(th);
			return NULL;
		}
	}

B
Bruce Momjian 已提交
1088
	while (filename != NULL && strcmp(th->targetFile, filename) != 0)
P
Philip Warner 已提交
1089
	{
1090
		ahlog(AH, 4, "skipping tar member %s\n", th->targetFile);
P
Philip Warner 已提交
1091 1092 1093

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

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

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

		if (!_tarGetHeader(AH, th))
1106
			die_horribly(AH, modulename, "could not find header for file %s in tar archive\n", filename);
P
Philip Warner 已提交
1107 1108 1109 1110 1111 1112 1113 1114 1115 1116

	}

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

	return th;
}

/* Read & verify a header */
B
Bruce Momjian 已提交
1117 1118
static int
_tarGetHeader(ArchiveHandle *AH, TAR_MEMBER *th)
P
Philip Warner 已提交
1119
{
B
Bruce Momjian 已提交
1120 1121 1122 1123 1124 1125 1126 1127 1128
	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 已提交
1129

1130 1131
	while (!gotBlock)
	{
1132 1133 1134 1135 1136 1137
#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 已提交
1138

1139 1140
		/* Save the pos for reporting purposes */
		hPos = ctx->tarFHpos;
P
Philip Warner 已提交
1141

1142 1143
		/* Read a 512 byte block, return EOF, exit if short */
		len = _tarReadRaw(AH, &h[0], 512, NULL, ctx->tarFH);
B
Bruce Momjian 已提交
1144
		if (len == 0)			/* EOF */
1145 1146 1147
			return 0;

		if (len != 512)
1148
			die_horribly(AH, modulename, "incomplete tar header found (%d bytes)\n", len);
1149 1150 1151

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

1153
		/*
B
Bruce Momjian 已提交
1154 1155
		 * If the checksum failed, see if it is a null block. If so, then
		 * just try with next block...
1156 1157
		 */

B
Bruce Momjian 已提交
1158
		if (chk == sum)
1159
			gotBlock = true;
B
Bruce Momjian 已提交
1160 1161 1162
		else
		{
			for (i = 0; i < 512; i++)
1163
			{
B
Bruce Momjian 已提交
1164
				if (h[0] != 0)
1165
				{
B
Bruce Momjian 已提交
1166
					gotBlock = true;
1167 1168 1169 1170 1171
					break;
				}
			}
		}
	}
P
Philip Warner 已提交
1172 1173 1174 1175 1176

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

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

	if (chk != sum)
1180 1181
		die_horribly(AH, modulename,
					 "corrupt tar header found in %s "
1182
		"(expected %d (%o), computed %d (%o)) file position %ld (%lx)\n",
1183
					 &name[0], sum, sum, chk, chk, ftell(ctx->tarFH), ftell(ctx->tarFH));
P
Philip Warner 已提交
1184 1185 1186 1187 1188 1189 1190

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

	return 1;
}

B
Bruce Momjian 已提交
1191 1192
static void
_tarWriteHeader(TAR_MEMBER *th)
P
Philip Warner 已提交
1193 1194 1195 1196 1197 1198
{
	char		h[512];
	int			i;
	int			lastSum = 0;
	int			sum;

B
Bruce Momjian 已提交
1199
	for (i = 0; i < 512; i++)
P
Philip Warner 已提交
1200 1201 1202 1203 1204 1205 1206 1207 1208
		h[i] = '\0';

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

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

	/* User ID 8 */
P
Philip Warner 已提交
1209
	sprintf(&h[108], " 04000 ");
P
Philip Warner 已提交
1210 1211

	/* Group 8 */
P
Philip Warner 已提交
1212
	sprintf(&h[116], " 02000 ");
P
Philip Warner 已提交
1213 1214

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

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

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

P
Philip Warner 已提交
1223 1224 1225
	/* Type 1 */
	/* sprintf(&h[156], "%c", LF_NORMAL); */
	sprintf(&h[156], "0");
P
Philip Warner 已提交
1226 1227 1228 1229 1230 1231

	/* Link name 100 (NULL) */

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

B
Bruce Momjian 已提交
1232 1233 1234
	/*
	 * GNU Version... sprintf(&h[257], "ustar"); sprintf(&h[263], "00");
	 */
P
Philip Warner 已提交
1235

P
Philip Warner 已提交
1236
	/* User 32 */
B
Bruce Momjian 已提交
1237 1238
	sprintf(&h[265], "%.31s", "");		/* How do I get username reliably?
										 * Do I need to? */
P
Philip Warner 已提交
1239 1240

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

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

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


B
Bruce Momjian 已提交
1251
	while ((sum = _tarChecksum(h)) != lastSum)
P
Philip Warner 已提交
1252
	{
P
Philip Warner 已提交
1253
		sprintf(&h[148], "%6o ", sum);
P
Philip Warner 已提交
1254 1255 1256
		lastSum = sum;
	}

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

P
Philip Warner 已提交
1260
}