xref: /plan9/sys/src/cmd/gs/zlib/example.c (revision 593dc095aefb2a85c828727bbfa9da139a49bdf4)
17dd7cddfSDavid du Colombier /* example.c -- usage example of the zlib compression library
2*593dc095SDavid du Colombier  * Copyright (C) 1995-2003 Jean-loup Gailly.
37dd7cddfSDavid du Colombier  * For conditions of distribution and use, see copyright notice in zlib.h
47dd7cddfSDavid du Colombier  */
57dd7cddfSDavid du Colombier 
6*593dc095SDavid du Colombier /* @(#) $Id: example.c,v 1.1.1.1 2005/04/24 21:39:37 giles Exp $ */
77dd7cddfSDavid du Colombier 
87dd7cddfSDavid du Colombier #include <stdio.h>
97dd7cddfSDavid du Colombier #include "zlib.h"
107dd7cddfSDavid du Colombier 
117dd7cddfSDavid du Colombier #ifdef STDC
127dd7cddfSDavid du Colombier #  include <string.h>
137dd7cddfSDavid du Colombier #  include <stdlib.h>
147dd7cddfSDavid du Colombier #else
157dd7cddfSDavid du Colombier    extern void exit  OF((int));
167dd7cddfSDavid du Colombier #endif
177dd7cddfSDavid du Colombier 
18*593dc095SDavid du Colombier #if defined(VMS) || defined(RISCOS)
19*593dc095SDavid du Colombier #  define TESTFILE "foo-gz"
20*593dc095SDavid du Colombier #else
21*593dc095SDavid du Colombier #  define TESTFILE "foo.gz"
22*593dc095SDavid du Colombier #endif
23*593dc095SDavid du Colombier 
247dd7cddfSDavid du Colombier #define CHECK_ERR(err, msg) { \
257dd7cddfSDavid du Colombier     if (err != Z_OK) { \
267dd7cddfSDavid du Colombier         fprintf(stderr, "%s error: %d\n", msg, err); \
277dd7cddfSDavid du Colombier         exit(1); \
287dd7cddfSDavid du Colombier     } \
297dd7cddfSDavid du Colombier }
307dd7cddfSDavid du Colombier 
317dd7cddfSDavid du Colombier const char hello[] = "hello, hello!";
327dd7cddfSDavid du Colombier /* "hello world" would be more standard, but the repeated "hello"
337dd7cddfSDavid du Colombier  * stresses the compression code better, sorry...
347dd7cddfSDavid du Colombier  */
357dd7cddfSDavid du Colombier 
367dd7cddfSDavid du Colombier const char dictionary[] = "hello";
377dd7cddfSDavid du Colombier uLong dictId; /* Adler32 value of the dictionary */
387dd7cddfSDavid du Colombier 
397dd7cddfSDavid du Colombier void test_compress      OF((Byte *compr, uLong comprLen,
407dd7cddfSDavid du Colombier                             Byte *uncompr, uLong uncomprLen));
41*593dc095SDavid du Colombier void test_gzio          OF((const char *fname,
42*593dc095SDavid du Colombier                             Byte *uncompr, uLong uncomprLen));
437dd7cddfSDavid du Colombier void test_deflate       OF((Byte *compr, uLong comprLen));
447dd7cddfSDavid du Colombier void test_inflate       OF((Byte *compr, uLong comprLen,
457dd7cddfSDavid du Colombier                             Byte *uncompr, uLong uncomprLen));
467dd7cddfSDavid du Colombier void test_large_deflate OF((Byte *compr, uLong comprLen,
477dd7cddfSDavid du Colombier                             Byte *uncompr, uLong uncomprLen));
487dd7cddfSDavid du Colombier void test_large_inflate OF((Byte *compr, uLong comprLen,
497dd7cddfSDavid du Colombier                             Byte *uncompr, uLong uncomprLen));
50*593dc095SDavid du Colombier void test_flush         OF((Byte *compr, uLong *comprLen));
517dd7cddfSDavid du Colombier void test_sync          OF((Byte *compr, uLong comprLen,
527dd7cddfSDavid du Colombier                             Byte *uncompr, uLong uncomprLen));
537dd7cddfSDavid du Colombier void test_dict_deflate  OF((Byte *compr, uLong comprLen));
547dd7cddfSDavid du Colombier void test_dict_inflate  OF((Byte *compr, uLong comprLen,
557dd7cddfSDavid du Colombier                             Byte *uncompr, uLong uncomprLen));
567dd7cddfSDavid du Colombier int  main               OF((int argc, char *argv[]));
577dd7cddfSDavid du Colombier 
587dd7cddfSDavid du Colombier /* ===========================================================================
597dd7cddfSDavid du Colombier  * Test compress() and uncompress()
607dd7cddfSDavid du Colombier  */
test_compress(compr,comprLen,uncompr,uncomprLen)617dd7cddfSDavid du Colombier void test_compress(compr, comprLen, uncompr, uncomprLen)
627dd7cddfSDavid du Colombier     Byte *compr, *uncompr;
637dd7cddfSDavid du Colombier     uLong comprLen, uncomprLen;
647dd7cddfSDavid du Colombier {
657dd7cddfSDavid du Colombier     int err;
66*593dc095SDavid du Colombier     uLong len = (uLong)strlen(hello)+1;
677dd7cddfSDavid du Colombier 
687dd7cddfSDavid du Colombier     err = compress(compr, &comprLen, (const Bytef*)hello, len);
697dd7cddfSDavid du Colombier     CHECK_ERR(err, "compress");
707dd7cddfSDavid du Colombier 
717dd7cddfSDavid du Colombier     strcpy((char*)uncompr, "garbage");
727dd7cddfSDavid du Colombier 
737dd7cddfSDavid du Colombier     err = uncompress(uncompr, &uncomprLen, compr, comprLen);
747dd7cddfSDavid du Colombier     CHECK_ERR(err, "uncompress");
757dd7cddfSDavid du Colombier 
767dd7cddfSDavid du Colombier     if (strcmp((char*)uncompr, hello)) {
777dd7cddfSDavid du Colombier         fprintf(stderr, "bad uncompress\n");
78*593dc095SDavid du Colombier         exit(1);
797dd7cddfSDavid du Colombier     } else {
80*593dc095SDavid du Colombier         printf("uncompress(): %s\n", (char *)uncompr);
817dd7cddfSDavid du Colombier     }
827dd7cddfSDavid du Colombier }
837dd7cddfSDavid du Colombier 
847dd7cddfSDavid du Colombier /* ===========================================================================
857dd7cddfSDavid du Colombier  * Test read/write of .gz files
867dd7cddfSDavid du Colombier  */
test_gzio(fname,uncompr,uncomprLen)87*593dc095SDavid du Colombier void test_gzio(fname, uncompr, uncomprLen)
88*593dc095SDavid du Colombier     const char *fname; /* compressed file name */
897dd7cddfSDavid du Colombier     Byte *uncompr;
90*593dc095SDavid du Colombier     uLong uncomprLen;
917dd7cddfSDavid du Colombier {
92*593dc095SDavid du Colombier #ifdef NO_GZCOMPRESS
93*593dc095SDavid du Colombier     fprintf(stderr, "NO_GZCOMPRESS -- gz* functions cannot compress\n");
94*593dc095SDavid du Colombier #else
957dd7cddfSDavid du Colombier     int err;
96*593dc095SDavid du Colombier     int len = (int)strlen(hello)+1;
977dd7cddfSDavid du Colombier     gzFile file;
98*593dc095SDavid du Colombier     z_off_t pos;
997dd7cddfSDavid du Colombier 
100*593dc095SDavid du Colombier     file = gzopen(fname, "wb");
1017dd7cddfSDavid du Colombier     if (file == NULL) {
1027dd7cddfSDavid du Colombier         fprintf(stderr, "gzopen error\n");
1037dd7cddfSDavid du Colombier         exit(1);
1047dd7cddfSDavid du Colombier     }
105*593dc095SDavid du Colombier     gzputc(file, 'h');
106*593dc095SDavid du Colombier     if (gzputs(file, "ello") != 4) {
107*593dc095SDavid du Colombier         fprintf(stderr, "gzputs err: %s\n", gzerror(file, &err));
108*593dc095SDavid du Colombier         exit(1);
1097dd7cddfSDavid du Colombier     }
110*593dc095SDavid du Colombier     if (gzprintf(file, ", %s!", "hello") != 8) {
111*593dc095SDavid du Colombier         fprintf(stderr, "gzprintf err: %s\n", gzerror(file, &err));
112*593dc095SDavid du Colombier         exit(1);
113*593dc095SDavid du Colombier     }
114*593dc095SDavid du Colombier     gzseek(file, 1L, SEEK_CUR); /* add one zero byte */
1157dd7cddfSDavid du Colombier     gzclose(file);
1167dd7cddfSDavid du Colombier 
117*593dc095SDavid du Colombier     file = gzopen(fname, "rb");
1187dd7cddfSDavid du Colombier     if (file == NULL) {
1197dd7cddfSDavid du Colombier         fprintf(stderr, "gzopen error\n");
120*593dc095SDavid du Colombier         exit(1);
1217dd7cddfSDavid du Colombier     }
1227dd7cddfSDavid du Colombier     strcpy((char*)uncompr, "garbage");
1237dd7cddfSDavid du Colombier 
124*593dc095SDavid du Colombier     if (gzread(file, uncompr, (unsigned)uncomprLen) != len) {
1257dd7cddfSDavid du Colombier         fprintf(stderr, "gzread err: %s\n", gzerror(file, &err));
126*593dc095SDavid du Colombier         exit(1);
1277dd7cddfSDavid du Colombier     }
1287dd7cddfSDavid du Colombier     if (strcmp((char*)uncompr, hello)) {
129*593dc095SDavid du Colombier         fprintf(stderr, "bad gzread: %s\n", (char*)uncompr);
130*593dc095SDavid du Colombier         exit(1);
1317dd7cddfSDavid du Colombier     } else {
132*593dc095SDavid du Colombier         printf("gzread(): %s\n", (char*)uncompr);
1337dd7cddfSDavid du Colombier     }
134*593dc095SDavid du Colombier 
135*593dc095SDavid du Colombier     pos = gzseek(file, -8L, SEEK_CUR);
136*593dc095SDavid du Colombier     if (pos != 6 || gztell(file) != pos) {
137*593dc095SDavid du Colombier         fprintf(stderr, "gzseek error, pos=%ld, gztell=%ld\n",
138*593dc095SDavid du Colombier                 (long)pos, (long)gztell(file));
139*593dc095SDavid du Colombier         exit(1);
140*593dc095SDavid du Colombier     }
141*593dc095SDavid du Colombier 
142*593dc095SDavid du Colombier     if (gzgetc(file) != ' ') {
143*593dc095SDavid du Colombier         fprintf(stderr, "gzgetc error\n");
144*593dc095SDavid du Colombier         exit(1);
145*593dc095SDavid du Colombier     }
146*593dc095SDavid du Colombier 
147*593dc095SDavid du Colombier     if (gzungetc(' ', file) != ' ') {
148*593dc095SDavid du Colombier         fprintf(stderr, "gzungetc error\n");
149*593dc095SDavid du Colombier         exit(1);
150*593dc095SDavid du Colombier     }
151*593dc095SDavid du Colombier 
152*593dc095SDavid du Colombier     gzgets(file, (char*)uncompr, (int)uncomprLen);
153*593dc095SDavid du Colombier     if (strlen((char*)uncompr) != 7) { /* " hello!" */
154*593dc095SDavid du Colombier         fprintf(stderr, "gzgets err after gzseek: %s\n", gzerror(file, &err));
155*593dc095SDavid du Colombier         exit(1);
156*593dc095SDavid du Colombier     }
157*593dc095SDavid du Colombier     if (strcmp((char*)uncompr, hello + 6)) {
158*593dc095SDavid du Colombier         fprintf(stderr, "bad gzgets after gzseek\n");
159*593dc095SDavid du Colombier         exit(1);
160*593dc095SDavid du Colombier     } else {
161*593dc095SDavid du Colombier         printf("gzgets() after gzseek: %s\n", (char*)uncompr);
162*593dc095SDavid du Colombier     }
163*593dc095SDavid du Colombier 
164*593dc095SDavid du Colombier     gzclose(file);
165*593dc095SDavid du Colombier #endif
1667dd7cddfSDavid du Colombier }
1677dd7cddfSDavid du Colombier 
1687dd7cddfSDavid du Colombier /* ===========================================================================
1697dd7cddfSDavid du Colombier  * Test deflate() with small buffers
1707dd7cddfSDavid du Colombier  */
test_deflate(compr,comprLen)1717dd7cddfSDavid du Colombier void test_deflate(compr, comprLen)
1727dd7cddfSDavid du Colombier     Byte *compr;
1737dd7cddfSDavid du Colombier     uLong comprLen;
1747dd7cddfSDavid du Colombier {
1757dd7cddfSDavid du Colombier     z_stream c_stream; /* compression stream */
1767dd7cddfSDavid du Colombier     int err;
177*593dc095SDavid du Colombier     uLong len = (uLong)strlen(hello)+1;
1787dd7cddfSDavid du Colombier 
1797dd7cddfSDavid du Colombier     c_stream.zalloc = (alloc_func)0;
1807dd7cddfSDavid du Colombier     c_stream.zfree = (free_func)0;
1817dd7cddfSDavid du Colombier     c_stream.opaque = (voidpf)0;
1827dd7cddfSDavid du Colombier 
1837dd7cddfSDavid du Colombier     err = deflateInit(&c_stream, Z_DEFAULT_COMPRESSION);
1847dd7cddfSDavid du Colombier     CHECK_ERR(err, "deflateInit");
1857dd7cddfSDavid du Colombier 
1867dd7cddfSDavid du Colombier     c_stream.next_in  = (Bytef*)hello;
1877dd7cddfSDavid du Colombier     c_stream.next_out = compr;
1887dd7cddfSDavid du Colombier 
189*593dc095SDavid du Colombier     while (c_stream.total_in != len && c_stream.total_out < comprLen) {
1907dd7cddfSDavid du Colombier         c_stream.avail_in = c_stream.avail_out = 1; /* force small buffers */
1917dd7cddfSDavid du Colombier         err = deflate(&c_stream, Z_NO_FLUSH);
1927dd7cddfSDavid du Colombier         CHECK_ERR(err, "deflate");
1937dd7cddfSDavid du Colombier     }
1947dd7cddfSDavid du Colombier     /* Finish the stream, still forcing small buffers: */
1957dd7cddfSDavid du Colombier     for (;;) {
1967dd7cddfSDavid du Colombier         c_stream.avail_out = 1;
1977dd7cddfSDavid du Colombier         err = deflate(&c_stream, Z_FINISH);
1987dd7cddfSDavid du Colombier         if (err == Z_STREAM_END) break;
1997dd7cddfSDavid du Colombier         CHECK_ERR(err, "deflate");
2007dd7cddfSDavid du Colombier     }
2017dd7cddfSDavid du Colombier 
2027dd7cddfSDavid du Colombier     err = deflateEnd(&c_stream);
2037dd7cddfSDavid du Colombier     CHECK_ERR(err, "deflateEnd");
2047dd7cddfSDavid du Colombier }
2057dd7cddfSDavid du Colombier 
2067dd7cddfSDavid du Colombier /* ===========================================================================
2077dd7cddfSDavid du Colombier  * Test inflate() with small buffers
2087dd7cddfSDavid du Colombier  */
test_inflate(compr,comprLen,uncompr,uncomprLen)2097dd7cddfSDavid du Colombier void test_inflate(compr, comprLen, uncompr, uncomprLen)
2107dd7cddfSDavid du Colombier     Byte *compr, *uncompr;
2117dd7cddfSDavid du Colombier     uLong comprLen, uncomprLen;
2127dd7cddfSDavid du Colombier {
2137dd7cddfSDavid du Colombier     int err;
2147dd7cddfSDavid du Colombier     z_stream d_stream; /* decompression stream */
2157dd7cddfSDavid du Colombier 
2167dd7cddfSDavid du Colombier     strcpy((char*)uncompr, "garbage");
2177dd7cddfSDavid du Colombier 
2187dd7cddfSDavid du Colombier     d_stream.zalloc = (alloc_func)0;
2197dd7cddfSDavid du Colombier     d_stream.zfree = (free_func)0;
2207dd7cddfSDavid du Colombier     d_stream.opaque = (voidpf)0;
2217dd7cddfSDavid du Colombier 
222*593dc095SDavid du Colombier     d_stream.next_in  = compr;
223*593dc095SDavid du Colombier     d_stream.avail_in = 0;
224*593dc095SDavid du Colombier     d_stream.next_out = uncompr;
225*593dc095SDavid du Colombier 
2267dd7cddfSDavid du Colombier     err = inflateInit(&d_stream);
2277dd7cddfSDavid du Colombier     CHECK_ERR(err, "inflateInit");
2287dd7cddfSDavid du Colombier 
2297dd7cddfSDavid du Colombier     while (d_stream.total_out < uncomprLen && d_stream.total_in < comprLen) {
2307dd7cddfSDavid du Colombier         d_stream.avail_in = d_stream.avail_out = 1; /* force small buffers */
2317dd7cddfSDavid du Colombier         err = inflate(&d_stream, Z_NO_FLUSH);
2327dd7cddfSDavid du Colombier         if (err == Z_STREAM_END) break;
2337dd7cddfSDavid du Colombier         CHECK_ERR(err, "inflate");
2347dd7cddfSDavid du Colombier     }
2357dd7cddfSDavid du Colombier 
2367dd7cddfSDavid du Colombier     err = inflateEnd(&d_stream);
2377dd7cddfSDavid du Colombier     CHECK_ERR(err, "inflateEnd");
2387dd7cddfSDavid du Colombier 
2397dd7cddfSDavid du Colombier     if (strcmp((char*)uncompr, hello)) {
2407dd7cddfSDavid du Colombier         fprintf(stderr, "bad inflate\n");
241*593dc095SDavid du Colombier         exit(1);
2427dd7cddfSDavid du Colombier     } else {
243*593dc095SDavid du Colombier         printf("inflate(): %s\n", (char *)uncompr);
2447dd7cddfSDavid du Colombier     }
2457dd7cddfSDavid du Colombier }
2467dd7cddfSDavid du Colombier 
2477dd7cddfSDavid du Colombier /* ===========================================================================
2487dd7cddfSDavid du Colombier  * Test deflate() with large buffers and dynamic change of compression level
2497dd7cddfSDavid du Colombier  */
test_large_deflate(compr,comprLen,uncompr,uncomprLen)2507dd7cddfSDavid du Colombier void test_large_deflate(compr, comprLen, uncompr, uncomprLen)
2517dd7cddfSDavid du Colombier     Byte *compr, *uncompr;
2527dd7cddfSDavid du Colombier     uLong comprLen, uncomprLen;
2537dd7cddfSDavid du Colombier {
2547dd7cddfSDavid du Colombier     z_stream c_stream; /* compression stream */
2557dd7cddfSDavid du Colombier     int err;
2567dd7cddfSDavid du Colombier 
2577dd7cddfSDavid du Colombier     c_stream.zalloc = (alloc_func)0;
2587dd7cddfSDavid du Colombier     c_stream.zfree = (free_func)0;
2597dd7cddfSDavid du Colombier     c_stream.opaque = (voidpf)0;
2607dd7cddfSDavid du Colombier 
2617dd7cddfSDavid du Colombier     err = deflateInit(&c_stream, Z_BEST_SPEED);
2627dd7cddfSDavid du Colombier     CHECK_ERR(err, "deflateInit");
2637dd7cddfSDavid du Colombier 
2647dd7cddfSDavid du Colombier     c_stream.next_out = compr;
2657dd7cddfSDavid du Colombier     c_stream.avail_out = (uInt)comprLen;
2667dd7cddfSDavid du Colombier 
2677dd7cddfSDavid du Colombier     /* At this point, uncompr is still mostly zeroes, so it should compress
2687dd7cddfSDavid du Colombier      * very well:
2697dd7cddfSDavid du Colombier      */
2707dd7cddfSDavid du Colombier     c_stream.next_in = uncompr;
2717dd7cddfSDavid du Colombier     c_stream.avail_in = (uInt)uncomprLen;
2727dd7cddfSDavid du Colombier     err = deflate(&c_stream, Z_NO_FLUSH);
2737dd7cddfSDavid du Colombier     CHECK_ERR(err, "deflate");
2747dd7cddfSDavid du Colombier     if (c_stream.avail_in != 0) {
2757dd7cddfSDavid du Colombier         fprintf(stderr, "deflate not greedy\n");
276*593dc095SDavid du Colombier         exit(1);
2777dd7cddfSDavid du Colombier     }
2787dd7cddfSDavid du Colombier 
2797dd7cddfSDavid du Colombier     /* Feed in already compressed data and switch to no compression: */
2807dd7cddfSDavid du Colombier     deflateParams(&c_stream, Z_NO_COMPRESSION, Z_DEFAULT_STRATEGY);
2817dd7cddfSDavid du Colombier     c_stream.next_in = compr;
2827dd7cddfSDavid du Colombier     c_stream.avail_in = (uInt)comprLen/2;
2837dd7cddfSDavid du Colombier     err = deflate(&c_stream, Z_NO_FLUSH);
2847dd7cddfSDavid du Colombier     CHECK_ERR(err, "deflate");
2857dd7cddfSDavid du Colombier 
2867dd7cddfSDavid du Colombier     /* Switch back to compressing mode: */
2877dd7cddfSDavid du Colombier     deflateParams(&c_stream, Z_BEST_COMPRESSION, Z_FILTERED);
2887dd7cddfSDavid du Colombier     c_stream.next_in = uncompr;
2897dd7cddfSDavid du Colombier     c_stream.avail_in = (uInt)uncomprLen;
2907dd7cddfSDavid du Colombier     err = deflate(&c_stream, Z_NO_FLUSH);
2917dd7cddfSDavid du Colombier     CHECK_ERR(err, "deflate");
2927dd7cddfSDavid du Colombier 
2937dd7cddfSDavid du Colombier     err = deflate(&c_stream, Z_FINISH);
2947dd7cddfSDavid du Colombier     if (err != Z_STREAM_END) {
2957dd7cddfSDavid du Colombier         fprintf(stderr, "deflate should report Z_STREAM_END\n");
296*593dc095SDavid du Colombier         exit(1);
2977dd7cddfSDavid du Colombier     }
2987dd7cddfSDavid du Colombier     err = deflateEnd(&c_stream);
2997dd7cddfSDavid du Colombier     CHECK_ERR(err, "deflateEnd");
3007dd7cddfSDavid du Colombier }
3017dd7cddfSDavid du Colombier 
3027dd7cddfSDavid du Colombier /* ===========================================================================
3037dd7cddfSDavid du Colombier  * Test inflate() with large buffers
3047dd7cddfSDavid du Colombier  */
test_large_inflate(compr,comprLen,uncompr,uncomprLen)3057dd7cddfSDavid du Colombier void test_large_inflate(compr, comprLen, uncompr, uncomprLen)
3067dd7cddfSDavid du Colombier     Byte *compr, *uncompr;
3077dd7cddfSDavid du Colombier     uLong comprLen, uncomprLen;
3087dd7cddfSDavid du Colombier {
3097dd7cddfSDavid du Colombier     int err;
3107dd7cddfSDavid du Colombier     z_stream d_stream; /* decompression stream */
3117dd7cddfSDavid du Colombier 
3127dd7cddfSDavid du Colombier     strcpy((char*)uncompr, "garbage");
3137dd7cddfSDavid du Colombier 
3147dd7cddfSDavid du Colombier     d_stream.zalloc = (alloc_func)0;
3157dd7cddfSDavid du Colombier     d_stream.zfree = (free_func)0;
3167dd7cddfSDavid du Colombier     d_stream.opaque = (voidpf)0;
3177dd7cddfSDavid du Colombier 
3187dd7cddfSDavid du Colombier     d_stream.next_in  = compr;
3197dd7cddfSDavid du Colombier     d_stream.avail_in = (uInt)comprLen;
3207dd7cddfSDavid du Colombier 
321*593dc095SDavid du Colombier     err = inflateInit(&d_stream);
322*593dc095SDavid du Colombier     CHECK_ERR(err, "inflateInit");
323*593dc095SDavid du Colombier 
3247dd7cddfSDavid du Colombier     for (;;) {
3257dd7cddfSDavid du Colombier         d_stream.next_out = uncompr;            /* discard the output */
3267dd7cddfSDavid du Colombier         d_stream.avail_out = (uInt)uncomprLen;
3277dd7cddfSDavid du Colombier         err = inflate(&d_stream, Z_NO_FLUSH);
3287dd7cddfSDavid du Colombier         if (err == Z_STREAM_END) break;
3297dd7cddfSDavid du Colombier         CHECK_ERR(err, "large inflate");
3307dd7cddfSDavid du Colombier     }
3317dd7cddfSDavid du Colombier 
3327dd7cddfSDavid du Colombier     err = inflateEnd(&d_stream);
3337dd7cddfSDavid du Colombier     CHECK_ERR(err, "inflateEnd");
3347dd7cddfSDavid du Colombier 
3357dd7cddfSDavid du Colombier     if (d_stream.total_out != 2*uncomprLen + comprLen/2) {
3367dd7cddfSDavid du Colombier         fprintf(stderr, "bad large inflate: %ld\n", d_stream.total_out);
337*593dc095SDavid du Colombier         exit(1);
3387dd7cddfSDavid du Colombier     } else {
3397dd7cddfSDavid du Colombier         printf("large_inflate(): OK\n");
3407dd7cddfSDavid du Colombier     }
3417dd7cddfSDavid du Colombier }
3427dd7cddfSDavid du Colombier 
3437dd7cddfSDavid du Colombier /* ===========================================================================
3447dd7cddfSDavid du Colombier  * Test deflate() with full flush
3457dd7cddfSDavid du Colombier  */
test_flush(compr,comprLen)3467dd7cddfSDavid du Colombier void test_flush(compr, comprLen)
3477dd7cddfSDavid du Colombier     Byte *compr;
348*593dc095SDavid du Colombier     uLong *comprLen;
3497dd7cddfSDavid du Colombier {
3507dd7cddfSDavid du Colombier     z_stream c_stream; /* compression stream */
3517dd7cddfSDavid du Colombier     int err;
352*593dc095SDavid du Colombier     uInt len = (uInt)strlen(hello)+1;
3537dd7cddfSDavid du Colombier 
3547dd7cddfSDavid du Colombier     c_stream.zalloc = (alloc_func)0;
3557dd7cddfSDavid du Colombier     c_stream.zfree = (free_func)0;
3567dd7cddfSDavid du Colombier     c_stream.opaque = (voidpf)0;
3577dd7cddfSDavid du Colombier 
3587dd7cddfSDavid du Colombier     err = deflateInit(&c_stream, Z_DEFAULT_COMPRESSION);
3597dd7cddfSDavid du Colombier     CHECK_ERR(err, "deflateInit");
3607dd7cddfSDavid du Colombier 
3617dd7cddfSDavid du Colombier     c_stream.next_in  = (Bytef*)hello;
3627dd7cddfSDavid du Colombier     c_stream.next_out = compr;
3637dd7cddfSDavid du Colombier     c_stream.avail_in = 3;
364*593dc095SDavid du Colombier     c_stream.avail_out = (uInt)*comprLen;
3657dd7cddfSDavid du Colombier     err = deflate(&c_stream, Z_FULL_FLUSH);
3667dd7cddfSDavid du Colombier     CHECK_ERR(err, "deflate");
3677dd7cddfSDavid du Colombier 
3687dd7cddfSDavid du Colombier     compr[3]++; /* force an error in first compressed block */
3697dd7cddfSDavid du Colombier     c_stream.avail_in = len - 3;
3707dd7cddfSDavid du Colombier 
3717dd7cddfSDavid du Colombier     err = deflate(&c_stream, Z_FINISH);
3727dd7cddfSDavid du Colombier     if (err != Z_STREAM_END) {
3737dd7cddfSDavid du Colombier         CHECK_ERR(err, "deflate");
3747dd7cddfSDavid du Colombier     }
3757dd7cddfSDavid du Colombier     err = deflateEnd(&c_stream);
3767dd7cddfSDavid du Colombier     CHECK_ERR(err, "deflateEnd");
377*593dc095SDavid du Colombier 
378*593dc095SDavid du Colombier     *comprLen = c_stream.total_out;
3797dd7cddfSDavid du Colombier }
3807dd7cddfSDavid du Colombier 
3817dd7cddfSDavid du Colombier /* ===========================================================================
3827dd7cddfSDavid du Colombier  * Test inflateSync()
3837dd7cddfSDavid du Colombier  */
test_sync(compr,comprLen,uncompr,uncomprLen)3847dd7cddfSDavid du Colombier void test_sync(compr, comprLen, uncompr, uncomprLen)
3857dd7cddfSDavid du Colombier     Byte *compr, *uncompr;
3867dd7cddfSDavid du Colombier     uLong comprLen, uncomprLen;
3877dd7cddfSDavid du Colombier {
3887dd7cddfSDavid du Colombier     int err;
3897dd7cddfSDavid du Colombier     z_stream d_stream; /* decompression stream */
3907dd7cddfSDavid du Colombier 
3917dd7cddfSDavid du Colombier     strcpy((char*)uncompr, "garbage");
3927dd7cddfSDavid du Colombier 
3937dd7cddfSDavid du Colombier     d_stream.zalloc = (alloc_func)0;
3947dd7cddfSDavid du Colombier     d_stream.zfree = (free_func)0;
3957dd7cddfSDavid du Colombier     d_stream.opaque = (voidpf)0;
3967dd7cddfSDavid du Colombier 
397*593dc095SDavid du Colombier     d_stream.next_in  = compr;
398*593dc095SDavid du Colombier     d_stream.avail_in = 2; /* just read the zlib header */
399*593dc095SDavid du Colombier 
4007dd7cddfSDavid du Colombier     err = inflateInit(&d_stream);
4017dd7cddfSDavid du Colombier     CHECK_ERR(err, "inflateInit");
4027dd7cddfSDavid du Colombier 
4037dd7cddfSDavid du Colombier     d_stream.next_out = uncompr;
4047dd7cddfSDavid du Colombier     d_stream.avail_out = (uInt)uncomprLen;
4057dd7cddfSDavid du Colombier 
4067dd7cddfSDavid du Colombier     inflate(&d_stream, Z_NO_FLUSH);
4077dd7cddfSDavid du Colombier     CHECK_ERR(err, "inflate");
4087dd7cddfSDavid du Colombier 
4097dd7cddfSDavid du Colombier     d_stream.avail_in = (uInt)comprLen-2;   /* read all compressed data */
4107dd7cddfSDavid du Colombier     err = inflateSync(&d_stream);           /* but skip the damaged part */
4117dd7cddfSDavid du Colombier     CHECK_ERR(err, "inflateSync");
4127dd7cddfSDavid du Colombier 
4137dd7cddfSDavid du Colombier     err = inflate(&d_stream, Z_FINISH);
4147dd7cddfSDavid du Colombier     if (err != Z_DATA_ERROR) {
4157dd7cddfSDavid du Colombier         fprintf(stderr, "inflate should report DATA_ERROR\n");
4167dd7cddfSDavid du Colombier         /* Because of incorrect adler32 */
417*593dc095SDavid du Colombier         exit(1);
4187dd7cddfSDavid du Colombier     }
4197dd7cddfSDavid du Colombier     err = inflateEnd(&d_stream);
4207dd7cddfSDavid du Colombier     CHECK_ERR(err, "inflateEnd");
4217dd7cddfSDavid du Colombier 
422*593dc095SDavid du Colombier     printf("after inflateSync(): hel%s\n", (char *)uncompr);
4237dd7cddfSDavid du Colombier }
4247dd7cddfSDavid du Colombier 
4257dd7cddfSDavid du Colombier /* ===========================================================================
4267dd7cddfSDavid du Colombier  * Test deflate() with preset dictionary
4277dd7cddfSDavid du Colombier  */
test_dict_deflate(compr,comprLen)4287dd7cddfSDavid du Colombier void test_dict_deflate(compr, comprLen)
4297dd7cddfSDavid du Colombier     Byte *compr;
4307dd7cddfSDavid du Colombier     uLong comprLen;
4317dd7cddfSDavid du Colombier {
4327dd7cddfSDavid du Colombier     z_stream c_stream; /* compression stream */
4337dd7cddfSDavid du Colombier     int err;
4347dd7cddfSDavid du Colombier 
4357dd7cddfSDavid du Colombier     c_stream.zalloc = (alloc_func)0;
4367dd7cddfSDavid du Colombier     c_stream.zfree = (free_func)0;
4377dd7cddfSDavid du Colombier     c_stream.opaque = (voidpf)0;
4387dd7cddfSDavid du Colombier 
4397dd7cddfSDavid du Colombier     err = deflateInit(&c_stream, Z_BEST_COMPRESSION);
4407dd7cddfSDavid du Colombier     CHECK_ERR(err, "deflateInit");
4417dd7cddfSDavid du Colombier 
4427dd7cddfSDavid du Colombier     err = deflateSetDictionary(&c_stream,
4437dd7cddfSDavid du Colombier                                (const Bytef*)dictionary, sizeof(dictionary));
4447dd7cddfSDavid du Colombier     CHECK_ERR(err, "deflateSetDictionary");
4457dd7cddfSDavid du Colombier 
4467dd7cddfSDavid du Colombier     dictId = c_stream.adler;
4477dd7cddfSDavid du Colombier     c_stream.next_out = compr;
4487dd7cddfSDavid du Colombier     c_stream.avail_out = (uInt)comprLen;
4497dd7cddfSDavid du Colombier 
4507dd7cddfSDavid du Colombier     c_stream.next_in = (Bytef*)hello;
4517dd7cddfSDavid du Colombier     c_stream.avail_in = (uInt)strlen(hello)+1;
4527dd7cddfSDavid du Colombier 
4537dd7cddfSDavid du Colombier     err = deflate(&c_stream, Z_FINISH);
4547dd7cddfSDavid du Colombier     if (err != Z_STREAM_END) {
4557dd7cddfSDavid du Colombier         fprintf(stderr, "deflate should report Z_STREAM_END\n");
456*593dc095SDavid du Colombier         exit(1);
4577dd7cddfSDavid du Colombier     }
4587dd7cddfSDavid du Colombier     err = deflateEnd(&c_stream);
4597dd7cddfSDavid du Colombier     CHECK_ERR(err, "deflateEnd");
4607dd7cddfSDavid du Colombier }
4617dd7cddfSDavid du Colombier 
4627dd7cddfSDavid du Colombier /* ===========================================================================
4637dd7cddfSDavid du Colombier  * Test inflate() with a preset dictionary
4647dd7cddfSDavid du Colombier  */
test_dict_inflate(compr,comprLen,uncompr,uncomprLen)4657dd7cddfSDavid du Colombier void test_dict_inflate(compr, comprLen, uncompr, uncomprLen)
4667dd7cddfSDavid du Colombier     Byte *compr, *uncompr;
4677dd7cddfSDavid du Colombier     uLong comprLen, uncomprLen;
4687dd7cddfSDavid du Colombier {
4697dd7cddfSDavid du Colombier     int err;
4707dd7cddfSDavid du Colombier     z_stream d_stream; /* decompression stream */
4717dd7cddfSDavid du Colombier 
4727dd7cddfSDavid du Colombier     strcpy((char*)uncompr, "garbage");
4737dd7cddfSDavid du Colombier 
4747dd7cddfSDavid du Colombier     d_stream.zalloc = (alloc_func)0;
4757dd7cddfSDavid du Colombier     d_stream.zfree = (free_func)0;
4767dd7cddfSDavid du Colombier     d_stream.opaque = (voidpf)0;
4777dd7cddfSDavid du Colombier 
4787dd7cddfSDavid du Colombier     d_stream.next_in  = compr;
4797dd7cddfSDavid du Colombier     d_stream.avail_in = (uInt)comprLen;
4807dd7cddfSDavid du Colombier 
481*593dc095SDavid du Colombier     err = inflateInit(&d_stream);
482*593dc095SDavid du Colombier     CHECK_ERR(err, "inflateInit");
483*593dc095SDavid du Colombier 
4847dd7cddfSDavid du Colombier     d_stream.next_out = uncompr;
4857dd7cddfSDavid du Colombier     d_stream.avail_out = (uInt)uncomprLen;
4867dd7cddfSDavid du Colombier 
4877dd7cddfSDavid du Colombier     for (;;) {
4887dd7cddfSDavid du Colombier         err = inflate(&d_stream, Z_NO_FLUSH);
4897dd7cddfSDavid du Colombier         if (err == Z_STREAM_END) break;
4907dd7cddfSDavid du Colombier         if (err == Z_NEED_DICT) {
4917dd7cddfSDavid du Colombier             if (d_stream.adler != dictId) {
4927dd7cddfSDavid du Colombier                 fprintf(stderr, "unexpected dictionary");
4937dd7cddfSDavid du Colombier                 exit(1);
4947dd7cddfSDavid du Colombier             }
4957dd7cddfSDavid du Colombier             err = inflateSetDictionary(&d_stream, (const Bytef*)dictionary,
4967dd7cddfSDavid du Colombier                                        sizeof(dictionary));
4977dd7cddfSDavid du Colombier         }
4987dd7cddfSDavid du Colombier         CHECK_ERR(err, "inflate with dict");
4997dd7cddfSDavid du Colombier     }
5007dd7cddfSDavid du Colombier 
5017dd7cddfSDavid du Colombier     err = inflateEnd(&d_stream);
5027dd7cddfSDavid du Colombier     CHECK_ERR(err, "inflateEnd");
5037dd7cddfSDavid du Colombier 
5047dd7cddfSDavid du Colombier     if (strcmp((char*)uncompr, hello)) {
5057dd7cddfSDavid du Colombier         fprintf(stderr, "bad inflate with dict\n");
506*593dc095SDavid du Colombier         exit(1);
5077dd7cddfSDavid du Colombier     } else {
508*593dc095SDavid du Colombier         printf("inflate with dictionary: %s\n", (char *)uncompr);
5097dd7cddfSDavid du Colombier     }
5107dd7cddfSDavid du Colombier }
5117dd7cddfSDavid du Colombier 
5127dd7cddfSDavid du Colombier /* ===========================================================================
5137dd7cddfSDavid du Colombier  * Usage:  example [output.gz  [input.gz]]
5147dd7cddfSDavid du Colombier  */
5157dd7cddfSDavid du Colombier 
main(argc,argv)5167dd7cddfSDavid du Colombier int main(argc, argv)
5177dd7cddfSDavid du Colombier     int argc;
5187dd7cddfSDavid du Colombier     char *argv[];
5197dd7cddfSDavid du Colombier {
5207dd7cddfSDavid du Colombier     Byte *compr, *uncompr;
5217dd7cddfSDavid du Colombier     uLong comprLen = 10000*sizeof(int); /* don't overflow on MSDOS */
5227dd7cddfSDavid du Colombier     uLong uncomprLen = comprLen;
523*593dc095SDavid du Colombier     static const char* myVersion = ZLIB_VERSION;
5247dd7cddfSDavid du Colombier 
525*593dc095SDavid du Colombier     if (zlibVersion()[0] != myVersion[0]) {
5267dd7cddfSDavid du Colombier         fprintf(stderr, "incompatible zlib version\n");
5277dd7cddfSDavid du Colombier         exit(1);
5287dd7cddfSDavid du Colombier 
5297dd7cddfSDavid du Colombier     } else if (strcmp(zlibVersion(), ZLIB_VERSION) != 0) {
5307dd7cddfSDavid du Colombier         fprintf(stderr, "warning: different zlib version\n");
5317dd7cddfSDavid du Colombier     }
5327dd7cddfSDavid du Colombier 
533*593dc095SDavid du Colombier     printf("zlib version %s = 0x%04x, compile flags = 0x%lx\n",
534*593dc095SDavid du Colombier             ZLIB_VERSION, ZLIB_VERNUM, zlibCompileFlags());
535*593dc095SDavid du Colombier 
5367dd7cddfSDavid du Colombier     compr    = (Byte*)calloc((uInt)comprLen, 1);
5377dd7cddfSDavid du Colombier     uncompr  = (Byte*)calloc((uInt)uncomprLen, 1);
5387dd7cddfSDavid du Colombier     /* compr and uncompr are cleared to avoid reading uninitialized
5397dd7cddfSDavid du Colombier      * data and to ensure that uncompr compresses well.
5407dd7cddfSDavid du Colombier      */
5417dd7cddfSDavid du Colombier     if (compr == Z_NULL || uncompr == Z_NULL) {
5427dd7cddfSDavid du Colombier         printf("out of memory\n");
5437dd7cddfSDavid du Colombier         exit(1);
5447dd7cddfSDavid du Colombier     }
5457dd7cddfSDavid du Colombier     test_compress(compr, comprLen, uncompr, uncomprLen);
5467dd7cddfSDavid du Colombier 
547*593dc095SDavid du Colombier     test_gzio((argc > 1 ? argv[1] : TESTFILE),
548*593dc095SDavid du Colombier               uncompr, uncomprLen);
5497dd7cddfSDavid du Colombier 
5507dd7cddfSDavid du Colombier     test_deflate(compr, comprLen);
5517dd7cddfSDavid du Colombier     test_inflate(compr, comprLen, uncompr, uncomprLen);
5527dd7cddfSDavid du Colombier 
5537dd7cddfSDavid du Colombier     test_large_deflate(compr, comprLen, uncompr, uncomprLen);
5547dd7cddfSDavid du Colombier     test_large_inflate(compr, comprLen, uncompr, uncomprLen);
5557dd7cddfSDavid du Colombier 
556*593dc095SDavid du Colombier     test_flush(compr, &comprLen);
5577dd7cddfSDavid du Colombier     test_sync(compr, comprLen, uncompr, uncomprLen);
558*593dc095SDavid du Colombier     comprLen = uncomprLen;
5597dd7cddfSDavid du Colombier 
5607dd7cddfSDavid du Colombier     test_dict_deflate(compr, comprLen);
5617dd7cddfSDavid du Colombier     test_dict_inflate(compr, comprLen, uncompr, uncomprLen);
5627dd7cddfSDavid du Colombier 
563*593dc095SDavid du Colombier     free(compr);
564*593dc095SDavid du Colombier     free(uncompr);
565*593dc095SDavid du Colombier 
566*593dc095SDavid du Colombier     return 0;
5677dd7cddfSDavid du Colombier }
568