2
0
mirror of https://github.com/vasi/pixz synced 2024-11-03 09:40:24 +00:00
pixz/pixz.c
2012-10-14 07:33:33 -04:00

163 lines
4.5 KiB
C

#include "pixz.h"
#include <unistd.h>
#include <getopt.h>
typedef enum {
OP_WRITE,
OP_READ,
OP_EXTRACT,
OP_LIST
} pixz_op_t;
static bool strsuf(char *big, char *small);
static char *subsuf(char *in, char *suf1, char *suf2);
static char *auto_output(pixz_op_t op, char *in);
static void usage(const char *msg) {
if (msg)
fprintf(stderr, "%s\n\n", msg);
fprintf(stderr,
"pixz: Parallel Indexing XZ compression, fully compatible with XZ\n"
"\n"
"Basic usage:\n"
" pixz input output.pxz # Compress a file in parallel\n"
" pixz -d input.pxz output # Decompress\n"
"\n"
"Tarballs:\n"
" pixz input.tar output.tpxz # Compress and index a tarball\n"
" pixz -d input.tpxz output.tar # Decompress\n"
" pixz -l input.tpxz # List tarball contents very fast\n"
" pixz -x path/to/file < input.tpxz | tar x # Extract one file very fast\n"
" tar -Ipixz -cf output.tpxz dir # Make tar use pixz automatically\n"
"\n"
"Input and output:\n"
" pixz < input > output.pxz # Same as `pixz input output.pxz`\n"
" pixz -i input -o output.pxz # Ditto\n"
" pixz [-d] input # Automatically choose output filename\n"
"\n"
"Other flags:\n"
" -0, -1 ... -9 Set compression level, from fastest to strongest\n"
" -p NUM Use a maximum of NUM CPU-intensive threads\n"
" -t Don't assume input is in tar format\n"
" -h Print this help\n"
"\n"
"(C) 2009-2012 Dave Vasilevsky <dave@vasilevsky.ca>\n"
"https://github.com/vasi/pixz\n"
"You may use this software under the FreeBSD License\n"
);
exit(2);
}
int main(int argc, char **argv) {
uint32_t level = LZMA_PRESET_DEFAULT;
bool tar = true;
pixz_op_t op = OP_WRITE;
char *ipath = NULL, *opath = NULL;
int ch;
char *optend;
long optint;
while ((ch = getopt(argc, argv, "dxli:o:tvhp:0123456789")) != -1) {
switch (ch) {
case 'd': op = OP_READ; break;
case 'x': op = OP_EXTRACT; break;
case 'l': op = OP_LIST; break;
case 'i': ipath = optarg; break;
case 'o': opath = optarg; break;
case 't': tar = false; break;
case 'h': usage(NULL); break;
case 'p':
optint = strtol(optarg, &optend, 10);
if (optint < 0 || *optend)
usage("Need a non-negative integer argument to -p");
gPipelineProcessMax = optint;
break;
default:
if (ch >= '0' && ch <= '9') {
level = ch - '0';
} else {
usage("");
}
}
}
argc -= optind;
argv += optind;
gInFile = stdin;
gOutFile = stdout;
bool iremove = false;
if (op != OP_EXTRACT && argc >= 1) {
if (argc > 2 || (op == OP_LIST && argc == 2))
usage("Too many arguments");
if (ipath)
usage("Multiple input files specified");
ipath = argv[0];
if (argc == 2) {
if (opath)
usage("Multiple output files specified");
opath = argv[1];
} else if (op != OP_LIST) {
iremove = true;
opath = auto_output(op, argv[0]);
if (!opath)
usage("Unknown suffix");
}
}
if (ipath && !(gInFile = fopen(ipath, "r")))
die("Can't open input file");
if (opath && !(gOutFile = fopen(opath, "w")))
die("Can't open output file");
switch (op) {
case OP_WRITE:
if (isatty(fileno(gOutFile)) == -1)
usage("Refusing to output to a TTY");
pixz_write(tar, level);
break;
case OP_READ: pixz_read(tar, 0, NULL); break;
case OP_EXTRACT: pixz_read(tar, argc, argv); break;
case OP_LIST: pixz_list(tar);
}
if (iremove)
unlink(ipath);
return 0;
}
#define SUF(_op, _s1, _s2) ({ \
if (op == OP_##_op) { \
char *r = subsuf(in, _s1, _s2); \
if (r) \
return r; \
} \
})
static char *auto_output(pixz_op_t op, char *in) {
SUF(READ, ".tar.xz", ".tar");
SUF(READ, ".tpxz", ".tar");
SUF(READ, ".xz", "");
SUF(WRITE, ".tar", ".tpxz");
SUF(WRITE, "", ".xz");
return NULL;
}
static bool strsuf(char *big, char *small) {
size_t bl = strlen(big), sl = strlen(small);
return strcmp(big + bl - sl, small) == 0;
}
static char *subsuf(char *in, char *suf1, char *suf2) {
if (!strsuf(in, suf1))
return NULL;
size_t li = strlen(in), l1 = strlen(suf1), l2 = strlen(suf2);
char *r = malloc(li + l2 - l1 + 1);
memcpy(r, in, li - l1);
strcpy(r + li - l1, suf2);
return r;
}