1 | /* |
2 | * The deflate_quick deflate strategy, designed to be used when cycles are |
3 | * at a premium. |
4 | * |
5 | * Copyright (C) 2013 Intel Corporation. All rights reserved. |
6 | * Authors: |
7 | * Wajdi Feghali <wajdi.k.feghali@intel.com> |
8 | * Jim Guilford <james.guilford@intel.com> |
9 | * Vinodh Gopal <vinodh.gopal@intel.com> |
10 | * Erdinc Ozturk <erdinc.ozturk@intel.com> |
11 | * Jim Kukunas <james.t.kukunas@linux.intel.com> |
12 | * |
13 | * Portions are Copyright (C) 2016 12Sided Technology, LLC. |
14 | * Author: |
15 | * Phil Vachon <pvachon@12sidedtech.com> |
16 | * |
17 | * For conditions of distribution and use, see copyright notice in zlib.h |
18 | */ |
19 | |
20 | #include "zbuild.h" |
21 | #include "deflate.h" |
22 | #include "deflate_p.h" |
23 | #include "functable.h" |
24 | #include "trees_emit.h" |
25 | |
26 | extern const ct_data static_ltree[L_CODES+2]; |
27 | extern const ct_data static_dtree[D_CODES]; |
28 | |
29 | #define QUICK_START_BLOCK(s, last) { \ |
30 | zng_tr_emit_tree(s, STATIC_TREES, last); \ |
31 | s->block_open = 1 + (int)last; \ |
32 | s->block_start = (int)s->strstart; \ |
33 | } |
34 | |
35 | #define QUICK_END_BLOCK(s, last) { \ |
36 | if (s->block_open) { \ |
37 | zng_tr_emit_end_block(s, static_ltree, last); \ |
38 | s->block_open = 0; \ |
39 | s->block_start = (int)s->strstart; \ |
40 | flush_pending(s->strm); \ |
41 | if (s->strm->avail_out == 0) \ |
42 | return (last) ? finish_started : need_more; \ |
43 | } \ |
44 | } |
45 | |
46 | Z_INTERNAL block_state deflate_quick(deflate_state *s, int flush) { |
47 | Pos hash_head; |
48 | int64_t dist; |
49 | unsigned match_len, last; |
50 | |
51 | |
52 | last = (flush == Z_FINISH) ? 1 : 0; |
53 | if (UNLIKELY(last && s->block_open != 2)) { |
54 | /* Emit end of previous block */ |
55 | QUICK_END_BLOCK(s, 0); |
56 | /* Emit start of last block */ |
57 | QUICK_START_BLOCK(s, last); |
58 | } else if (UNLIKELY(s->block_open == 0 && s->lookahead > 0)) { |
59 | /* Start new block only when we have lookahead data, so that if no |
60 | input data is given an empty block will not be written */ |
61 | QUICK_START_BLOCK(s, last); |
62 | } |
63 | |
64 | for (;;) { |
65 | if (UNLIKELY(s->pending + ((BIT_BUF_SIZE + 7) >> 3) >= s->pending_buf_size)) { |
66 | flush_pending(strm: s->strm); |
67 | if (s->strm->avail_out == 0) { |
68 | return (last && s->strm->avail_in == 0 && s->bi_valid == 0 && s->block_open == 0) ? finish_started : need_more; |
69 | } |
70 | } |
71 | |
72 | if (UNLIKELY(s->lookahead < MIN_LOOKAHEAD)) { |
73 | fill_window(s); |
74 | if (UNLIKELY(s->lookahead < MIN_LOOKAHEAD && flush == Z_NO_FLUSH)) { |
75 | return need_more; |
76 | } |
77 | if (UNLIKELY(s->lookahead == 0)) |
78 | break; |
79 | |
80 | if (UNLIKELY(s->block_open == 0)) { |
81 | /* Start new block when we have lookahead data, so that if no |
82 | input data is given an empty block will not be written */ |
83 | QUICK_START_BLOCK(s, last); |
84 | } |
85 | } |
86 | |
87 | if (LIKELY(s->lookahead >= MIN_MATCH)) { |
88 | hash_head = functable.quick_insert_string(s, s->strstart); |
89 | dist = (int64_t)s->strstart - hash_head; |
90 | |
91 | if (dist <= MAX_DIST(s) && dist > 0) { |
92 | match_len = functable.compare258(s->window + s->strstart, s->window + hash_head); |
93 | |
94 | if (match_len >= MIN_MATCH) { |
95 | if (UNLIKELY(match_len > s->lookahead)) |
96 | match_len = s->lookahead; |
97 | |
98 | check_match(s, s->strstart, hash_head, match_len); |
99 | |
100 | zng_tr_emit_dist(s, ltree: static_ltree, dtree: static_dtree, lc: match_len - MIN_MATCH, dist: (uint32_t)dist); |
101 | s->lookahead -= match_len; |
102 | s->strstart += match_len; |
103 | continue; |
104 | } |
105 | } |
106 | } |
107 | |
108 | zng_tr_emit_lit(s, ltree: static_ltree, c: s->window[s->strstart]); |
109 | s->strstart++; |
110 | s->lookahead--; |
111 | } |
112 | |
113 | s->insert = s->strstart < MIN_MATCH-1 ? s->strstart : MIN_MATCH-1; |
114 | if (UNLIKELY(last)) { |
115 | QUICK_END_BLOCK(s, 1); |
116 | return finish_done; |
117 | } |
118 | |
119 | QUICK_END_BLOCK(s, 0); |
120 | return block_done; |
121 | } |
122 | |