163dab8eeSAdrian Chadd /*
263dab8eeSAdrian Chadd * Simple XZ decoder command line tool
363dab8eeSAdrian Chadd *
463dab8eeSAdrian Chadd * Author: Lasse Collin <lasse.collin@tukaani.org>
563dab8eeSAdrian Chadd *
663dab8eeSAdrian Chadd * This file has been put into the public domain.
763dab8eeSAdrian Chadd * You can do whatever you want with this file.
863dab8eeSAdrian Chadd */
963dab8eeSAdrian Chadd
1063dab8eeSAdrian Chadd /*
1163dab8eeSAdrian Chadd * This is really limited: Not all filters from .xz format are supported,
1263dab8eeSAdrian Chadd * only CRC32 is supported as the integrity check, and decoding of
1363dab8eeSAdrian Chadd * concatenated .xz streams is not supported. Thus, you may want to look
1463dab8eeSAdrian Chadd * at xzdec from XZ Utils if a few KiB bigger tool is not a problem.
1563dab8eeSAdrian Chadd */
1663dab8eeSAdrian Chadd
1763dab8eeSAdrian Chadd #include <stdbool.h>
1863dab8eeSAdrian Chadd #include <stdio.h>
1963dab8eeSAdrian Chadd #include <string.h>
2063dab8eeSAdrian Chadd #include "xz.h"
2163dab8eeSAdrian Chadd
2263dab8eeSAdrian Chadd static uint8_t in[BUFSIZ];
2363dab8eeSAdrian Chadd static uint8_t out[BUFSIZ];
2463dab8eeSAdrian Chadd
main(int argc,char ** argv)2563dab8eeSAdrian Chadd int main(int argc, char **argv)
2663dab8eeSAdrian Chadd {
2763dab8eeSAdrian Chadd struct xz_buf b;
2863dab8eeSAdrian Chadd struct xz_dec *s;
2963dab8eeSAdrian Chadd enum xz_ret ret;
3063dab8eeSAdrian Chadd const char *msg;
3163dab8eeSAdrian Chadd
3263dab8eeSAdrian Chadd if (argc >= 2 && strcmp(argv[1], "--help") == 0) {
3363dab8eeSAdrian Chadd fputs("Uncompress a .xz file from stdin to stdout.\n"
3463dab8eeSAdrian Chadd "Arguments other than `--help' are ignored.\n",
3563dab8eeSAdrian Chadd stdout);
3663dab8eeSAdrian Chadd return 0;
3763dab8eeSAdrian Chadd }
3863dab8eeSAdrian Chadd
3963dab8eeSAdrian Chadd xz_crc32_init();
40f0bd5302SXin LI #ifdef XZ_USE_CRC64
41f0bd5302SXin LI xz_crc64_init();
42f0bd5302SXin LI #endif
4363dab8eeSAdrian Chadd
4463dab8eeSAdrian Chadd /*
4563dab8eeSAdrian Chadd * Support up to 64 MiB dictionary. The actually needed memory
4663dab8eeSAdrian Chadd * is allocated once the headers have been parsed.
4763dab8eeSAdrian Chadd */
4863dab8eeSAdrian Chadd s = xz_dec_init(XZ_DYNALLOC, 1 << 26);
4963dab8eeSAdrian Chadd if (s == NULL) {
5063dab8eeSAdrian Chadd msg = "Memory allocation failed\n";
5163dab8eeSAdrian Chadd goto error;
5263dab8eeSAdrian Chadd }
5363dab8eeSAdrian Chadd
5463dab8eeSAdrian Chadd b.in = in;
5563dab8eeSAdrian Chadd b.in_pos = 0;
5663dab8eeSAdrian Chadd b.in_size = 0;
5763dab8eeSAdrian Chadd b.out = out;
5863dab8eeSAdrian Chadd b.out_pos = 0;
5963dab8eeSAdrian Chadd b.out_size = BUFSIZ;
6063dab8eeSAdrian Chadd
6163dab8eeSAdrian Chadd while (true) {
6263dab8eeSAdrian Chadd if (b.in_pos == b.in_size) {
6363dab8eeSAdrian Chadd b.in_size = fread(in, 1, sizeof(in), stdin);
64*cd3a777bSXin LI
65*cd3a777bSXin LI if (ferror(stdin)) {
66*cd3a777bSXin LI msg = "Read error\n";
67*cd3a777bSXin LI goto error;
68*cd3a777bSXin LI }
69*cd3a777bSXin LI
7063dab8eeSAdrian Chadd b.in_pos = 0;
7163dab8eeSAdrian Chadd }
7263dab8eeSAdrian Chadd
73*cd3a777bSXin LI /*
74*cd3a777bSXin LI * There are a few ways to set the "finish" (the third)
75*cd3a777bSXin LI * argument. We could use feof(stdin) but testing in_size
76*cd3a777bSXin LI * is fine too and may also work in applications that don't
77*cd3a777bSXin LI * use FILEs.
78*cd3a777bSXin LI */
79*cd3a777bSXin LI ret = xz_dec_catrun(s, &b, b.in_size == 0);
8063dab8eeSAdrian Chadd
8163dab8eeSAdrian Chadd if (b.out_pos == sizeof(out)) {
8263dab8eeSAdrian Chadd if (fwrite(out, 1, b.out_pos, stdout) != b.out_pos) {
8363dab8eeSAdrian Chadd msg = "Write error\n";
8463dab8eeSAdrian Chadd goto error;
8563dab8eeSAdrian Chadd }
8663dab8eeSAdrian Chadd
8763dab8eeSAdrian Chadd b.out_pos = 0;
8863dab8eeSAdrian Chadd }
8963dab8eeSAdrian Chadd
9063dab8eeSAdrian Chadd if (ret == XZ_OK)
9163dab8eeSAdrian Chadd continue;
9263dab8eeSAdrian Chadd
9363dab8eeSAdrian Chadd #ifdef XZ_DEC_ANY_CHECK
9463dab8eeSAdrian Chadd if (ret == XZ_UNSUPPORTED_CHECK) {
9563dab8eeSAdrian Chadd fputs(argv[0], stderr);
9663dab8eeSAdrian Chadd fputs(": ", stderr);
9763dab8eeSAdrian Chadd fputs("Unsupported check; not verifying "
9863dab8eeSAdrian Chadd "file integrity\n", stderr);
9963dab8eeSAdrian Chadd continue;
10063dab8eeSAdrian Chadd }
10163dab8eeSAdrian Chadd #endif
10263dab8eeSAdrian Chadd
10363dab8eeSAdrian Chadd if (fwrite(out, 1, b.out_pos, stdout) != b.out_pos
10463dab8eeSAdrian Chadd || fclose(stdout)) {
10563dab8eeSAdrian Chadd msg = "Write error\n";
10663dab8eeSAdrian Chadd goto error;
10763dab8eeSAdrian Chadd }
10863dab8eeSAdrian Chadd
10963dab8eeSAdrian Chadd switch (ret) {
11063dab8eeSAdrian Chadd case XZ_STREAM_END:
11163dab8eeSAdrian Chadd xz_dec_end(s);
11263dab8eeSAdrian Chadd return 0;
11363dab8eeSAdrian Chadd
11463dab8eeSAdrian Chadd case XZ_MEM_ERROR:
11563dab8eeSAdrian Chadd msg = "Memory allocation failed\n";
11663dab8eeSAdrian Chadd goto error;
11763dab8eeSAdrian Chadd
11863dab8eeSAdrian Chadd case XZ_MEMLIMIT_ERROR:
11963dab8eeSAdrian Chadd msg = "Memory usage limit reached\n";
12063dab8eeSAdrian Chadd goto error;
12163dab8eeSAdrian Chadd
12263dab8eeSAdrian Chadd case XZ_FORMAT_ERROR:
12363dab8eeSAdrian Chadd msg = "Not a .xz file\n";
12463dab8eeSAdrian Chadd goto error;
12563dab8eeSAdrian Chadd
12663dab8eeSAdrian Chadd case XZ_OPTIONS_ERROR:
12763dab8eeSAdrian Chadd msg = "Unsupported options in the .xz headers\n";
12863dab8eeSAdrian Chadd goto error;
12963dab8eeSAdrian Chadd
13063dab8eeSAdrian Chadd case XZ_DATA_ERROR:
13163dab8eeSAdrian Chadd case XZ_BUF_ERROR:
13263dab8eeSAdrian Chadd msg = "File is corrupt\n";
13363dab8eeSAdrian Chadd goto error;
13463dab8eeSAdrian Chadd
13563dab8eeSAdrian Chadd default:
13663dab8eeSAdrian Chadd msg = "Bug!\n";
13763dab8eeSAdrian Chadd goto error;
13863dab8eeSAdrian Chadd }
13963dab8eeSAdrian Chadd }
14063dab8eeSAdrian Chadd
14163dab8eeSAdrian Chadd error:
14263dab8eeSAdrian Chadd xz_dec_end(s);
14363dab8eeSAdrian Chadd fputs(argv[0], stderr);
14463dab8eeSAdrian Chadd fputs(": ", stderr);
14563dab8eeSAdrian Chadd fputs(msg, stderr);
14663dab8eeSAdrian Chadd return 1;
14763dab8eeSAdrian Chadd }
148