xref: /freebsd/sys/contrib/xz-embedded/userspace/xzminidec.c (revision cd3a777bca91669fc4711d1eff66c40f3f62a223)
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