
Similar to commit 7e735035f2, this commit makes the order of header file inclusion consistent for non-backend modules. In passing, fix the case where we were using angle brackets (<>) for the local module includes instead of quotes (""). Author: Vignesh C Reviewed-by: Amit Kapila Discussion: https://postgr.es/m/CALDaNm2Sznv8RR6Ex-iJO6xAdsxgWhCoETkaYX=+9DW3q0QCfA@mail.gmail.com
207 lines
4.6 KiB
C
207 lines
4.6 KiB
C
#include "c.h"
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include "pgtar.h"
|
|
|
|
/*
|
|
* Print a numeric field in a tar header. The field starts at *s and is of
|
|
* length len; val is the value to be written.
|
|
*
|
|
* Per POSIX, the way to write a number is in octal with leading zeroes and
|
|
* one trailing space (or NUL, but we use space) at the end of the specified
|
|
* field width.
|
|
*
|
|
* However, the given value may not fit in the available space in octal form.
|
|
* If that's true, we use the GNU extension of writing \200 followed by the
|
|
* number in base-256 form (ie, stored in binary MSB-first). (Note: here we
|
|
* support only non-negative numbers, so we don't worry about the GNU rules
|
|
* for handling negative numbers.)
|
|
*/
|
|
void
|
|
print_tar_number(char *s, int len, uint64 val)
|
|
{
|
|
if (val < (((uint64) 1) << ((len - 1) * 3)))
|
|
{
|
|
/* Use octal with trailing space */
|
|
s[--len] = ' ';
|
|
while (len)
|
|
{
|
|
s[--len] = (val & 7) + '0';
|
|
val >>= 3;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* Use base-256 with leading \200 */
|
|
s[0] = '\200';
|
|
while (len > 1)
|
|
{
|
|
s[--len] = (val & 255);
|
|
val >>= 8;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* Read a numeric field in a tar header. The field starts at *s and is of
|
|
* length len.
|
|
*
|
|
* The POSIX-approved format for a number is octal, ending with a space or
|
|
* NUL. However, for values that don't fit, we recognize the GNU extension
|
|
* of \200 followed by the number in base-256 form (ie, stored in binary
|
|
* MSB-first). (Note: here we support only non-negative numbers, so we don't
|
|
* worry about the GNU rules for handling negative numbers.)
|
|
*/
|
|
uint64
|
|
read_tar_number(const char *s, int len)
|
|
{
|
|
uint64 result = 0;
|
|
|
|
if (*s == '\200')
|
|
{
|
|
/* base-256 */
|
|
while (--len)
|
|
{
|
|
result <<= 8;
|
|
result |= (unsigned char) (*++s);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* octal */
|
|
while (len-- && *s >= '0' && *s <= '7')
|
|
{
|
|
result <<= 3;
|
|
result |= (*s - '0');
|
|
s++;
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
|
|
/*
|
|
* Calculate the tar checksum for a header. The header is assumed to always
|
|
* be 512 bytes, per the tar standard.
|
|
*/
|
|
int
|
|
tarChecksum(char *header)
|
|
{
|
|
int i,
|
|
sum;
|
|
|
|
/*
|
|
* Per POSIX, the checksum is the simple sum of all bytes in the header,
|
|
* treating the bytes as unsigned, and treating the checksum field (at
|
|
* offset 148) as though it contained 8 spaces.
|
|
*/
|
|
sum = 8 * ' '; /* presumed value for checksum field */
|
|
for (i = 0; i < 512; i++)
|
|
if (i < 148 || i >= 156)
|
|
sum += 0xFF & header[i];
|
|
return sum;
|
|
}
|
|
|
|
|
|
/*
|
|
* Fill in the buffer pointed to by h with a tar format header. This buffer
|
|
* must always have space for 512 characters, which is a requirement of
|
|
* the tar format.
|
|
*/
|
|
enum tarError
|
|
tarCreateHeader(char *h, const char *filename, const char *linktarget,
|
|
pgoff_t size, mode_t mode, uid_t uid, gid_t gid, time_t mtime)
|
|
{
|
|
if (strlen(filename) > 99)
|
|
return TAR_NAME_TOO_LONG;
|
|
|
|
if (linktarget && strlen(linktarget) > 99)
|
|
return TAR_SYMLINK_TOO_LONG;
|
|
|
|
memset(h, 0, 512); /* assume tar header size */
|
|
|
|
/* Name 100 */
|
|
strlcpy(&h[0], filename, 100);
|
|
if (linktarget != NULL || S_ISDIR(mode))
|
|
{
|
|
/*
|
|
* We only support symbolic links to directories, and this is
|
|
* indicated in the tar format by adding a slash at the end of the
|
|
* name, the same as for regular directories.
|
|
*/
|
|
int flen = strlen(filename);
|
|
|
|
flen = Min(flen, 99);
|
|
h[flen] = '/';
|
|
h[flen + 1] = '\0';
|
|
}
|
|
|
|
/* Mode 8 - this doesn't include the file type bits (S_IFMT) */
|
|
print_tar_number(&h[100], 8, (mode & 07777));
|
|
|
|
/* User ID 8 */
|
|
print_tar_number(&h[108], 8, uid);
|
|
|
|
/* Group 8 */
|
|
print_tar_number(&h[116], 8, gid);
|
|
|
|
/* File size 12 */
|
|
if (linktarget != NULL || S_ISDIR(mode))
|
|
/* Symbolic link or directory has size zero */
|
|
print_tar_number(&h[124], 12, 0);
|
|
else
|
|
print_tar_number(&h[124], 12, size);
|
|
|
|
/* Mod Time 12 */
|
|
print_tar_number(&h[136], 12, mtime);
|
|
|
|
/* Checksum 8 cannot be calculated until we've filled all other fields */
|
|
|
|
if (linktarget != NULL)
|
|
{
|
|
/* Type - Symbolic link */
|
|
h[156] = '2';
|
|
/* Link Name 100 */
|
|
strlcpy(&h[157], linktarget, 100);
|
|
}
|
|
else if (S_ISDIR(mode))
|
|
{
|
|
/* Type - directory */
|
|
h[156] = '5';
|
|
}
|
|
else
|
|
{
|
|
/* Type - regular file */
|
|
h[156] = '0';
|
|
}
|
|
|
|
/* Magic 6 */
|
|
strcpy(&h[257], "ustar");
|
|
|
|
/* Version 2 */
|
|
memcpy(&h[263], "00", 2);
|
|
|
|
/* User 32 */
|
|
/* XXX: Do we need to care about setting correct username? */
|
|
strlcpy(&h[265], "postgres", 32);
|
|
|
|
/* Group 32 */
|
|
/* XXX: Do we need to care about setting correct group name? */
|
|
strlcpy(&h[297], "postgres", 32);
|
|
|
|
/* Major Dev 8 */
|
|
print_tar_number(&h[329], 8, 0);
|
|
|
|
/* Minor Dev 8 */
|
|
print_tar_number(&h[337], 8, 0);
|
|
|
|
/* Prefix 155 - not used, leave as nulls */
|
|
|
|
/* Finally, compute and insert the checksum */
|
|
print_tar_number(&h[148], 8, tarChecksum(h));
|
|
|
|
return TAR_OK;
|
|
}
|