Blame contrib/untar.c

Packit Service 1d0348
/*
Packit Service 1d0348
 * This file is in the public domain.  Use it as you see fit.
Packit Service 1d0348
 */
Packit Service 1d0348
Packit Service 1d0348
/*
Packit Service 1d0348
 * "untar" is an extremely simple tar extractor:
Packit Service 1d0348
 *  * A single C source file, so it should be easy to compile
Packit Service 1d0348
 *    and run on any system with a C compiler.
Packit Service 1d0348
 *  * Extremely portable standard C.  The only non-ANSI function
Packit Service 1d0348
 *    used is mkdir().
Packit Service 1d0348
 *  * Reads basic ustar tar archives.
Packit Service 1d0348
 *  * Does not require libarchive or any other special library.
Packit Service 1d0348
 *
Packit Service 1d0348
 * To compile: cc -o untar untar.c
Packit Service 1d0348
 *
Packit Service 1d0348
 * Usage:  untar <archive>
Packit Service 1d0348
 *
Packit Service 1d0348
 * In particular, this program should be sufficient to extract the
Packit Service 1d0348
 * distribution for libarchive, allowing people to bootstrap
Packit Service 1d0348
 * libarchive on systems that do not already have a tar program.
Packit Service 1d0348
 *
Packit Service 1d0348
 * To unpack libarchive-x.y.z.tar.gz:
Packit Service 1d0348
 *    * gunzip libarchive-x.y.z.tar.gz
Packit Service 1d0348
 *    * untar libarchive-x.y.z.tar
Packit Service 1d0348
 *
Packit Service 1d0348
 * Written by Tim Kientzle, March 2009.
Packit Service 1d0348
 *
Packit Service 1d0348
 * Released into the public domain.
Packit Service 1d0348
 */
Packit Service 1d0348
Packit Service 1d0348
/* These are all highly standard and portable headers. */
Packit Service 1d0348
#include <stdio.h>
Packit Service 1d0348
#include <stdlib.h>
Packit Service 1d0348
#include <string.h>
Packit Service 1d0348
Packit Service 1d0348
/* This is for mkdir(); this may need to be changed for some platforms. */
Packit Service 1d0348
#include <sys/stat.h>  /* For mkdir() */
Packit Service 1d0348
Packit Service 1d0348
/* Parse an octal number, ignoring leading and trailing nonsense. */
Packit Service 1d0348
static int
Packit Service 1d0348
parseoct(const char *p, size_t n)
Packit Service 1d0348
{
Packit Service 1d0348
	int i = 0;
Packit Service 1d0348
Packit Service 1d0348
	while ((*p < '0' || *p > '7') && n > 0) {
Packit Service 1d0348
		++p;
Packit Service 1d0348
		--n;
Packit Service 1d0348
	}
Packit Service 1d0348
	while (*p >= '0' && *p <= '7' && n > 0) {
Packit Service 1d0348
		i *= 8;
Packit Service 1d0348
		i += *p - '0';
Packit Service 1d0348
		++p;
Packit Service 1d0348
		--n;
Packit Service 1d0348
	}
Packit Service 1d0348
	return (i);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
/* Returns true if this is 512 zero bytes. */
Packit Service 1d0348
static int
Packit Service 1d0348
is_end_of_archive(const char *p)
Packit Service 1d0348
{
Packit Service 1d0348
	int n;
Packit Service 1d0348
	for (n = 511; n >= 0; --n)
Packit Service 1d0348
		if (p[n] != '\0')
Packit Service 1d0348
			return (0);
Packit Service 1d0348
	return (1);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
/* Create a directory, including parent directories as necessary. */
Packit Service 1d0348
static void
Packit Service 1d0348
create_dir(char *pathname, int mode)
Packit Service 1d0348
{
Packit Service 1d0348
	char *p;
Packit Service 1d0348
	int r;
Packit Service 1d0348
Packit Service 1d0348
	/* Strip trailing '/' */
Packit Service 1d0348
	if (pathname[strlen(pathname) - 1] == '/')
Packit Service 1d0348
		pathname[strlen(pathname) - 1] = '\0';
Packit Service 1d0348
Packit Service 1d0348
	/* Try creating the directory. */
Packit Service 1d0348
	r = mkdir(pathname, mode);
Packit Service 1d0348
Packit Service 1d0348
	if (r != 0) {
Packit Service 1d0348
		/* On failure, try creating parent directory. */
Packit Service 1d0348
		p = strrchr(pathname, '/');
Packit Service 1d0348
		if (p != NULL) {
Packit Service 1d0348
			*p = '\0';
Packit Service 1d0348
			create_dir(pathname, 0755);
Packit Service 1d0348
			*p = '/';
Packit Service 1d0348
			r = mkdir(pathname, mode);
Packit Service 1d0348
		}
Packit Service 1d0348
	}
Packit Service 1d0348
	if (r != 0)
Packit Service 1d0348
		fprintf(stderr, "Could not create directory %s\n", pathname);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
/* Create a file, including parent directory as necessary. */
Packit Service 1d0348
static FILE *
Packit Service 1d0348
create_file(char *pathname, int mode)
Packit Service 1d0348
{
Packit Service 1d0348
	FILE *f;
Packit Service 1d0348
	f = fopen(pathname, "wb+");
Packit Service 1d0348
	if (f == NULL) {
Packit Service 1d0348
		/* Try creating parent dir and then creating file. */
Packit Service 1d0348
		char *p = strrchr(pathname, '/');
Packit Service 1d0348
		if (p != NULL) {
Packit Service 1d0348
			*p = '\0';
Packit Service 1d0348
			create_dir(pathname, 0755);
Packit Service 1d0348
			*p = '/';
Packit Service 1d0348
			f = fopen(pathname, "wb+");
Packit Service 1d0348
		}
Packit Service 1d0348
	}
Packit Service 1d0348
	return (f);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
/* Verify the tar checksum. */
Packit Service 1d0348
static int
Packit Service 1d0348
verify_checksum(const char *p)
Packit Service 1d0348
{
Packit Service 1d0348
	int n, u = 0;
Packit Service 1d0348
	for (n = 0; n < 512; ++n) {
Packit Service 1d0348
		if (n < 148 || n > 155)
Packit Service 1d0348
			/* Standard tar checksum adds unsigned bytes. */
Packit Service 1d0348
			u += ((unsigned char *)p)[n];
Packit Service 1d0348
		else
Packit Service 1d0348
			u += 0x20;
Packit Service 1d0348
Packit Service 1d0348
	}
Packit Service 1d0348
	return (u == parseoct(p + 148, 8));
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
/* Extract a tar archive. */
Packit Service 1d0348
static void
Packit Service 1d0348
untar(FILE *a, const char *path)
Packit Service 1d0348
{
Packit Service 1d0348
	char buff[512];
Packit Service 1d0348
	FILE *f = NULL;
Packit Service 1d0348
	size_t bytes_read;
Packit Service 1d0348
	int filesize;
Packit Service 1d0348
Packit Service 1d0348
	printf("Extracting from %s\n", path);
Packit Service 1d0348
	for (;;) {
Packit Service 1d0348
		bytes_read = fread(buff, 1, 512, a);
Packit Service 1d0348
		if (bytes_read < 512) {
Packit Service 1d0348
			fprintf(stderr,
Packit Service 1d0348
			    "Short read on %s: expected 512, got %d\n",
Packit Service 1d0348
			    path, (int)bytes_read);
Packit Service 1d0348
			return;
Packit Service 1d0348
		}
Packit Service 1d0348
		if (is_end_of_archive(buff)) {
Packit Service 1d0348
			printf("End of %s\n", path);
Packit Service 1d0348
			return;
Packit Service 1d0348
		}
Packit Service 1d0348
		if (!verify_checksum(buff)) {
Packit Service 1d0348
			fprintf(stderr, "Checksum failure\n");
Packit Service 1d0348
			return;
Packit Service 1d0348
		}
Packit Service 1d0348
		filesize = parseoct(buff + 124, 12);
Packit Service 1d0348
		switch (buff[156]) {
Packit Service 1d0348
		case '1':
Packit Service 1d0348
			printf(" Ignoring hardlink %s\n", buff);
Packit Service 1d0348
			break;
Packit Service 1d0348
		case '2':
Packit Service 1d0348
			printf(" Ignoring symlink %s\n", buff);
Packit Service 1d0348
			break;
Packit Service 1d0348
		case '3':
Packit Service 1d0348
			printf(" Ignoring character device %s\n", buff);
Packit Service 1d0348
				break;
Packit Service 1d0348
		case '4':
Packit Service 1d0348
			printf(" Ignoring block device %s\n", buff);
Packit Service 1d0348
			break;
Packit Service 1d0348
		case '5':
Packit Service 1d0348
			printf(" Extracting dir %s\n", buff);
Packit Service 1d0348
			create_dir(buff, parseoct(buff + 100, 8));
Packit Service 1d0348
			filesize = 0;
Packit Service 1d0348
			break;
Packit Service 1d0348
		case '6':
Packit Service 1d0348
			printf(" Ignoring FIFO %s\n", buff);
Packit Service 1d0348
			break;
Packit Service 1d0348
		default:
Packit Service 1d0348
			printf(" Extracting file %s\n", buff);
Packit Service 1d0348
			f = create_file(buff, parseoct(buff + 100, 8));
Packit Service 1d0348
			break;
Packit Service 1d0348
		}
Packit Service 1d0348
		while (filesize > 0) {
Packit Service 1d0348
			bytes_read = fread(buff, 1, 512, a);
Packit Service 1d0348
			if (bytes_read < 512) {
Packit Service 1d0348
				fprintf(stderr,
Packit Service 1d0348
				    "Short read on %s: Expected 512, got %d\n",
Packit Service 1d0348
				    path, (int)bytes_read);
Packit Service 1d0348
				return;
Packit Service 1d0348
			}
Packit Service 1d0348
			if (filesize < 512)
Packit Service 1d0348
				bytes_read = filesize;
Packit Service 1d0348
			if (f != NULL) {
Packit Service 1d0348
				if (fwrite(buff, 1, bytes_read, f)
Packit Service 1d0348
				    != bytes_read)
Packit Service 1d0348
				{
Packit Service 1d0348
					fprintf(stderr, "Failed write\n");
Packit Service 1d0348
					fclose(f);
Packit Service 1d0348
					f = NULL;
Packit Service 1d0348
				}
Packit Service 1d0348
			}
Packit Service 1d0348
			filesize -= bytes_read;
Packit Service 1d0348
		}
Packit Service 1d0348
		if (f != NULL) {
Packit Service 1d0348
			fclose(f);
Packit Service 1d0348
			f = NULL;
Packit Service 1d0348
		}
Packit Service 1d0348
	}
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
int
Packit Service 1d0348
main(int argc, char **argv)
Packit Service 1d0348
{
Packit Service 1d0348
	FILE *a;
Packit Service 1d0348
Packit Service 1d0348
	++argv; /* Skip program name */
Packit Service 1d0348
	for ( ;*argv != NULL; ++argv) {
Packit Service 1d0348
		a = fopen(*argv, "rb");
Packit Service 1d0348
		if (a == NULL)
Packit Service 1d0348
			fprintf(stderr, "Unable to open %s\n", *argv);
Packit Service 1d0348
		else {
Packit Service 1d0348
			untar(a, *argv);
Packit Service 1d0348
			fclose(a);
Packit Service 1d0348
		}
Packit Service 1d0348
	}
Packit Service 1d0348
	return (0);
Packit Service 1d0348
}