10c10c611SXin LI /* $NetBSD: unbzip2.c,v 1.12 2009/10/11 05:17:20 mrg Exp $ */ 29a9ea25fSXin LI 39a9ea25fSXin LI /*- 49a9ea25fSXin LI * Copyright (c) 2006 The NetBSD Foundation, Inc. 59a9ea25fSXin LI * All rights reserved. 69a9ea25fSXin LI * 79a9ea25fSXin LI * This code is derived from software contributed to The NetBSD Foundation 89a9ea25fSXin LI * by Simon Burge. 99a9ea25fSXin LI * 109a9ea25fSXin LI * Redistribution and use in source and binary forms, with or without 119a9ea25fSXin LI * modification, are permitted provided that the following conditions 129a9ea25fSXin LI * are met: 139a9ea25fSXin LI * 1. Redistributions of source code must retain the above copyright 149a9ea25fSXin LI * notice, this list of conditions and the following disclaimer. 159a9ea25fSXin LI * 2. Redistributions in binary form must reproduce the above copyright 169a9ea25fSXin LI * notice, this list of conditions and the following disclaimer in the 179a9ea25fSXin LI * documentation and/or other materials provided with the distribution. 189a9ea25fSXin LI * 199a9ea25fSXin LI * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS 209a9ea25fSXin LI * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED 219a9ea25fSXin LI * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR 229a9ea25fSXin LI * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS 239a9ea25fSXin LI * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 249a9ea25fSXin LI * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 259a9ea25fSXin LI * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 269a9ea25fSXin LI * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 279a9ea25fSXin LI * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 289a9ea25fSXin LI * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 299a9ea25fSXin LI * POSSIBILITY OF SUCH DAMAGE. 309a9ea25fSXin LI * 319a9ea25fSXin LI * $FreeBSD$ 329a9ea25fSXin LI */ 339a9ea25fSXin LI 349a9ea25fSXin LI /* This file is #included by gzip.c */ 359a9ea25fSXin LI 369a9ea25fSXin LI static off_t 379a9ea25fSXin LI unbzip2(int in, int out, char *pre, size_t prelen, off_t *bytes_in) 389a9ea25fSXin LI { 39a0b4046bSXin LI int ret, end_of_file, cold = 0; 409a9ea25fSXin LI off_t bytes_out = 0; 419a9ea25fSXin LI bz_stream bzs; 429a9ea25fSXin LI static char *inbuf, *outbuf; 439a9ea25fSXin LI 449a9ea25fSXin LI if (inbuf == NULL) 459a9ea25fSXin LI inbuf = malloc(BUFLEN); 469a9ea25fSXin LI if (outbuf == NULL) 479a9ea25fSXin LI outbuf = malloc(BUFLEN); 489a9ea25fSXin LI if (inbuf == NULL || outbuf == NULL) 499a9ea25fSXin LI maybe_err("malloc"); 509a9ea25fSXin LI 519a9ea25fSXin LI bzs.bzalloc = NULL; 529a9ea25fSXin LI bzs.bzfree = NULL; 539a9ea25fSXin LI bzs.opaque = NULL; 549a9ea25fSXin LI 559a9ea25fSXin LI end_of_file = 0; 569a9ea25fSXin LI ret = BZ2_bzDecompressInit(&bzs, 0, 0); 579a9ea25fSXin LI if (ret != BZ_OK) 589a9ea25fSXin LI maybe_errx("bzip2 init"); 599a9ea25fSXin LI 609a9ea25fSXin LI /* Prepend. */ 619a9ea25fSXin LI bzs.avail_in = prelen; 629a9ea25fSXin LI bzs.next_in = pre; 639a9ea25fSXin LI 649a9ea25fSXin LI if (bytes_in) 659a9ea25fSXin LI *bytes_in = prelen; 669a9ea25fSXin LI 670c10c611SXin LI while (ret == BZ_OK) { 689a9ea25fSXin LI if (bzs.avail_in == 0 && !end_of_file) { 699a9ea25fSXin LI ssize_t n; 709a9ea25fSXin LI 719a9ea25fSXin LI n = read(in, inbuf, BUFLEN); 729a9ea25fSXin LI if (n < 0) 739a9ea25fSXin LI maybe_err("read"); 749a9ea25fSXin LI if (n == 0) 759a9ea25fSXin LI end_of_file = 1; 769a9ea25fSXin LI bzs.next_in = inbuf; 779a9ea25fSXin LI bzs.avail_in = n; 789a9ea25fSXin LI if (bytes_in) 799a9ea25fSXin LI *bytes_in += n; 809a9ea25fSXin LI } 819a9ea25fSXin LI 829a9ea25fSXin LI bzs.next_out = outbuf; 839a9ea25fSXin LI bzs.avail_out = BUFLEN; 849a9ea25fSXin LI ret = BZ2_bzDecompress(&bzs); 859a9ea25fSXin LI 869a9ea25fSXin LI switch (ret) { 879a9ea25fSXin LI case BZ_STREAM_END: 889a9ea25fSXin LI case BZ_OK: 89a0b4046bSXin LI if (ret == BZ_OK && end_of_file) { 90a0b4046bSXin LI /* 91a0b4046bSXin LI * If we hit this after a stream end, consider 92a0b4046bSXin LI * it as the end of the whole file and don't 93a0b4046bSXin LI * bail out. 94a0b4046bSXin LI */ 95a0b4046bSXin LI if (cold == 1) 96a0b4046bSXin LI ret = BZ_STREAM_END; 97a0b4046bSXin LI else 98a0b4046bSXin LI maybe_errx("truncated file"); 99a0b4046bSXin LI } 100a0b4046bSXin LI cold = 0; 1010c10c611SXin LI if (!tflag && bzs.avail_out != BUFLEN) { 1029a9ea25fSXin LI ssize_t n; 1039a9ea25fSXin LI 1049a9ea25fSXin LI n = write(out, outbuf, BUFLEN - bzs.avail_out); 1059a9ea25fSXin LI if (n < 0) 1069a9ea25fSXin LI maybe_err("write"); 1079a9ea25fSXin LI bytes_out += n; 1089a9ea25fSXin LI } 1090c10c611SXin LI if (ret == BZ_STREAM_END && !end_of_file) { 1100c10c611SXin LI if (BZ2_bzDecompressEnd(&bzs) != BZ_OK || 1110c10c611SXin LI BZ2_bzDecompressInit(&bzs, 0, 0) != BZ_OK) 1120c10c611SXin LI maybe_errx("bzip2 re-init"); 113a0b4046bSXin LI cold = 1; 1140c10c611SXin LI ret = BZ_OK; 1150c10c611SXin LI } 1169a9ea25fSXin LI break; 1179a9ea25fSXin LI 1189a9ea25fSXin LI case BZ_DATA_ERROR: 1199a9ea25fSXin LI maybe_warnx("bzip2 data integrity error"); 1209a9ea25fSXin LI break; 1219a9ea25fSXin LI 1229a9ea25fSXin LI case BZ_DATA_ERROR_MAGIC: 1239a9ea25fSXin LI maybe_warnx("bzip2 magic number error"); 1249a9ea25fSXin LI break; 1259a9ea25fSXin LI 1269a9ea25fSXin LI case BZ_MEM_ERROR: 1279a9ea25fSXin LI maybe_warnx("bzip2 out of memory"); 1289a9ea25fSXin LI break; 1299a9ea25fSXin LI 1300c10c611SXin LI default: 1310c10c611SXin LI maybe_warnx("unknown bzip2 error: %d", ret); 1320c10c611SXin LI break; 1339a9ea25fSXin LI } 1349a9ea25fSXin LI } 1359a9ea25fSXin LI 1369a9ea25fSXin LI if (ret != BZ_STREAM_END || BZ2_bzDecompressEnd(&bzs) != BZ_OK) 1379a9ea25fSXin LI return (-1); 1389a9ea25fSXin LI 1399a9ea25fSXin LI return (bytes_out); 1409a9ea25fSXin LI } 1419a9ea25fSXin LI 142