lrzip/runzip.c
Con Kolivas 8d9b64e1ec Change byte width to be dependant on file size.
This will increase speed of compression and generate a smaller file, but not be backward compatible.
Tweak the way memory is allocated to optimise chances of success and minimise slowdown for the machine.
fsync to empty dirty data before allocating large ram to increase chance of mem allocation and decrease disk thrash of write vs read.
Add lots more information to verbose mode.
Lots of code tidying and minor tweaks.
2010-10-31 15:09:05 +11:00

243 lines
6.5 KiB
C

/*
Copyright (C) Andrew Tridgell 1998-2003
Con Kolivas 2006-2010
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
/* rzip decompression algorithm */
#include "rzip.h"
static inline uchar read_u8(void *ss, int stream)
{
uchar b;
if (read_stream(ss, stream, &b, 1) != 1)
fatal("Stream read u8 failed\n");
return b;
}
static inline u32 read_u32(void *ss, int stream)
{
u32 ret;
if (read_stream(ss, stream, (uchar *)&ret, 4) != 4)
fatal("Stream read u32 failed\n");
return ret;
}
/* Read a variable length of chars dependant on how big the chunk was */
static inline i64 read_vchars(void *ss, int stream, int length)
{
int bytes;
i64 s = 0;
for (bytes = 0; bytes < length; bytes++) {
int bits = bytes * 8;
uchar sb = read_u8(ss, stream);
s |= (i64)sb << bits;
}
return s;
}
static i64 read_header(void *ss, uchar *head)
{
int chunk_bytes = 2;
/* All chunks were unnecessarily encoded 8 bytes wide version 0.4x */
if (control.major_version == 0 && control.minor_version == 4)
chunk_bytes = 8;
*head = read_u8(ss, 0);
return read_vchars(ss, 0, chunk_bytes);
}
static i64 unzip_literal(void *ss, i64 len, int fd_out, uint32 *cksum)
{
uchar *buf;
if (len < 0)
fatal("len %lld is negative in unzip_literal!\n",len);
if (sizeof(long) == 4 && len > (1UL << 31))
fatal("Unable to allocate a chunk this big on 32bit userspace. Can't decompress this file on this hardware\n");
buf = malloc((size_t)len);
if (!buf)
fatal("Failed to allocate literal buffer of size %lld\n", len);
read_stream(ss, 1, buf, len);
if (write_1g(fd_out, buf, (size_t)len) != (ssize_t)len)
fatal("Failed to write literal buffer of size %lld\n", len);
*cksum = CrcUpdate(*cksum, buf, len);
free(buf);
return len;
}
static i64 unzip_match(void *ss, i64 len, int fd_out, int fd_hist, uint32 *cksum, int chunk_bytes)
{
i64 offset, n, total, cur_pos;
if (len < 0)
fatal("len %lld is negative in unzip_match!\n",len);
total = 0;
cur_pos = lseek(fd_out, 0, SEEK_CUR);
if (cur_pos == -1)
fatal("Seek failed on out file in unzip_match.\n");
/* Note the offset is in a different format v0.40+ */
offset = read_vchars(ss, 0, chunk_bytes);
if (lseek(fd_hist, cur_pos - offset, SEEK_SET) == -1)
fatal("Seek failed by %d from %d on history file in unzip_match - %s\n",
offset, cur_pos, strerror(errno));
while (len) {
uchar *buf;
n = MIN(len, offset);
buf = malloc((size_t)n);
if (!buf)
fatal("Failed to allocate %d bytes in unzip_match\n", n);
if (read_1g(fd_hist, buf, (size_t)n) != (ssize_t)n)
fatal("Failed to read %d bytes in unzip_match\n", n);
if (write_1g(fd_out, buf, (size_t)n) != (ssize_t)n)
fatal("Failed to write %d bytes in unzip_match\n", n);
*cksum = CrcUpdate(*cksum, buf, n);
len -= n;
free(buf);
total += n;
}
return total;
}
/* decompress a section of an open file. Call fatal() on error
return the number of bytes that have been retrieved
*/
static i64 runzip_chunk(int fd_in, int fd_out, int fd_hist, i64 expected_size, i64 tally, char chunk_bytes)
{
i64 len, ofs, total = 0;
uint32 good_cksum, cksum = 0;
int l = -1, p = 0;
struct stat st;
uchar head;
void *ss;
/* for display of progress */
char *suffix[] = {"","KB","MB","GB"};
unsigned long divisor[] = {1,1024,1048576,1073741824U};
int divisor_index;
double prog_done, prog_tsize;
if (expected_size > (i64)10737418240ULL) /* > 10GB */
divisor_index = 3;
else if (expected_size > 10485760) /* > 10MB */
divisor_index = 2;
else if (expected_size > 10240) /* > 10KB */
divisor_index = 1;
else
divisor_index = 0;
prog_tsize = (long double)expected_size / (long double)divisor[divisor_index];
ofs = lseek(fd_in, 0, SEEK_CUR);
if (ofs == -1)
fatal("Failed to seek input file in runzip_fd\n");
if (fstat(fd_in, &st) != 0 || st.st_size-ofs == 0)
return 0;
ss = open_stream_in(fd_in, NUM_STREAMS);
if (!ss)
fatal(NULL);
while ((len = read_header(ss, &head)) || head) {
switch (head) {
case 0:
total += unzip_literal(ss, len, fd_out, &cksum);
break;
default:
total += unzip_match(ss, len, fd_out, fd_hist, &cksum, chunk_bytes);
break;
}
p = 100 * ((double)(tally + total) / (double)expected_size);
if (control.flags & FLAG_SHOW_PROGRESS) {
if ( p != l ) {
prog_done = (double)(tally + total) / (double)divisor[divisor_index];
fprintf(control.msgout, "%3d%% %9.2f / %9.2f %s\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b",
p, prog_done, prog_tsize, suffix[divisor_index] );
fflush(control.msgout);
l = p;
}
}
}
good_cksum = read_u32(ss, 0);
if (good_cksum != cksum)
fatal("Bad checksum 0x%08x - expected 0x%08x\n", cksum, good_cksum);
if (close_stream_in(ss) != 0)
fatal("Failed to close stream!\n");
return total;
}
/* decompress a open file. Call fatal() on error
return the number of bytes that have been retrieved
*/
i64 runzip_fd(int fd_in, int fd_out, int fd_hist, i64 expected_size)
{
struct timeval start,end;
char chunk_bytes;
i64 total = 0;
gettimeofday(&start,NULL);
/* Determine the chunk_byte width size. Versions < 0.4 used 4
* bytes for all offsets, version 0.4 used 8 bytes. Versions 0.5+ use
* a variable number of bytes depending on file size.*/
if (control.major_version == 0 && control.minor_version < 4)
chunk_bytes = 4;
else if (control.major_version == 0 && control.minor_version == 4)
chunk_bytes = 8;
else {
int bits = 0;
while (expected_size >> bits > 0)
bits++;
chunk_bytes = bits / 8;
if (bits % 8)
chunk_bytes++;
}
if (control.flags & FLAG_VERBOSE)
fprintf(control.msgout, "Expected size: %lld\nChunk byte width: %d\n", expected_size, chunk_bytes);
while (total < expected_size)
total += runzip_chunk(fd_in, fd_out, fd_hist, expected_size, total, chunk_bytes);
gettimeofday(&end,NULL);
if (control.flags & FLAG_SHOW_PROGRESS)
fprintf(control.msgout, "\nAverage DeCompression Speed: %6.3fMB/s\n",
(total / 1024 / 1024) / (double)((end.tv_sec-start.tv_sec)? : 1));
return total;
}