Commit 05db8b50 authored by Neil Conway's avatar Neil Conway

Correct some code in pg_restore when reading the header of a tar archive:

(1) The code doesn't initialize `sum', so the initial "does the checksum
    match?" test is wrong.

(2) The loop that is intended to check for a "null block" just checks
    the first byte of the tar block 512 times, rather than each of the
    512 bytes one time (!), which I'm guessing was the intent.

It was only through sheer luck that this worked in the first place.

Per Coverity static analysis performed by EnterpriseDB.
parent 06ecacde
...@@ -16,7 +16,7 @@ ...@@ -16,7 +16,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/bin/pg_dump/pg_backup_tar.c,v 1.47 2005/01/25 22:44:31 tgl Exp $ * $PostgreSQL: pgsql/src/bin/pg_dump/pg_backup_tar.c,v 1.48 2005/06/22 02:00:47 neilc Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -1155,7 +1155,6 @@ _tarGetHeader(ArchiveHandle *AH, TAR_MEMBER *th) ...@@ -1155,7 +1155,6 @@ _tarGetHeader(ArchiveHandle *AH, TAR_MEMBER *th)
size_t len; size_t len;
unsigned long ullen; unsigned long ullen;
off_t hPos; off_t hPos;
int i;
bool gotBlock = false; bool gotBlock = false;
while (!gotBlock) while (!gotBlock)
...@@ -1178,7 +1177,7 @@ _tarGetHeader(ArchiveHandle *AH, TAR_MEMBER *th) ...@@ -1178,7 +1177,7 @@ _tarGetHeader(ArchiveHandle *AH, TAR_MEMBER *th)
hPos = ctx->tarFHpos; hPos = ctx->tarFHpos;
/* Read a 512 byte block, return EOF, exit if short */ /* Read a 512 byte block, return EOF, exit if short */
len = _tarReadRaw(AH, &h[0], 512, NULL, ctx->tarFH); len = _tarReadRaw(AH, h, 512, NULL, ctx->tarFH);
if (len == 0) /* EOF */ if (len == 0) /* EOF */
return 0; return 0;
...@@ -1188,20 +1187,22 @@ _tarGetHeader(ArchiveHandle *AH, TAR_MEMBER *th) ...@@ -1188,20 +1187,22 @@ _tarGetHeader(ArchiveHandle *AH, TAR_MEMBER *th)
(unsigned long) len); (unsigned long) len);
/* Calc checksum */ /* Calc checksum */
chk = _tarChecksum(&h[0]); chk = _tarChecksum(h);
sscanf(&h[148], "%8o", &sum);
/* /*
* If the checksum failed, see if it is a null block. If so, then * If the checksum failed, see if it is a null block. If so,
* just try with next block... * silently continue to the next block.
*/ */
if (chk == sum) if (chk == sum)
gotBlock = true; gotBlock = true;
else else
{ {
int i;
for (i = 0; i < 512; i++) for (i = 0; i < 512; i++)
{ {
if (h[0] != 0) if (h[i] != 0)
{ {
gotBlock = true; gotBlock = true;
break; break;
...@@ -1213,7 +1214,6 @@ _tarGetHeader(ArchiveHandle *AH, TAR_MEMBER *th) ...@@ -1213,7 +1214,6 @@ _tarGetHeader(ArchiveHandle *AH, TAR_MEMBER *th)
sscanf(&h[0], "%99s", tag); sscanf(&h[0], "%99s", tag);
sscanf(&h[124], "%12lo", &ullen); sscanf(&h[124], "%12lo", &ullen);
len = (size_t) ullen; len = (size_t) ullen;
sscanf(&h[148], "%8o", &sum);
{ {
char buf[100]; char buf[100];
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment