xref: /freebsd/usr.bin/gzip/unzstd.c (revision f9349d427431eaa5b3bd7e9e7e87725a138facd1)
1*f9349d42SDag-Erling Smørgrav /*-
2*f9349d42SDag-Erling Smørgrav  * SPDX-License-Identifier: BSD-2-Clause
3*f9349d42SDag-Erling Smørgrav  *
4*f9349d42SDag-Erling Smørgrav  * Copyright (c) 2022 Klara, Inc.
5*f9349d42SDag-Erling Smørgrav  *
6*f9349d42SDag-Erling Smørgrav  * Redistribution and use in source and binary forms, with or without
7*f9349d42SDag-Erling Smørgrav  * modification, are permitted provided that the following conditions
8*f9349d42SDag-Erling Smørgrav  * are met:
9*f9349d42SDag-Erling Smørgrav  * 1. Redistributions of source code must retain the above copyright
10*f9349d42SDag-Erling Smørgrav  *    notice, this list of conditions and the following disclaimer.
11*f9349d42SDag-Erling Smørgrav  * 2. Redistributions in binary form must reproduce the above copyright
12*f9349d42SDag-Erling Smørgrav  *    notice, this list of conditions and the following disclaimer in the
13*f9349d42SDag-Erling Smørgrav  *    documentation and/or other materials provided with the distribution.
14*f9349d42SDag-Erling Smørgrav  *
15*f9349d42SDag-Erling Smørgrav  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
16*f9349d42SDag-Erling Smørgrav  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
17*f9349d42SDag-Erling Smørgrav  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
18*f9349d42SDag-Erling Smørgrav  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
19*f9349d42SDag-Erling Smørgrav  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
20*f9349d42SDag-Erling Smørgrav  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
21*f9349d42SDag-Erling Smørgrav  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
22*f9349d42SDag-Erling Smørgrav  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
23*f9349d42SDag-Erling Smørgrav  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
24*f9349d42SDag-Erling Smørgrav  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
25*f9349d42SDag-Erling Smørgrav  * SUCH DAMAGE.
26*f9349d42SDag-Erling Smørgrav  */
27*f9349d42SDag-Erling Smørgrav 
28*f9349d42SDag-Erling Smørgrav /* This file is #included by gzip.c */
29*f9349d42SDag-Erling Smørgrav 
30*f9349d42SDag-Erling Smørgrav static off_t
unzstd(int in,int out,char * pre,size_t prelen,off_t * bytes_in)31*f9349d42SDag-Erling Smørgrav unzstd(int in, int out, char *pre, size_t prelen, off_t *bytes_in)
32*f9349d42SDag-Erling Smørgrav {
33*f9349d42SDag-Erling Smørgrav 	static char *ibuf, *obuf;
34*f9349d42SDag-Erling Smørgrav 	ZSTD_inBuffer zib;
35*f9349d42SDag-Erling Smørgrav 	ZSTD_outBuffer zob;
36*f9349d42SDag-Erling Smørgrav 	ZSTD_DCtx *zds;
37*f9349d42SDag-Erling Smørgrav 	ssize_t res;
38*f9349d42SDag-Erling Smørgrav 	size_t zres;
39*f9349d42SDag-Erling Smørgrav 	size_t bytes_out = 0;
40*f9349d42SDag-Erling Smørgrav 	int eof = 0;
41*f9349d42SDag-Erling Smørgrav 
42*f9349d42SDag-Erling Smørgrav 	if (ibuf == NULL)
43*f9349d42SDag-Erling Smørgrav 		ibuf = malloc(BUFLEN);
44*f9349d42SDag-Erling Smørgrav 	if (obuf == NULL)
45*f9349d42SDag-Erling Smørgrav 		obuf = malloc(BUFLEN);
46*f9349d42SDag-Erling Smørgrav 	if (ibuf == NULL || obuf == NULL)
47*f9349d42SDag-Erling Smørgrav 		maybe_err("malloc");
48*f9349d42SDag-Erling Smørgrav 
49*f9349d42SDag-Erling Smørgrav 	zds = ZSTD_createDStream();
50*f9349d42SDag-Erling Smørgrav 	ZSTD_initDStream(zds);
51*f9349d42SDag-Erling Smørgrav 
52*f9349d42SDag-Erling Smørgrav 	zib.src = pre;
53*f9349d42SDag-Erling Smørgrav 	zib.size = prelen;
54*f9349d42SDag-Erling Smørgrav 	zib.pos = 0;
55*f9349d42SDag-Erling Smørgrav 	if (bytes_in != NULL)
56*f9349d42SDag-Erling Smørgrav 		*bytes_in = prelen;
57*f9349d42SDag-Erling Smørgrav 	zob.dst = obuf;
58*f9349d42SDag-Erling Smørgrav 	zob.size = BUFLEN;
59*f9349d42SDag-Erling Smørgrav 	zob.pos = 0;
60*f9349d42SDag-Erling Smørgrav 
61*f9349d42SDag-Erling Smørgrav 	while (!eof) {
62*f9349d42SDag-Erling Smørgrav 		if (zib.pos >= zib.size) {
63*f9349d42SDag-Erling Smørgrav 			res = read(in, ibuf, BUFLEN);
64*f9349d42SDag-Erling Smørgrav 			if (res < 0)
65*f9349d42SDag-Erling Smørgrav 				maybe_err("read");
66*f9349d42SDag-Erling Smørgrav 			if (res == 0)
67*f9349d42SDag-Erling Smørgrav 				eof = 1;
68*f9349d42SDag-Erling Smørgrav 			infile_newdata(res);
69*f9349d42SDag-Erling Smørgrav 			zib.src = ibuf;
70*f9349d42SDag-Erling Smørgrav 			zib.size = res;
71*f9349d42SDag-Erling Smørgrav 			zib.pos = 0;
72*f9349d42SDag-Erling Smørgrav 			if (bytes_in != NULL)
73*f9349d42SDag-Erling Smørgrav 				*bytes_in += res;
74*f9349d42SDag-Erling Smørgrav 		}
75*f9349d42SDag-Erling Smørgrav 		zres = ZSTD_decompressStream(zds, &zob, &zib);
76*f9349d42SDag-Erling Smørgrav 		if (ZSTD_isError(zres)) {
77*f9349d42SDag-Erling Smørgrav 			maybe_errx("%s", ZSTD_getErrorName(zres));
78*f9349d42SDag-Erling Smørgrav 		}
79*f9349d42SDag-Erling Smørgrav 		if (zob.pos > 0) {
80*f9349d42SDag-Erling Smørgrav 			res = write(out, obuf, zob.pos);
81*f9349d42SDag-Erling Smørgrav 			if (res < 0)
82*f9349d42SDag-Erling Smørgrav 				maybe_err("write");
83*f9349d42SDag-Erling Smørgrav 			zob.pos = 0;
84*f9349d42SDag-Erling Smørgrav 			bytes_out += res;
85*f9349d42SDag-Erling Smørgrav 		}
86*f9349d42SDag-Erling Smørgrav 	}
87*f9349d42SDag-Erling Smørgrav 	ZSTD_freeDStream(zds);
88*f9349d42SDag-Erling Smørgrav 	return (bytes_out);
89*f9349d42SDag-Erling Smørgrav }
90