1*27162e4eSAndroid Build Coastguard Worker /* LZ4frame API example : compress a file
2*27162e4eSAndroid Build Coastguard Worker * Modified from an example code by Zbigniew Jędrzejewski-Szmek
3*27162e4eSAndroid Build Coastguard Worker *
4*27162e4eSAndroid Build Coastguard Worker * This example streams an input file into an output file
5*27162e4eSAndroid Build Coastguard Worker * using a bounded memory budget.
6*27162e4eSAndroid Build Coastguard Worker * Input is read in chunks of IN_CHUNK_SIZE */
7*27162e4eSAndroid Build Coastguard Worker
8*27162e4eSAndroid Build Coastguard Worker #include <stdio.h>
9*27162e4eSAndroid Build Coastguard Worker #include <stdlib.h>
10*27162e4eSAndroid Build Coastguard Worker #include <string.h>
11*27162e4eSAndroid Build Coastguard Worker #include <errno.h>
12*27162e4eSAndroid Build Coastguard Worker #include <assert.h>
13*27162e4eSAndroid Build Coastguard Worker
14*27162e4eSAndroid Build Coastguard Worker #include <getopt.h>
15*27162e4eSAndroid Build Coastguard Worker #include <lz4frame.h>
16*27162e4eSAndroid Build Coastguard Worker #include <lz4frame_static.h>
17*27162e4eSAndroid Build Coastguard Worker
18*27162e4eSAndroid Build Coastguard Worker #define IN_CHUNK_SIZE (16*1024)
19*27162e4eSAndroid Build Coastguard Worker
20*27162e4eSAndroid Build Coastguard Worker static const LZ4F_preferences_t kPrefs = {
21*27162e4eSAndroid Build Coastguard Worker { LZ4F_max256KB, LZ4F_blockLinked, LZ4F_noContentChecksum, LZ4F_frame,
22*27162e4eSAndroid Build Coastguard Worker 0 /* unknown content size */, 0 /* no dictID */ , LZ4F_noBlockChecksum },
23*27162e4eSAndroid Build Coastguard Worker 0, /* compression level; 0 == default */
24*27162e4eSAndroid Build Coastguard Worker 0, /* autoflush */
25*27162e4eSAndroid Build Coastguard Worker 0, /* favor decompression speed */
26*27162e4eSAndroid Build Coastguard Worker { 0, 0, 0 }, /* reserved, must be set to 0 */
27*27162e4eSAndroid Build Coastguard Worker };
28*27162e4eSAndroid Build Coastguard Worker
29*27162e4eSAndroid Build Coastguard Worker
30*27162e4eSAndroid Build Coastguard Worker /* safe_fwrite() :
31*27162e4eSAndroid Build Coastguard Worker * performs fwrite(), ensure operation success, or immediately exit() */
safe_fwrite(void * buf,size_t eltSize,size_t nbElt,FILE * f)32*27162e4eSAndroid Build Coastguard Worker static void safe_fwrite(void* buf, size_t eltSize, size_t nbElt, FILE* f)
33*27162e4eSAndroid Build Coastguard Worker {
34*27162e4eSAndroid Build Coastguard Worker size_t const writtenSize = fwrite(buf, eltSize, nbElt, f);
35*27162e4eSAndroid Build Coastguard Worker size_t const expectedSize = eltSize * nbElt;
36*27162e4eSAndroid Build Coastguard Worker if (nbElt>0) assert(expectedSize / nbElt == eltSize); /* check overflow */
37*27162e4eSAndroid Build Coastguard Worker if (writtenSize < expectedSize) {
38*27162e4eSAndroid Build Coastguard Worker if (ferror(f)) /* note : ferror() must follow fwrite */
39*27162e4eSAndroid Build Coastguard Worker fprintf(stderr, "Write failed \n");
40*27162e4eSAndroid Build Coastguard Worker else
41*27162e4eSAndroid Build Coastguard Worker fprintf(stderr, "Write too short \n");
42*27162e4eSAndroid Build Coastguard Worker exit(1);
43*27162e4eSAndroid Build Coastguard Worker }
44*27162e4eSAndroid Build Coastguard Worker }
45*27162e4eSAndroid Build Coastguard Worker
46*27162e4eSAndroid Build Coastguard Worker
47*27162e4eSAndroid Build Coastguard Worker /* ================================================= */
48*27162e4eSAndroid Build Coastguard Worker /* Streaming Compression example */
49*27162e4eSAndroid Build Coastguard Worker /* ================================================= */
50*27162e4eSAndroid Build Coastguard Worker
51*27162e4eSAndroid Build Coastguard Worker typedef struct {
52*27162e4eSAndroid Build Coastguard Worker int error;
53*27162e4eSAndroid Build Coastguard Worker unsigned long long size_in;
54*27162e4eSAndroid Build Coastguard Worker unsigned long long size_out;
55*27162e4eSAndroid Build Coastguard Worker } compressResult_t;
56*27162e4eSAndroid Build Coastguard Worker
57*27162e4eSAndroid Build Coastguard Worker static compressResult_t
compress_file_internal(FILE * f_in,FILE * f_out,LZ4F_compressionContext_t ctx,void * inBuff,size_t inChunkSize,void * outBuff,size_t outCapacity,FILE * f_unc,long uncOffset)58*27162e4eSAndroid Build Coastguard Worker compress_file_internal(FILE* f_in, FILE* f_out,
59*27162e4eSAndroid Build Coastguard Worker LZ4F_compressionContext_t ctx,
60*27162e4eSAndroid Build Coastguard Worker void* inBuff, size_t inChunkSize,
61*27162e4eSAndroid Build Coastguard Worker void* outBuff, size_t outCapacity,
62*27162e4eSAndroid Build Coastguard Worker FILE* f_unc, long uncOffset)
63*27162e4eSAndroid Build Coastguard Worker {
64*27162e4eSAndroid Build Coastguard Worker compressResult_t result = { 1, 0, 0 }; /* result for an error */
65*27162e4eSAndroid Build Coastguard Worker long long count_in = 0, count_out, bytesToOffset = -1;
66*27162e4eSAndroid Build Coastguard Worker
67*27162e4eSAndroid Build Coastguard Worker assert(f_in != NULL); assert(f_out != NULL);
68*27162e4eSAndroid Build Coastguard Worker assert(ctx != NULL);
69*27162e4eSAndroid Build Coastguard Worker assert(outCapacity >= LZ4F_HEADER_SIZE_MAX);
70*27162e4eSAndroid Build Coastguard Worker assert(outCapacity >= LZ4F_compressBound(inChunkSize, &kPrefs));
71*27162e4eSAndroid Build Coastguard Worker
72*27162e4eSAndroid Build Coastguard Worker /* write frame header */
73*27162e4eSAndroid Build Coastguard Worker { size_t const headerSize = LZ4F_compressBegin(ctx, outBuff, outCapacity, &kPrefs);
74*27162e4eSAndroid Build Coastguard Worker if (LZ4F_isError(headerSize)) {
75*27162e4eSAndroid Build Coastguard Worker printf("Failed to start compression: error %u \n", (unsigned)headerSize);
76*27162e4eSAndroid Build Coastguard Worker return result;
77*27162e4eSAndroid Build Coastguard Worker }
78*27162e4eSAndroid Build Coastguard Worker count_out = headerSize;
79*27162e4eSAndroid Build Coastguard Worker printf("Buffer size is %u bytes, header size %u bytes \n",
80*27162e4eSAndroid Build Coastguard Worker (unsigned)outCapacity, (unsigned)headerSize);
81*27162e4eSAndroid Build Coastguard Worker safe_fwrite(outBuff, 1, headerSize, f_out);
82*27162e4eSAndroid Build Coastguard Worker }
83*27162e4eSAndroid Build Coastguard Worker
84*27162e4eSAndroid Build Coastguard Worker /* stream file */
85*27162e4eSAndroid Build Coastguard Worker for (;;) {
86*27162e4eSAndroid Build Coastguard Worker size_t compressedSize;
87*27162e4eSAndroid Build Coastguard Worker long long inSize = IN_CHUNK_SIZE;
88*27162e4eSAndroid Build Coastguard Worker if (uncOffset >= 0) {
89*27162e4eSAndroid Build Coastguard Worker bytesToOffset = uncOffset - count_in;
90*27162e4eSAndroid Build Coastguard Worker
91*27162e4eSAndroid Build Coastguard Worker /* read only remaining bytes to offset position */
92*27162e4eSAndroid Build Coastguard Worker if (bytesToOffset < IN_CHUNK_SIZE && bytesToOffset > 0) {
93*27162e4eSAndroid Build Coastguard Worker inSize = bytesToOffset;
94*27162e4eSAndroid Build Coastguard Worker }
95*27162e4eSAndroid Build Coastguard Worker }
96*27162e4eSAndroid Build Coastguard Worker
97*27162e4eSAndroid Build Coastguard Worker /* input data is at uncompressed data offset */
98*27162e4eSAndroid Build Coastguard Worker if (bytesToOffset <= 0 && uncOffset >= 0 && f_unc) {
99*27162e4eSAndroid Build Coastguard Worker size_t const readSize = fread(inBuff, 1, inSize, f_unc);
100*27162e4eSAndroid Build Coastguard Worker if (readSize == 0) {
101*27162e4eSAndroid Build Coastguard Worker uncOffset = -1;
102*27162e4eSAndroid Build Coastguard Worker continue;
103*27162e4eSAndroid Build Coastguard Worker }
104*27162e4eSAndroid Build Coastguard Worker count_in += readSize;
105*27162e4eSAndroid Build Coastguard Worker compressedSize = LZ4F_uncompressedUpdate(ctx,
106*27162e4eSAndroid Build Coastguard Worker outBuff, outCapacity,
107*27162e4eSAndroid Build Coastguard Worker inBuff, readSize,
108*27162e4eSAndroid Build Coastguard Worker NULL);
109*27162e4eSAndroid Build Coastguard Worker } else {
110*27162e4eSAndroid Build Coastguard Worker size_t const readSize = fread(inBuff, 1, inSize, f_in);
111*27162e4eSAndroid Build Coastguard Worker if (readSize == 0) break; /* nothing left to read from input file */
112*27162e4eSAndroid Build Coastguard Worker count_in += readSize;
113*27162e4eSAndroid Build Coastguard Worker compressedSize = LZ4F_compressUpdate(ctx,
114*27162e4eSAndroid Build Coastguard Worker outBuff, outCapacity,
115*27162e4eSAndroid Build Coastguard Worker inBuff, readSize,
116*27162e4eSAndroid Build Coastguard Worker NULL);
117*27162e4eSAndroid Build Coastguard Worker
118*27162e4eSAndroid Build Coastguard Worker }
119*27162e4eSAndroid Build Coastguard Worker
120*27162e4eSAndroid Build Coastguard Worker if (LZ4F_isError(compressedSize)) {
121*27162e4eSAndroid Build Coastguard Worker printf("Compression failed: error %u \n", (unsigned)compressedSize);
122*27162e4eSAndroid Build Coastguard Worker return result;
123*27162e4eSAndroid Build Coastguard Worker }
124*27162e4eSAndroid Build Coastguard Worker
125*27162e4eSAndroid Build Coastguard Worker printf("Writing %u bytes\n", (unsigned)compressedSize);
126*27162e4eSAndroid Build Coastguard Worker safe_fwrite(outBuff, 1, compressedSize, f_out);
127*27162e4eSAndroid Build Coastguard Worker count_out += compressedSize;
128*27162e4eSAndroid Build Coastguard Worker }
129*27162e4eSAndroid Build Coastguard Worker
130*27162e4eSAndroid Build Coastguard Worker /* flush whatever remains within internal buffers */
131*27162e4eSAndroid Build Coastguard Worker { size_t const compressedSize = LZ4F_compressEnd(ctx,
132*27162e4eSAndroid Build Coastguard Worker outBuff, outCapacity,
133*27162e4eSAndroid Build Coastguard Worker NULL);
134*27162e4eSAndroid Build Coastguard Worker if (LZ4F_isError(compressedSize)) {
135*27162e4eSAndroid Build Coastguard Worker printf("Failed to end compression: error %u \n", (unsigned)compressedSize);
136*27162e4eSAndroid Build Coastguard Worker return result;
137*27162e4eSAndroid Build Coastguard Worker }
138*27162e4eSAndroid Build Coastguard Worker
139*27162e4eSAndroid Build Coastguard Worker printf("Writing %u bytes \n", (unsigned)compressedSize);
140*27162e4eSAndroid Build Coastguard Worker safe_fwrite(outBuff, 1, compressedSize, f_out);
141*27162e4eSAndroid Build Coastguard Worker count_out += compressedSize;
142*27162e4eSAndroid Build Coastguard Worker }
143*27162e4eSAndroid Build Coastguard Worker
144*27162e4eSAndroid Build Coastguard Worker result.size_in = count_in;
145*27162e4eSAndroid Build Coastguard Worker result.size_out = count_out;
146*27162e4eSAndroid Build Coastguard Worker result.error = 0;
147*27162e4eSAndroid Build Coastguard Worker return result;
148*27162e4eSAndroid Build Coastguard Worker }
149*27162e4eSAndroid Build Coastguard Worker
150*27162e4eSAndroid Build Coastguard Worker static compressResult_t
compress_file(FILE * f_in,FILE * f_out,FILE * f_unc,int uncOffset)151*27162e4eSAndroid Build Coastguard Worker compress_file(FILE* f_in, FILE* f_out,
152*27162e4eSAndroid Build Coastguard Worker FILE* f_unc, int uncOffset)
153*27162e4eSAndroid Build Coastguard Worker {
154*27162e4eSAndroid Build Coastguard Worker assert(f_in != NULL);
155*27162e4eSAndroid Build Coastguard Worker assert(f_out != NULL);
156*27162e4eSAndroid Build Coastguard Worker
157*27162e4eSAndroid Build Coastguard Worker /* resource allocation */
158*27162e4eSAndroid Build Coastguard Worker LZ4F_compressionContext_t ctx;
159*27162e4eSAndroid Build Coastguard Worker size_t const ctxCreation = LZ4F_createCompressionContext(&ctx, LZ4F_VERSION);
160*27162e4eSAndroid Build Coastguard Worker void* const src = malloc(IN_CHUNK_SIZE);
161*27162e4eSAndroid Build Coastguard Worker size_t const outbufCapacity = LZ4F_compressBound(IN_CHUNK_SIZE, &kPrefs); /* large enough for any input <= IN_CHUNK_SIZE */
162*27162e4eSAndroid Build Coastguard Worker void* const outbuff = malloc(outbufCapacity);
163*27162e4eSAndroid Build Coastguard Worker
164*27162e4eSAndroid Build Coastguard Worker compressResult_t result = { 1, 0, 0 }; /* == error (default) */
165*27162e4eSAndroid Build Coastguard Worker if (!LZ4F_isError(ctxCreation) && src && outbuff) {
166*27162e4eSAndroid Build Coastguard Worker result = compress_file_internal(f_in, f_out,
167*27162e4eSAndroid Build Coastguard Worker ctx,
168*27162e4eSAndroid Build Coastguard Worker src, IN_CHUNK_SIZE,
169*27162e4eSAndroid Build Coastguard Worker outbuff, outbufCapacity,
170*27162e4eSAndroid Build Coastguard Worker f_unc, uncOffset);
171*27162e4eSAndroid Build Coastguard Worker } else {
172*27162e4eSAndroid Build Coastguard Worker printf("error : resource allocation failed \n");
173*27162e4eSAndroid Build Coastguard Worker }
174*27162e4eSAndroid Build Coastguard Worker
175*27162e4eSAndroid Build Coastguard Worker LZ4F_freeCompressionContext(ctx); /* supports free on NULL */
176*27162e4eSAndroid Build Coastguard Worker free(src);
177*27162e4eSAndroid Build Coastguard Worker free(outbuff);
178*27162e4eSAndroid Build Coastguard Worker return result;
179*27162e4eSAndroid Build Coastguard Worker }
180*27162e4eSAndroid Build Coastguard Worker
181*27162e4eSAndroid Build Coastguard Worker
182*27162e4eSAndroid Build Coastguard Worker /* ================================================= */
183*27162e4eSAndroid Build Coastguard Worker /* Streaming decompression example */
184*27162e4eSAndroid Build Coastguard Worker /* ================================================= */
185*27162e4eSAndroid Build Coastguard Worker
get_block_size(const LZ4F_frameInfo_t * info)186*27162e4eSAndroid Build Coastguard Worker static size_t get_block_size(const LZ4F_frameInfo_t* info) {
187*27162e4eSAndroid Build Coastguard Worker switch (info->blockSizeID) {
188*27162e4eSAndroid Build Coastguard Worker case LZ4F_default:
189*27162e4eSAndroid Build Coastguard Worker case LZ4F_max64KB: return 1 << 16;
190*27162e4eSAndroid Build Coastguard Worker case LZ4F_max256KB: return 1 << 18;
191*27162e4eSAndroid Build Coastguard Worker case LZ4F_max1MB: return 1 << 20;
192*27162e4eSAndroid Build Coastguard Worker case LZ4F_max4MB: return 1 << 22;
193*27162e4eSAndroid Build Coastguard Worker default:
194*27162e4eSAndroid Build Coastguard Worker printf("Impossible with expected frame specification (<=v1.6.1)\n");
195*27162e4eSAndroid Build Coastguard Worker exit(1);
196*27162e4eSAndroid Build Coastguard Worker }
197*27162e4eSAndroid Build Coastguard Worker }
198*27162e4eSAndroid Build Coastguard Worker
199*27162e4eSAndroid Build Coastguard Worker /* @return : 1==error, 0==success */
200*27162e4eSAndroid Build Coastguard Worker static int
decompress_file_internal(FILE * f_in,FILE * f_out,LZ4F_dctx * dctx,void * src,size_t srcCapacity,size_t filled,size_t alreadyConsumed,void * dst,size_t dstCapacity)201*27162e4eSAndroid Build Coastguard Worker decompress_file_internal(FILE* f_in, FILE* f_out,
202*27162e4eSAndroid Build Coastguard Worker LZ4F_dctx* dctx,
203*27162e4eSAndroid Build Coastguard Worker void* src, size_t srcCapacity, size_t filled, size_t alreadyConsumed,
204*27162e4eSAndroid Build Coastguard Worker void* dst, size_t dstCapacity)
205*27162e4eSAndroid Build Coastguard Worker {
206*27162e4eSAndroid Build Coastguard Worker int firstChunk = 1;
207*27162e4eSAndroid Build Coastguard Worker size_t ret = 1;
208*27162e4eSAndroid Build Coastguard Worker
209*27162e4eSAndroid Build Coastguard Worker assert(f_in != NULL); assert(f_out != NULL);
210*27162e4eSAndroid Build Coastguard Worker assert(dctx != NULL);
211*27162e4eSAndroid Build Coastguard Worker assert(src != NULL); assert(srcCapacity > 0); assert(filled <= srcCapacity); assert(alreadyConsumed <= filled);
212*27162e4eSAndroid Build Coastguard Worker assert(dst != NULL); assert(dstCapacity > 0);
213*27162e4eSAndroid Build Coastguard Worker
214*27162e4eSAndroid Build Coastguard Worker /* Decompression */
215*27162e4eSAndroid Build Coastguard Worker while (ret != 0) {
216*27162e4eSAndroid Build Coastguard Worker /* Load more input */
217*27162e4eSAndroid Build Coastguard Worker size_t readSize = firstChunk ? filled : fread(src, 1, srcCapacity, f_in); firstChunk=0;
218*27162e4eSAndroid Build Coastguard Worker const void* srcPtr = (const char*)src + alreadyConsumed; alreadyConsumed=0;
219*27162e4eSAndroid Build Coastguard Worker const void* const srcEnd = (const char*)srcPtr + readSize;
220*27162e4eSAndroid Build Coastguard Worker if (readSize == 0 || ferror(f_in)) {
221*27162e4eSAndroid Build Coastguard Worker printf("Decompress: not enough input or error reading file\n");
222*27162e4eSAndroid Build Coastguard Worker return 1;
223*27162e4eSAndroid Build Coastguard Worker }
224*27162e4eSAndroid Build Coastguard Worker
225*27162e4eSAndroid Build Coastguard Worker /* Decompress:
226*27162e4eSAndroid Build Coastguard Worker * Continue while there is more input to read (srcPtr != srcEnd)
227*27162e4eSAndroid Build Coastguard Worker * and the frame isn't over (ret != 0)
228*27162e4eSAndroid Build Coastguard Worker */
229*27162e4eSAndroid Build Coastguard Worker while (srcPtr < srcEnd && ret != 0) {
230*27162e4eSAndroid Build Coastguard Worker /* Any data within dst has been flushed at this stage */
231*27162e4eSAndroid Build Coastguard Worker size_t dstSize = dstCapacity;
232*27162e4eSAndroid Build Coastguard Worker size_t srcSize = (const char*)srcEnd - (const char*)srcPtr;
233*27162e4eSAndroid Build Coastguard Worker ret = LZ4F_decompress(dctx, dst, &dstSize, srcPtr, &srcSize, /* LZ4F_decompressOptions_t */ NULL);
234*27162e4eSAndroid Build Coastguard Worker if (LZ4F_isError(ret)) {
235*27162e4eSAndroid Build Coastguard Worker printf("Decompression error: %s\n", LZ4F_getErrorName(ret));
236*27162e4eSAndroid Build Coastguard Worker return 1;
237*27162e4eSAndroid Build Coastguard Worker }
238*27162e4eSAndroid Build Coastguard Worker /* Flush output */
239*27162e4eSAndroid Build Coastguard Worker if (dstSize != 0) safe_fwrite(dst, 1, dstSize, f_out);
240*27162e4eSAndroid Build Coastguard Worker /* Update input */
241*27162e4eSAndroid Build Coastguard Worker srcPtr = (const char*)srcPtr + srcSize;
242*27162e4eSAndroid Build Coastguard Worker }
243*27162e4eSAndroid Build Coastguard Worker
244*27162e4eSAndroid Build Coastguard Worker assert(srcPtr <= srcEnd);
245*27162e4eSAndroid Build Coastguard Worker
246*27162e4eSAndroid Build Coastguard Worker /* Ensure all input data has been consumed.
247*27162e4eSAndroid Build Coastguard Worker * It is valid to have multiple frames in the same file,
248*27162e4eSAndroid Build Coastguard Worker * but this example only supports one frame.
249*27162e4eSAndroid Build Coastguard Worker */
250*27162e4eSAndroid Build Coastguard Worker if (srcPtr < srcEnd) {
251*27162e4eSAndroid Build Coastguard Worker printf("Decompress: Trailing data left in file after frame\n");
252*27162e4eSAndroid Build Coastguard Worker return 1;
253*27162e4eSAndroid Build Coastguard Worker }
254*27162e4eSAndroid Build Coastguard Worker }
255*27162e4eSAndroid Build Coastguard Worker
256*27162e4eSAndroid Build Coastguard Worker /* Check that there isn't trailing data in the file after the frame.
257*27162e4eSAndroid Build Coastguard Worker * It is valid to have multiple frames in the same file,
258*27162e4eSAndroid Build Coastguard Worker * but this example only supports one frame.
259*27162e4eSAndroid Build Coastguard Worker */
260*27162e4eSAndroid Build Coastguard Worker { size_t const readSize = fread(src, 1, 1, f_in);
261*27162e4eSAndroid Build Coastguard Worker if (readSize != 0 || !feof(f_in)) {
262*27162e4eSAndroid Build Coastguard Worker printf("Decompress: Trailing data left in file after frame\n");
263*27162e4eSAndroid Build Coastguard Worker return 1;
264*27162e4eSAndroid Build Coastguard Worker } }
265*27162e4eSAndroid Build Coastguard Worker
266*27162e4eSAndroid Build Coastguard Worker return 0;
267*27162e4eSAndroid Build Coastguard Worker }
268*27162e4eSAndroid Build Coastguard Worker
269*27162e4eSAndroid Build Coastguard Worker
270*27162e4eSAndroid Build Coastguard Worker /* @return : 1==error, 0==completed */
271*27162e4eSAndroid Build Coastguard Worker static int
decompress_file_allocDst(FILE * f_in,FILE * f_out,LZ4F_dctx * dctx,void * src,size_t srcCapacity)272*27162e4eSAndroid Build Coastguard Worker decompress_file_allocDst(FILE* f_in, FILE* f_out,
273*27162e4eSAndroid Build Coastguard Worker LZ4F_dctx* dctx,
274*27162e4eSAndroid Build Coastguard Worker void* src, size_t srcCapacity)
275*27162e4eSAndroid Build Coastguard Worker {
276*27162e4eSAndroid Build Coastguard Worker assert(f_in != NULL); assert(f_out != NULL);
277*27162e4eSAndroid Build Coastguard Worker assert(dctx != NULL);
278*27162e4eSAndroid Build Coastguard Worker assert(src != NULL);
279*27162e4eSAndroid Build Coastguard Worker assert(srcCapacity >= LZ4F_HEADER_SIZE_MAX); /* ensure LZ4F_getFrameInfo() can read enough data */
280*27162e4eSAndroid Build Coastguard Worker
281*27162e4eSAndroid Build Coastguard Worker /* Read Frame header */
282*27162e4eSAndroid Build Coastguard Worker size_t const readSize = fread(src, 1, srcCapacity, f_in);
283*27162e4eSAndroid Build Coastguard Worker if (readSize == 0 || ferror(f_in)) {
284*27162e4eSAndroid Build Coastguard Worker printf("Decompress: not enough input or error reading file\n");
285*27162e4eSAndroid Build Coastguard Worker return 1;
286*27162e4eSAndroid Build Coastguard Worker }
287*27162e4eSAndroid Build Coastguard Worker
288*27162e4eSAndroid Build Coastguard Worker LZ4F_frameInfo_t info;
289*27162e4eSAndroid Build Coastguard Worker size_t consumedSize = readSize;
290*27162e4eSAndroid Build Coastguard Worker { size_t const fires = LZ4F_getFrameInfo(dctx, &info, src, &consumedSize);
291*27162e4eSAndroid Build Coastguard Worker if (LZ4F_isError(fires)) {
292*27162e4eSAndroid Build Coastguard Worker printf("LZ4F_getFrameInfo error: %s\n", LZ4F_getErrorName(fires));
293*27162e4eSAndroid Build Coastguard Worker return 1;
294*27162e4eSAndroid Build Coastguard Worker } }
295*27162e4eSAndroid Build Coastguard Worker
296*27162e4eSAndroid Build Coastguard Worker /* Allocating enough space for an entire block isn't necessary for
297*27162e4eSAndroid Build Coastguard Worker * correctness, but it allows some memcpy's to be elided.
298*27162e4eSAndroid Build Coastguard Worker */
299*27162e4eSAndroid Build Coastguard Worker size_t const dstCapacity = get_block_size(&info);
300*27162e4eSAndroid Build Coastguard Worker void* const dst = malloc(dstCapacity);
301*27162e4eSAndroid Build Coastguard Worker if (!dst) { perror("decompress_file(dst)"); return 1; }
302*27162e4eSAndroid Build Coastguard Worker
303*27162e4eSAndroid Build Coastguard Worker int const decompressionResult = decompress_file_internal(
304*27162e4eSAndroid Build Coastguard Worker f_in, f_out,
305*27162e4eSAndroid Build Coastguard Worker dctx,
306*27162e4eSAndroid Build Coastguard Worker src, srcCapacity, readSize-consumedSize, consumedSize,
307*27162e4eSAndroid Build Coastguard Worker dst, dstCapacity);
308*27162e4eSAndroid Build Coastguard Worker
309*27162e4eSAndroid Build Coastguard Worker free(dst);
310*27162e4eSAndroid Build Coastguard Worker return decompressionResult;
311*27162e4eSAndroid Build Coastguard Worker }
312*27162e4eSAndroid Build Coastguard Worker
313*27162e4eSAndroid Build Coastguard Worker
314*27162e4eSAndroid Build Coastguard Worker /* @result : 1==error, 0==success */
decompress_file(FILE * f_in,FILE * f_out)315*27162e4eSAndroid Build Coastguard Worker static int decompress_file(FILE* f_in, FILE* f_out)
316*27162e4eSAndroid Build Coastguard Worker {
317*27162e4eSAndroid Build Coastguard Worker assert(f_in != NULL); assert(f_out != NULL);
318*27162e4eSAndroid Build Coastguard Worker
319*27162e4eSAndroid Build Coastguard Worker /* Resource allocation */
320*27162e4eSAndroid Build Coastguard Worker void* const src = malloc(IN_CHUNK_SIZE);
321*27162e4eSAndroid Build Coastguard Worker if (!src) { perror("decompress_file(src)"); return 1; }
322*27162e4eSAndroid Build Coastguard Worker
323*27162e4eSAndroid Build Coastguard Worker LZ4F_dctx* dctx;
324*27162e4eSAndroid Build Coastguard Worker { size_t const dctxStatus = LZ4F_createDecompressionContext(&dctx, LZ4F_VERSION);
325*27162e4eSAndroid Build Coastguard Worker if (LZ4F_isError(dctxStatus)) {
326*27162e4eSAndroid Build Coastguard Worker printf("LZ4F_dctx creation error: %s\n", LZ4F_getErrorName(dctxStatus));
327*27162e4eSAndroid Build Coastguard Worker } }
328*27162e4eSAndroid Build Coastguard Worker
329*27162e4eSAndroid Build Coastguard Worker int const result = !dctx ? 1 /* error */ :
330*27162e4eSAndroid Build Coastguard Worker decompress_file_allocDst(f_in, f_out, dctx, src, IN_CHUNK_SIZE);
331*27162e4eSAndroid Build Coastguard Worker
332*27162e4eSAndroid Build Coastguard Worker free(src);
333*27162e4eSAndroid Build Coastguard Worker LZ4F_freeDecompressionContext(dctx); /* note : free works on NULL */
334*27162e4eSAndroid Build Coastguard Worker return result;
335*27162e4eSAndroid Build Coastguard Worker }
336*27162e4eSAndroid Build Coastguard Worker
337*27162e4eSAndroid Build Coastguard Worker
compareFiles(FILE * fp0,FILE * fp1,FILE * fpUnc,long uncOffset)338*27162e4eSAndroid Build Coastguard Worker int compareFiles(FILE* fp0, FILE* fp1, FILE* fpUnc, long uncOffset)
339*27162e4eSAndroid Build Coastguard Worker {
340*27162e4eSAndroid Build Coastguard Worker int result = 0;
341*27162e4eSAndroid Build Coastguard Worker long bytesRead = 0;
342*27162e4eSAndroid Build Coastguard Worker long bytesToOffset = -1;
343*27162e4eSAndroid Build Coastguard Worker long b1Size = 1024;
344*27162e4eSAndroid Build Coastguard Worker
345*27162e4eSAndroid Build Coastguard Worker while (result==0) {
346*27162e4eSAndroid Build Coastguard Worker char b1[b1Size];
347*27162e4eSAndroid Build Coastguard Worker size_t r1;
348*27162e4eSAndroid Build Coastguard Worker size_t bytesToRead = sizeof b1;
349*27162e4eSAndroid Build Coastguard Worker if (uncOffset >= 0) {
350*27162e4eSAndroid Build Coastguard Worker bytesToOffset = uncOffset - bytesRead;
351*27162e4eSAndroid Build Coastguard Worker
352*27162e4eSAndroid Build Coastguard Worker /* read remainder to offset */
353*27162e4eSAndroid Build Coastguard Worker if (bytesToOffset < b1Size) {
354*27162e4eSAndroid Build Coastguard Worker bytesToRead = bytesToOffset;
355*27162e4eSAndroid Build Coastguard Worker }
356*27162e4eSAndroid Build Coastguard Worker }
357*27162e4eSAndroid Build Coastguard Worker
358*27162e4eSAndroid Build Coastguard Worker char b0[1024];
359*27162e4eSAndroid Build Coastguard Worker size_t r0;
360*27162e4eSAndroid Build Coastguard Worker if (bytesToOffset <= 0 && fpUnc) {
361*27162e4eSAndroid Build Coastguard Worker bytesToRead = sizeof b1;
362*27162e4eSAndroid Build Coastguard Worker r0 = fread(b0, 1,bytesToRead, fpUnc);
363*27162e4eSAndroid Build Coastguard Worker } else {
364*27162e4eSAndroid Build Coastguard Worker r0 = fread(b0, 1, bytesToRead, fp0);
365*27162e4eSAndroid Build Coastguard Worker }
366*27162e4eSAndroid Build Coastguard Worker
367*27162e4eSAndroid Build Coastguard Worker r1 = fread(b1, 1, r0, fp1);
368*27162e4eSAndroid Build Coastguard Worker
369*27162e4eSAndroid Build Coastguard Worker result = (r0 != r1);
370*27162e4eSAndroid Build Coastguard Worker if (!r0 || !r1) break;
371*27162e4eSAndroid Build Coastguard Worker if (!result) result = memcmp(b0, b1, r0);
372*27162e4eSAndroid Build Coastguard Worker
373*27162e4eSAndroid Build Coastguard Worker bytesRead += r1;
374*27162e4eSAndroid Build Coastguard Worker }
375*27162e4eSAndroid Build Coastguard Worker
376*27162e4eSAndroid Build Coastguard Worker return result;
377*27162e4eSAndroid Build Coastguard Worker }
378*27162e4eSAndroid Build Coastguard Worker
379*27162e4eSAndroid Build Coastguard Worker
main(int argc,char ** argv)380*27162e4eSAndroid Build Coastguard Worker int main(int argc, char **argv) {
381*27162e4eSAndroid Build Coastguard Worker char inpFilename[256] = { 0 };
382*27162e4eSAndroid Build Coastguard Worker char lz4Filename[256] = { 0 };
383*27162e4eSAndroid Build Coastguard Worker char decFilename[256] = { 0 };
384*27162e4eSAndroid Build Coastguard Worker
385*27162e4eSAndroid Build Coastguard Worker int uncOffset = -1;
386*27162e4eSAndroid Build Coastguard Worker char uncFilename[256] = { 0 };
387*27162e4eSAndroid Build Coastguard Worker int opt;
388*27162e4eSAndroid Build Coastguard Worker
389*27162e4eSAndroid Build Coastguard Worker if (argc < 2) {
390*27162e4eSAndroid Build Coastguard Worker printf("Please specify input filename\n");
391*27162e4eSAndroid Build Coastguard Worker return EXIT_FAILURE;
392*27162e4eSAndroid Build Coastguard Worker }
393*27162e4eSAndroid Build Coastguard Worker
394*27162e4eSAndroid Build Coastguard Worker snprintf(inpFilename, 256, "%s", argv[1]);
395*27162e4eSAndroid Build Coastguard Worker snprintf(lz4Filename, 256, "%s.lz4", argv[1]);
396*27162e4eSAndroid Build Coastguard Worker snprintf(decFilename, 256, "%s.lz4.dec", argv[1]);
397*27162e4eSAndroid Build Coastguard Worker
398*27162e4eSAndroid Build Coastguard Worker while ((opt = getopt(argc, argv, "o:d:")) != -1) {
399*27162e4eSAndroid Build Coastguard Worker switch (opt) {
400*27162e4eSAndroid Build Coastguard Worker case 'd':
401*27162e4eSAndroid Build Coastguard Worker snprintf(uncFilename, 256, "%s", optarg);
402*27162e4eSAndroid Build Coastguard Worker break;
403*27162e4eSAndroid Build Coastguard Worker case 'o':
404*27162e4eSAndroid Build Coastguard Worker uncOffset = atoi(optarg);
405*27162e4eSAndroid Build Coastguard Worker break;
406*27162e4eSAndroid Build Coastguard Worker default:
407*27162e4eSAndroid Build Coastguard Worker printf("usage: %s <input file> [-o <offset> -d <file>]\n", argv[0]);
408*27162e4eSAndroid Build Coastguard Worker printf("-o uncompressed data offset\n");
409*27162e4eSAndroid Build Coastguard Worker printf(" inject uncompressed data at this offset into the lz4 file\n");
410*27162e4eSAndroid Build Coastguard Worker printf("-d uncompressed file\n");
411*27162e4eSAndroid Build Coastguard Worker printf(" file to inject without compression into the lz4 file\n");
412*27162e4eSAndroid Build Coastguard Worker return EXIT_FAILURE;
413*27162e4eSAndroid Build Coastguard Worker }
414*27162e4eSAndroid Build Coastguard Worker }
415*27162e4eSAndroid Build Coastguard Worker
416*27162e4eSAndroid Build Coastguard Worker printf("inp = [%s]\n", inpFilename);
417*27162e4eSAndroid Build Coastguard Worker printf("lz4 = [%s]\n", lz4Filename);
418*27162e4eSAndroid Build Coastguard Worker printf("dec = [%s]\n", decFilename);
419*27162e4eSAndroid Build Coastguard Worker if (uncOffset > 0) {
420*27162e4eSAndroid Build Coastguard Worker printf("unc = [%s]\n", uncFilename);
421*27162e4eSAndroid Build Coastguard Worker printf("ofs = [%i]\n", uncOffset);
422*27162e4eSAndroid Build Coastguard Worker }
423*27162e4eSAndroid Build Coastguard Worker
424*27162e4eSAndroid Build Coastguard Worker /* compress */
425*27162e4eSAndroid Build Coastguard Worker { FILE* const inpFp = fopen(inpFilename, "rb");
426*27162e4eSAndroid Build Coastguard Worker FILE* const outFp = fopen(lz4Filename, "wb");
427*27162e4eSAndroid Build Coastguard Worker FILE* const uncFp = fopen(uncFilename, "rb");
428*27162e4eSAndroid Build Coastguard Worker
429*27162e4eSAndroid Build Coastguard Worker printf("compress : %s -> %s\n", inpFilename, lz4Filename);
430*27162e4eSAndroid Build Coastguard Worker compressResult_t const ret = compress_file(
431*27162e4eSAndroid Build Coastguard Worker inpFp, outFp,
432*27162e4eSAndroid Build Coastguard Worker uncFp, uncOffset);
433*27162e4eSAndroid Build Coastguard Worker
434*27162e4eSAndroid Build Coastguard Worker fclose(outFp);
435*27162e4eSAndroid Build Coastguard Worker fclose(inpFp);
436*27162e4eSAndroid Build Coastguard Worker if (uncFp)
437*27162e4eSAndroid Build Coastguard Worker fclose(uncFp);
438*27162e4eSAndroid Build Coastguard Worker
439*27162e4eSAndroid Build Coastguard Worker if (ret.error) {
440*27162e4eSAndroid Build Coastguard Worker printf("compress : failed with code %i\n", ret.error);
441*27162e4eSAndroid Build Coastguard Worker return ret.error;
442*27162e4eSAndroid Build Coastguard Worker }
443*27162e4eSAndroid Build Coastguard Worker printf("%s: %zu → %zu bytes, %.1f%%\n",
444*27162e4eSAndroid Build Coastguard Worker inpFilename,
445*27162e4eSAndroid Build Coastguard Worker (size_t)ret.size_in, (size_t)ret.size_out, /* might overflow is size_t is 32 bits and size_{in,out} > 4 GB */
446*27162e4eSAndroid Build Coastguard Worker (double)ret.size_out / ret.size_in * 100);
447*27162e4eSAndroid Build Coastguard Worker printf("compress : done\n");
448*27162e4eSAndroid Build Coastguard Worker }
449*27162e4eSAndroid Build Coastguard Worker
450*27162e4eSAndroid Build Coastguard Worker /* decompress */
451*27162e4eSAndroid Build Coastguard Worker { FILE* const inpFp = fopen(lz4Filename, "rb");
452*27162e4eSAndroid Build Coastguard Worker FILE* const outFp = fopen(decFilename, "wb");
453*27162e4eSAndroid Build Coastguard Worker
454*27162e4eSAndroid Build Coastguard Worker printf("decompress : %s -> %s\n", lz4Filename, decFilename);
455*27162e4eSAndroid Build Coastguard Worker int const ret = decompress_file(inpFp, outFp);
456*27162e4eSAndroid Build Coastguard Worker
457*27162e4eSAndroid Build Coastguard Worker fclose(outFp);
458*27162e4eSAndroid Build Coastguard Worker fclose(inpFp);
459*27162e4eSAndroid Build Coastguard Worker
460*27162e4eSAndroid Build Coastguard Worker if (ret) {
461*27162e4eSAndroid Build Coastguard Worker printf("decompress : failed with code %i\n", ret);
462*27162e4eSAndroid Build Coastguard Worker return ret;
463*27162e4eSAndroid Build Coastguard Worker }
464*27162e4eSAndroid Build Coastguard Worker printf("decompress : done\n");
465*27162e4eSAndroid Build Coastguard Worker }
466*27162e4eSAndroid Build Coastguard Worker
467*27162e4eSAndroid Build Coastguard Worker /* verify */
468*27162e4eSAndroid Build Coastguard Worker { FILE* const inpFp = fopen(inpFilename, "rb");
469*27162e4eSAndroid Build Coastguard Worker FILE* const decFp = fopen(decFilename, "rb");
470*27162e4eSAndroid Build Coastguard Worker FILE* const uncFp = fopen(uncFilename, "rb");
471*27162e4eSAndroid Build Coastguard Worker
472*27162e4eSAndroid Build Coastguard Worker printf("verify : %s <-> %s\n", inpFilename, decFilename);
473*27162e4eSAndroid Build Coastguard Worker int const cmp = compareFiles(inpFp, decFp,
474*27162e4eSAndroid Build Coastguard Worker uncFp, uncOffset);
475*27162e4eSAndroid Build Coastguard Worker
476*27162e4eSAndroid Build Coastguard Worker fclose(decFp);
477*27162e4eSAndroid Build Coastguard Worker fclose(inpFp);
478*27162e4eSAndroid Build Coastguard Worker if (uncFp)
479*27162e4eSAndroid Build Coastguard Worker fclose(uncFp);
480*27162e4eSAndroid Build Coastguard Worker
481*27162e4eSAndroid Build Coastguard Worker if (cmp) {
482*27162e4eSAndroid Build Coastguard Worker printf("corruption detected : decompressed file differs from original\n");
483*27162e4eSAndroid Build Coastguard Worker return cmp;
484*27162e4eSAndroid Build Coastguard Worker }
485*27162e4eSAndroid Build Coastguard Worker printf("verify : OK\n");
486*27162e4eSAndroid Build Coastguard Worker }
487*27162e4eSAndroid Build Coastguard Worker
488*27162e4eSAndroid Build Coastguard Worker return 0;
489*27162e4eSAndroid Build Coastguard Worker }
490