summaryrefslogtreecommitdiff
path: root/deflate_quick.c
diff options
context:
space:
mode:
authorNathan Moinvaziri <nathan@nathanm.com>2020-05-24 08:42:38 -0700
committerHans Kristian Rosbach <hk-github@circlestorm.org>2020-06-08 15:01:50 +0200
commitb6b8ad6c0fe22875add044da50efb181d8e33dc4 (patch)
tree9811b951c666a691e4b48977041cdb4bad79fef3 /deflate_quick.c
parent06d8ec517853b3314f685bb8d9a079223b3c8083 (diff)
Make deflate_quick algorithm available to all architectures. #205
Diffstat (limited to 'deflate_quick.c')
-rw-r--r--deflate_quick.c108
1 files changed, 108 insertions, 0 deletions
diff --git a/deflate_quick.c b/deflate_quick.c
new file mode 100644
index 0000000..cd79af4
--- /dev/null
+++ b/deflate_quick.c
@@ -0,0 +1,108 @@
+/*
+ * The deflate_quick deflate strategy, designed to be used when cycles are
+ * at a premium.
+ *
+ * Copyright (C) 2013 Intel Corporation. All rights reserved.
+ * Authors:
+ * Wajdi Feghali <wajdi.k.feghali@intel.com>
+ * Jim Guilford <james.guilford@intel.com>
+ * Vinodh Gopal <vinodh.gopal@intel.com>
+ * Erdinc Ozturk <erdinc.ozturk@intel.com>
+ * Jim Kukunas <james.t.kukunas@linux.intel.com>
+ *
+ * Portions are Copyright (C) 2016 12Sided Technology, LLC.
+ * Author:
+ * Phil Vachon <pvachon@12sidedtech.com>
+ *
+ * For conditions of distribution and use, see copyright notice in zlib.h
+ */
+
+#include "zbuild.h"
+#include "deflate.h"
+#include "deflate_p.h"
+#include "functable.h"
+#include "trees_emit.h"
+
+extern const ct_data static_ltree[L_CODES+2];
+extern const ct_data static_dtree[D_CODES];
+
+ZLIB_INTERNAL block_state deflate_quick(deflate_state *s, int flush) {
+ Pos hash_head;
+ unsigned dist, match_len, last;
+
+ if (s->block_open == 0) {
+ last = (flush == Z_FINISH) ? 1 : 0;
+ zng_tr_emit_tree(s, STATIC_TREES, last);
+ s->block_open = 1;
+ }
+
+ do {
+ if (s->pending + ((BIT_BUF_SIZE + 7) >> 3) >= s->pending_buf_size) {
+ flush_pending(s->strm);
+ if (s->strm->avail_in == 0 && flush != Z_FINISH) {
+ return need_more;
+ }
+ }
+
+ if (s->lookahead < MIN_LOOKAHEAD) {
+ fill_window(s);
+ if (s->lookahead < MIN_LOOKAHEAD && flush == Z_NO_FLUSH) {
+ zng_tr_emit_end_block(s, static_ltree, 0);
+ s->block_open = 0;
+ s->block_start = s->strstart;
+ flush_pending(s->strm);
+ return need_more;
+ }
+ if (s->lookahead == 0)
+ break;
+ }
+
+ if (s->lookahead >= MIN_MATCH) {
+ hash_head = functable.quick_insert_string(s, s->strstart);
+ dist = s->strstart - hash_head;
+
+ if (dist > 0 && dist < MAX_DIST(s)) {
+ match_len = functable.compare258(s->window + s->strstart, s->window + hash_head);
+
+ if (match_len >= MIN_MATCH) {
+ if (match_len > s->lookahead)
+ match_len = s->lookahead;
+
+ if (match_len > MAX_MATCH)
+ match_len = MAX_MATCH;
+
+ check_match(s, s->strstart, hash_head, match_len);
+
+ zng_tr_emit_dist(s, static_ltree, static_dtree, match_len - MIN_MATCH, dist);
+ s->lookahead -= match_len;
+ s->strstart += match_len;
+ continue;
+ }
+ }
+ }
+
+ zng_tr_emit_lit(s, static_ltree, s->window[s->strstart]);
+ s->strstart++;
+ s->lookahead--;
+ } while (s->strm->avail_out != 0);
+
+ if (s->strm->avail_out == 0 && flush != Z_FINISH)
+ return need_more;
+
+ s->insert = s->strstart < MIN_MATCH - 1 ? s->strstart : MIN_MATCH-1;
+
+ last = (flush == Z_FINISH) ? 1 : 0;
+ zng_tr_emit_end_block(s, static_ltree, last);
+ s->block_open = 0;
+ s->block_start = s->strstart;
+ flush_pending(s->strm);
+
+ if (last) {
+ if (s->strm->avail_out == 0)
+ return s->strm->avail_in == 0 ? finish_started : need_more;
+ else
+ return finish_done;
+ }
+
+ return block_done;
+}