#include "stand.h"
#include "../libz/zlib.h"
#define EOF (-1)
#define zmemcpy memcpy
#define Z_BUFSIZE 4096
static int gz_magic[2] = {0x1f, 0x8b};
#define ASCII_FLAG 0x01
#define HEAD_CRC 0x02
#define EXTRA_FIELD 0x04
#define ORIG_NAME 0x08
#define COMMENT 0x10
#define RESERVED 0xE0
static struct sd {
z_stream stream;
int z_err;
int z_eof;
int fd;
unsigned char *inbuf;
unsigned long crc;
int transparent;
} *ss[SOPEN_MAX];
#ifdef DEBUG
int z_verbose = 0;
#endif
void *zcalloc(void *, unsigned int, unsigned int);
void zcfree(void *, void *);
void *
zcalloc(void *opaque, unsigned int items, unsigned int size)
{
return(alloc(items * size));
}
void
zcfree(void *opaque, void *ptr)
{
free(ptr, 0);
}
static int
get_byte(struct sd *s)
{
if (s->z_eof)
return EOF;
if (s->stream.avail_in == 0) {
errno = 0;
s->stream.avail_in = oread(s->fd, s->inbuf, Z_BUFSIZE);
if (s->stream.avail_in <= 0) {
s->z_eof = 1;
if (errno)
s->z_err = Z_ERRNO;
return EOF;
}
s->stream.next_in = s->inbuf;
}
s->stream.avail_in--;
return *(s->stream.next_in)++;
}
static unsigned long
getLong(struct sd *s)
{
unsigned long x = (unsigned long)get_byte(s);
int c;
x += ((unsigned long)get_byte(s))<<8;
x += ((unsigned long)get_byte(s))<<16;
c = get_byte(s);
if (c == EOF)
s->z_err = Z_DATA_ERROR;
x += ((unsigned long)c)<<24;
return x;
}
static void
check_header(struct sd *s)
{
int method;
int flags;
unsigned int len;
int c;
for (len = 0; len < 2; len++) {
c = get_byte(s);
if (c != gz_magic[len]) {
if (len != 0) {
s->stream.avail_in++;
s->stream.next_in--;
}
if (c != EOF) {
s->stream.avail_in++;
s->stream.next_in--;
s->transparent = 1;
}
s->z_err = s->stream.avail_in != 0 ? Z_OK :
Z_STREAM_END;
return;
}
}
method = get_byte(s);
flags = get_byte(s);
if (method != Z_DEFLATED || (flags & RESERVED) != 0) {
s->z_err = Z_DATA_ERROR;
return;
}
for (len = 0; len < 6; len++)
(void)get_byte(s);
if ((flags & EXTRA_FIELD) != 0) {
len = (unsigned int)get_byte(s);
len += ((unsigned int)get_byte(s))<<8;
while (len-- != 0 && get_byte(s) != EOF)
;
}
if ((flags & ORIG_NAME) != 0) {
while ((c = get_byte(s)) != 0 && c != EOF)
;
}
if ((flags & COMMENT) != 0) {
while ((c = get_byte(s)) != 0 && c != EOF)
;
}
if ((flags & HEAD_CRC) != 0) {
for (len = 0; len < 2; len++)
(void)get_byte(s);
}
s->z_err = s->z_eof ? Z_DATA_ERROR : Z_OK;
}
int
open(const char *fname, int mode)
{
int fd;
struct sd *s = 0;
if (((fd = oopen(fname, mode)) == -1) ||
(mode != 0))
return(fd);
ss[fd] = s = alloc(sizeof(struct sd));
if (!s)
goto errout;
bzero(s, sizeof(struct sd));
if (inflateInit2(&(s->stream), -15) != Z_OK)
goto errout;
s->stream.next_in = s->inbuf = (unsigned char *)alloc(Z_BUFSIZE);
if (!s->inbuf) {
inflateEnd(&(s->stream));
goto errout;
}
s->fd = fd;
check_header(s);
return(fd);
errout:
if (s)
free(s, sizeof(struct sd));
oclose(fd);
return(-1);
}
int
close(int fd)
{
struct open_file *f;
struct sd *s;
if ((unsigned)fd >= SOPEN_MAX) {
errno = EBADF;
return (-1);
}
f = &files[fd];
if (!(f->f_flags & F_READ))
return(oclose(fd));
s = ss[fd];
if (s != NULL) {
inflateEnd(&(s->stream));
free(s->inbuf, Z_BUFSIZE);
free(s, sizeof(struct sd));
}
return(oclose(fd));
}
ssize_t
read(int fd, void *buf, size_t len)
{
struct sd *s;
unsigned char *start = buf;
s = ss[fd];
if (s->z_err == Z_DATA_ERROR || s->z_err == Z_ERRNO)
return -1;
if (s->z_err == Z_STREAM_END)
return 0;
s->stream.next_out = buf;
s->stream.avail_out = len;
while (s->stream.avail_out != 0) {
if (s->transparent) {
unsigned int n = s->stream.avail_in;
if (n > s->stream.avail_out)
n = s->stream.avail_out;
if (n > 0) {
zmemcpy(s->stream.next_out, s->stream.next_in, n);
s->stream.next_out += n;
s->stream.next_in += n;
s->stream.avail_out -= n;
s->stream.avail_in -= n;
}
if (s->stream.avail_out > 0) {
int n;
n = oread(fd, s->stream.next_out,
s->stream.avail_out);
if (n <= 0) {
s->z_eof = 1;
if (errno) {
s->z_err = Z_ERRNO;
break;
}
}
s->stream.avail_out -= n;
}
len -= s->stream.avail_out;
s->stream.total_in += (unsigned long)len;
s->stream.total_out += (unsigned long)len;
if (len == 0)
s->z_eof = 1;
return (int)len;
}
if (s->stream.avail_in == 0 && !s->z_eof) {
errno = 0;
s->stream.avail_in = oread(fd, s->inbuf, Z_BUFSIZE);
if (s->stream.avail_in <= 0) {
s->z_eof = 1;
if (errno) {
s->z_err = Z_ERRNO;
break;
}
}
s->stream.next_in = s->inbuf;
}
s->z_err = inflate(&(s->stream), Z_NO_FLUSH);
if (s->z_err == Z_STREAM_END) {
s->crc = crc32(s->crc, start,
(unsigned int)(s->stream.next_out - start));
start = s->stream.next_out;
if (getLong(s) != s->crc) {
s->z_err = Z_DATA_ERROR;
} else {
(void)getLong(s);
check_header(s);
if (s->z_err == Z_OK) {
unsigned long total_in = s->stream.total_in;
unsigned long total_out = s->stream.total_out;
inflateReset(&(s->stream));
s->stream.total_in = total_in;
s->stream.total_out = total_out;
s->crc = crc32(0L, Z_NULL, 0);
}
}
}
if (s->z_err != Z_OK || s->z_eof)
break;
}
s->crc = crc32(s->crc, start, (unsigned int)(s->stream.next_out - start));
return (int)(len - s->stream.avail_out);
}
off_t
lseek(int fd, off_t offset, int where)
{
struct open_file *f;
struct sd *s;
if ((unsigned)fd >= SOPEN_MAX) {
errno = EBADF;
return (-1);
}
f = &files[fd];
if (!(f->f_flags & F_READ))
return(olseek(fd, offset, where));
s = ss[fd];
if (s->transparent) {
off_t res = olseek(fd, offset, where);
if (res != (off_t)-1) {
s->stream.avail_in = 0;
}
return(res);
}
switch(where) {
case SEEK_CUR:
offset += s->stream.total_out;
case SEEK_SET:
if (offset < s->stream.total_out) {
off_t res;
void *sav_inbuf;
res = olseek(fd, 0, SEEK_SET);
if (res == (off_t)-1)
return(res);
inflateEnd(&(s->stream));
sav_inbuf = s->inbuf;
bzero(s, sizeof(struct sd));
inflateInit2(&(s->stream), -15);
s->stream.next_in = s->inbuf = sav_inbuf;
s->fd = fd;
check_header(s);
}
if (offset > s->stream.total_out) {
off_t toskip = offset - s->stream.total_out;
while(toskip > 0) {
#define DUMMYBUFSIZE 256
char dummybuf[DUMMYBUFSIZE];
size_t len = toskip;
ssize_t n;
if (len > DUMMYBUFSIZE)
len = DUMMYBUFSIZE;
n = read(fd, dummybuf, len);
if (n <= 0) {
if (n == 0)
errno = EINVAL;
return((off_t)-1);
}
toskip -= n;
}
}
#ifdef DEBUG
if (offset != s->stream.total_out)
panic("lseek compressed");
#endif
return(offset);
case SEEK_END:
errno = EOFFSET;
break;
default:
errno = EINVAL;
}
return((off_t)-1);
}