xref: /aosp_15_r20/external/lz4/examples/frameCompress.c (revision 27162e4e17433d5aa7cb38e7b6a433a09405fc7f)
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