1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
|
/* SPDX-License-Identifier: GPL-3.0-or-later */
/*
* istream.c
*
* Copyright (C) 2019 David Oberhollenzer <goliath@infraroot.at>
*/
#include "../internal.h"
typedef struct istream_xfrm_t {
istream_t base;
istream_t *wrapped;
xfrm_stream_t *xfrm;
sqfs_u8 uncompressed[BUFSZ];
} istream_xfrm_t;
static int xfrm_precache(istream_t *base)
{
istream_xfrm_t *xfrm = (istream_xfrm_t *)base;
int ret;
if (base->eof)
return 0;
assert(base->buffer >= xfrm->uncompressed);
assert(base->buffer <= (xfrm->uncompressed + BUFSZ));
assert(base->buffer_used <= BUFSZ);
assert((size_t)(base->buffer - xfrm->uncompressed) <=
(BUFSZ - base->buffer_used));
if (base->buffer_used > 0)
memmove(xfrm->uncompressed, base->buffer, base->buffer_used);
base->buffer = xfrm->uncompressed;
ret = istream_precache(xfrm->wrapped);
if (ret != 0)
return ret;
for (;;) {
sqfs_u32 in_off = 0, out_off = base->buffer_used;
int mode = XFRM_STREAM_FLUSH_NONE;
if (xfrm->wrapped->eof)
mode = XFRM_STREAM_FLUSH_FULL;
ret = xfrm->xfrm->process_data(xfrm->xfrm,
xfrm->wrapped->buffer,
xfrm->wrapped->buffer_used,
xfrm->uncompressed + out_off,
BUFSZ - out_off,
&in_off, &out_off, mode);
if (ret == XFRM_STREAM_ERROR) {
fprintf(stderr, "%s: internal error in decompressor.\n",
base->get_filename(base));
return -1;
}
base->buffer_used = out_off;
xfrm->wrapped->buffer += in_off;
xfrm->wrapped->buffer_used -= in_off;
if (ret == XFRM_STREAM_BUFFER_FULL || out_off >= BUFSZ)
break;
ret = istream_precache(xfrm->wrapped);
if (ret != 0)
return ret;
if (xfrm->wrapped->eof && xfrm->wrapped->buffer_used == 0) {
if (base->buffer_used == 0)
base->eof = true;
break;
}
}
return 0;
}
static const char *xfrm_get_filename(istream_t *strm)
{
istream_xfrm_t *xfrm = (istream_xfrm_t *)strm;
return xfrm->wrapped->get_filename(xfrm->wrapped);
}
static void xfrm_destroy(sqfs_object_t *obj)
{
istream_xfrm_t *xfrm = (istream_xfrm_t *)obj;
sqfs_drop(xfrm->xfrm);
sqfs_drop(xfrm->wrapped);
free(xfrm);
}
istream_t *istream_xfrm_create(istream_t *strm, xfrm_stream_t *xfrm)
{
istream_xfrm_t *stream = calloc(1, sizeof(*stream));
istream_t *base = (istream_t *)stream;
if (stream == NULL)
goto fail;
sqfs_object_init(stream, xfrm_destroy, NULL);
stream->wrapped = sqfs_grab(strm);
stream->xfrm = sqfs_grab(xfrm);
base->precache = xfrm_precache;
base->get_filename = xfrm_get_filename;
base->buffer = stream->uncompressed;
base->eof = false;
return base;
fail:
fprintf(stderr, "%s: error initializing decompressor stream.\n",
strm->get_filename(strm));
return NULL;
}
|