/src/zlib-ng/test/fuzz/fuzzer_example_dict.c
Line | Count | Source (jump to first uncovered line) |
1 | | #include <stdio.h> |
2 | | #include <assert.h> |
3 | | |
4 | | #include "zbuild.h" |
5 | | #ifdef ZLIB_COMPAT |
6 | | # include "zlib.h" |
7 | | #else |
8 | | # include "zlib-ng.h" |
9 | | #endif |
10 | | |
11 | 40.0k | #define CHECK_ERR(err, msg) { \ |
12 | 40.0k | if (err != Z_OK) { \ |
13 | 0 | fprintf(stderr, "%s error: %d\n", msg, err); \ |
14 | 0 | exit(1); \ |
15 | 0 | } \ |
16 | 40.0k | } |
17 | | |
18 | | static const uint8_t *data; |
19 | | static size_t dataLen; |
20 | | static alloc_func zalloc = NULL; |
21 | | static free_func zfree = NULL; |
22 | | static unsigned int dictionaryLen = 0; |
23 | | static unsigned long dictId; /* Adler32 value of the dictionary */ |
24 | | |
25 | | /* =========================================================================== |
26 | | * Test deflate() with preset dictionary |
27 | | */ |
28 | 6.72k | void test_dict_deflate(unsigned char **compr, size_t *comprLen) { |
29 | 6.72k | PREFIX3(stream) c_stream; /* compression stream */ |
30 | 6.72k | int err; |
31 | 6.72k | int level = data[0] % 11 - 1; /* [-1..9] |
32 | | compression levels |
33 | | #define Z_NO_COMPRESSION 0 |
34 | | #define Z_BEST_SPEED 1 |
35 | | #define Z_BEST_COMPRESSION 9 |
36 | | #define Z_DEFAULT_COMPRESSION (-1) */ |
37 | | |
38 | 6.72k | int method = Z_DEFLATED; /* The deflate compression method (the only one |
39 | | supported in this version) */ |
40 | 6.72k | int windowBits = 8 + data[(dataLen > 1) ? 1:0] % 8; /* The windowBits parameter is the base |
41 | | two logarithm of the window size (the size of the history buffer). It |
42 | | should be in the range 8..15 for this version of the library. */ |
43 | 6.72k | int memLevel = 1 + data[(dataLen > 2) ? 2:0] % 9; /* memLevel=1 uses minimum memory but is |
44 | | slow and reduces compression ratio; memLevel=9 uses maximum memory for |
45 | | optimal speed. */ |
46 | 6.72k | int strategy = data[(dataLen > 3) ? 3:0] % 5; /* [0..4] |
47 | | #define Z_FILTERED 1 |
48 | | #define Z_HUFFMAN_ONLY 2 |
49 | | #define Z_RLE 3 |
50 | | #define Z_FIXED 4 |
51 | | #define Z_DEFAULT_STRATEGY 0 */ |
52 | | |
53 | | /* deflate would fail for no-compression or for speed levels. */ |
54 | 6.72k | if (level == 0 || level == 1) |
55 | 1.37k | level = -1; |
56 | | |
57 | 6.72k | c_stream.zalloc = zalloc; |
58 | 6.72k | c_stream.zfree = zfree; |
59 | 6.72k | c_stream.opaque = (void *)0; |
60 | | |
61 | 6.72k | err = PREFIX(deflateInit2)(&c_stream, level, method, windowBits, memLevel, |
62 | 6.72k | strategy); |
63 | 6.72k | CHECK_ERR(err, "deflateInit"); |
64 | | |
65 | 6.72k | err = PREFIX(deflateSetDictionary)( |
66 | 6.72k | &c_stream, (const unsigned char *)data, dictionaryLen); |
67 | 6.72k | CHECK_ERR(err, "deflateSetDictionary"); |
68 | | |
69 | | /* deflateBound does not provide enough space for low compression levels. */ |
70 | 6.72k | *comprLen = 100 + 2 * PREFIX(deflateBound)(&c_stream, (unsigned long)dataLen); |
71 | 6.72k | *compr = (uint8_t *)calloc(1, *comprLen); |
72 | | |
73 | 6.72k | dictId = c_stream.adler; |
74 | 6.72k | c_stream.next_out = *compr; |
75 | 6.72k | c_stream.avail_out = (unsigned int)(*comprLen); |
76 | | |
77 | 6.72k | c_stream.next_in = (z_const unsigned char *)data; |
78 | 6.72k | c_stream.avail_in = (uint32_t)dataLen; |
79 | | |
80 | 6.72k | err = PREFIX(deflate)(&c_stream, Z_FINISH); |
81 | 6.72k | if (err != Z_STREAM_END) { |
82 | 0 | fprintf(stderr, "deflate dict should report Z_STREAM_END\n"); |
83 | 0 | exit(1); |
84 | 0 | } |
85 | 6.72k | err = PREFIX(deflateEnd)(&c_stream); |
86 | 6.72k | CHECK_ERR(err, "deflateEnd"); |
87 | 6.72k | } |
88 | | |
89 | | /* =========================================================================== |
90 | | * Test inflate() with a preset dictionary |
91 | | */ |
92 | 6.72k | void test_dict_inflate(unsigned char *compr, size_t comprLen) { |
93 | 6.72k | int err; |
94 | 6.72k | PREFIX3(stream) d_stream; /* decompression stream */ |
95 | 6.72k | unsigned char *uncompr; |
96 | | |
97 | 6.72k | d_stream.zalloc = zalloc; |
98 | 6.72k | d_stream.zfree = zfree; |
99 | 6.72k | d_stream.opaque = (void *)0; |
100 | | |
101 | 6.72k | d_stream.next_in = compr; |
102 | 6.72k | d_stream.avail_in = (unsigned int)comprLen; |
103 | | |
104 | 6.72k | err = PREFIX(inflateInit)(&d_stream); |
105 | 6.72k | CHECK_ERR(err, "inflateInit"); |
106 | | |
107 | 6.72k | uncompr = (uint8_t *)calloc(1, dataLen); |
108 | 6.72k | d_stream.next_out = uncompr; |
109 | 6.72k | d_stream.avail_out = (unsigned int)dataLen; |
110 | | |
111 | 13.1k | for (;;) { |
112 | 13.1k | err = PREFIX(inflate)(&d_stream, Z_NO_FLUSH); |
113 | 13.1k | if (err == Z_STREAM_END) |
114 | 6.72k | break; |
115 | 6.41k | if (err == Z_NEED_DICT) { |
116 | 6.41k | if (d_stream.adler != dictId) { |
117 | 0 | fprintf(stderr, "unexpected dictionary"); |
118 | 0 | exit(1); |
119 | 0 | } |
120 | 6.41k | err = PREFIX(inflateSetDictionary)( |
121 | 6.41k | &d_stream, (const unsigned char *)data, dictionaryLen); |
122 | 6.41k | } |
123 | 6.41k | CHECK_ERR(err, "inflate with dict"); |
124 | 6.41k | } |
125 | | |
126 | 6.72k | err = PREFIX(inflateEnd)(&d_stream); |
127 | 6.72k | CHECK_ERR(err, "inflateEnd"); |
128 | | |
129 | 6.72k | if (memcmp(uncompr, data, dataLen)) { |
130 | 0 | fprintf(stderr, "bad inflate with dict\n"); |
131 | 0 | exit(1); |
132 | 0 | } |
133 | | |
134 | 6.72k | free(uncompr); |
135 | 6.72k | } |
136 | | |
137 | 6.73k | int LLVMFuzzerTestOneInput(const uint8_t *d, size_t size) { |
138 | 6.73k | size_t comprLen = 0; |
139 | 6.73k | uint8_t *compr; |
140 | | |
141 | | /* Discard inputs larger than 100Kb. */ |
142 | 6.73k | static size_t kMaxSize = 100 * 1024; |
143 | | |
144 | 6.73k | if (size < 1 || size > kMaxSize) |
145 | 7 | return 0; |
146 | | |
147 | 6.72k | data = d; |
148 | 6.72k | dataLen = size; |
149 | | |
150 | | /* Set up the contents of the dictionary. The size of the dictionary is |
151 | | intentionally selected to be of unusual size. To help cover more corner |
152 | | cases, the size of the dictionary is read from the input data. */ |
153 | 6.72k | dictionaryLen = data[0]; |
154 | 6.72k | if (dictionaryLen > dataLen) |
155 | 2.16k | dictionaryLen = (unsigned int)dataLen; |
156 | | |
157 | 6.72k | test_dict_deflate(&compr, &comprLen); |
158 | 6.72k | test_dict_inflate(compr, comprLen); |
159 | | |
160 | 6.72k | free(compr); |
161 | | |
162 | | /* This function must return 0. */ |
163 | 6.72k | return 0; |
164 | 6.73k | } |