summaryrefslogtreecommitdiff
path: root/test
diff options
context:
space:
mode:
authorIlya Leoshkevich <iii@linux.ibm.com>2021-03-18 14:54:46 +0100
committerHans Kristian Rosbach <hk-github@circlestorm.org>2021-03-20 23:35:48 +0100
commitf426ac9db3e0ce35c83838aa8eaf248b2b624d0a (patch)
treea79814c5409f3dacff828d8c72f1df03e686bf7b /test
parentbc33b26ca5391eb6b2c952cd032920033be27a53 (diff)
Restore hash_head != 0 checks
Commit bc5915e2dec7 ("Fixed unsigned integer overflow ASAN error when hash_head > s->strstart.") removed hash_head != 0 checks in fast, medium and slow deflate, because it improved performance [1]. Unfortunately, the attached test started failing after that. Apparently, as the comments suggest, the code implicitly relies on matches with the beginning of the window being skipped. So restore the check. [1] https://github.com/zlib-ng/zlib-ng/pull/772#issuecomment-710760300
Diffstat (limited to 'test')
-rw-r--r--test/hash_head_0.c110
1 files changed, 110 insertions, 0 deletions
diff --git a/test/hash_head_0.c b/test/hash_head_0.c
new file mode 100644
index 0000000..128ae34
--- /dev/null
+++ b/test/hash_head_0.c
@@ -0,0 +1,110 @@
+/* Generated by fuzzing - test hash_head == 0 handling. */
+
+#include "zbuild.h"
+#ifdef ZLIB_COMPAT
+# include "zlib.h"
+#else
+# include "zlib-ng.h"
+#endif
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+
+int main() {
+ PREFIX3(stream) strm;
+ memset(&strm, 0, sizeof(strm));
+
+ int ret = PREFIX(deflateInit2)(&strm, 1, Z_DEFLATED, -15, 4, Z_HUFFMAN_ONLY);
+ if (ret != Z_OK) {
+ fprintf(stderr, "deflateInit2() failed with code %d\n", ret);
+ return EXIT_FAILURE;
+ }
+
+ unsigned char next_in[9698];
+ memset(next_in, 0x30, sizeof(next_in));
+ next_in[8193] = 0x00;
+ next_in[8194] = 0x00;
+ next_in[8195] = 0x00;
+ next_in[8199] = 0x8a;
+ strm.next_in = next_in;
+ unsigned char next_out[21572];
+ strm.next_out = next_out;
+
+ strm.avail_in = 0;
+ strm.avail_out = 1348;
+ ret = PREFIX(deflateParams(&strm, 3, Z_FILTERED));
+ if (ret != Z_OK) {
+ fprintf(stderr, "deflateParams() failed with code %d\n", ret);
+ return EXIT_FAILURE;
+ }
+
+ strm.avail_in = 6728;
+ strm.avail_out = 2696;
+ ret = PREFIX(deflate(&strm, Z_SYNC_FLUSH));
+ if (ret != Z_OK) {
+ fprintf(stderr, "deflate() failed with code %d\n", ret);
+ return EXIT_FAILURE;
+ }
+
+ strm.avail_in = 15;
+ strm.avail_out = 1348;
+ ret = PREFIX(deflateParams(&strm, 9, Z_FILTERED));
+ if (ret != Z_OK) {
+ fprintf(stderr, "deflateParams() failed with code %d\n", ret);
+ return EXIT_FAILURE;
+ }
+
+ strm.avail_in = 1453;
+ strm.avail_out = 1348;
+ ret = PREFIX(deflate(&strm, Z_FULL_FLUSH));
+ if (ret != Z_OK) {
+ fprintf(stderr, "deflate() failed with code %d\n", ret);
+ return EXIT_FAILURE;
+ }
+
+ strm.avail_in = next_in + sizeof(next_in) - strm.next_in;
+ strm.avail_out = next_out + sizeof(next_out) - strm.next_out;
+ ret = PREFIX(deflate)(&strm, Z_FINISH);
+ if (ret != Z_STREAM_END) {
+ fprintf(stderr, "deflate() failed with code %d\n", ret);
+ return EXIT_FAILURE;
+ }
+ uint32_t compressed_size = strm.next_out - next_out;
+
+ ret = PREFIX(deflateEnd)(&strm);
+ if (ret != Z_OK) {
+ fprintf(stderr, "deflateEnd() failed with code %d\n", ret);
+ return EXIT_FAILURE;
+ }
+
+ memset(&strm, 0, sizeof(strm));
+ ret = PREFIX(inflateInit2)(&strm, -15);
+ if (ret != Z_OK) {
+ fprintf(stderr, "inflateInit2() failed with code %d\n", ret);
+ return EXIT_FAILURE;
+ }
+
+ strm.next_in = next_out;
+ strm.avail_in = compressed_size;
+ unsigned char uncompressed[sizeof(next_in)];
+ strm.next_out = uncompressed;
+ strm.avail_out = sizeof(uncompressed);
+
+ ret = PREFIX(inflate)(&strm, Z_NO_FLUSH);
+ if (ret != Z_STREAM_END) {
+ fprintf(stderr, "inflate() failed with code %d\n", ret);
+ return EXIT_FAILURE;
+ }
+
+ ret = PREFIX(inflateEnd)(&strm);
+ if (ret != Z_OK) {
+ fprintf(stderr, "inflateEnd() failed with code %d\n", ret);
+ return EXIT_FAILURE;
+ }
+
+ if (memcmp(uncompressed, next_in, sizeof(uncompressed)) != 0) {
+ fprintf(stderr, "Uncompressed data differs from the original\n");
+ return EXIT_FAILURE;
+ }
+}