1*6777b538SAndroid Build Coastguard Worker /* Copyright 2010 Google Inc. All Rights Reserved.
2*6777b538SAndroid Build Coastguard Worker
3*6777b538SAndroid Build Coastguard Worker Distributed under MIT license.
4*6777b538SAndroid Build Coastguard Worker See file LICENSE for detail or copy at https://opensource.org/licenses/MIT
5*6777b538SAndroid Build Coastguard Worker */
6*6777b538SAndroid Build Coastguard Worker
7*6777b538SAndroid Build Coastguard Worker /* A (forgetful) hash table to the data seen by the compressor, to
8*6777b538SAndroid Build Coastguard Worker help create backward references to previous data. */
9*6777b538SAndroid Build Coastguard Worker
10*6777b538SAndroid Build Coastguard Worker #ifndef BROTLI_ENC_HASH_H_
11*6777b538SAndroid Build Coastguard Worker #define BROTLI_ENC_HASH_H_
12*6777b538SAndroid Build Coastguard Worker
13*6777b538SAndroid Build Coastguard Worker #include <stdlib.h> /* exit */
14*6777b538SAndroid Build Coastguard Worker #include <string.h> /* memcmp, memset */
15*6777b538SAndroid Build Coastguard Worker
16*6777b538SAndroid Build Coastguard Worker #include "../common/constants.h"
17*6777b538SAndroid Build Coastguard Worker #include "../common/dictionary.h"
18*6777b538SAndroid Build Coastguard Worker #include "../common/platform.h"
19*6777b538SAndroid Build Coastguard Worker #include <brotli/types.h>
20*6777b538SAndroid Build Coastguard Worker #include "encoder_dict.h"
21*6777b538SAndroid Build Coastguard Worker #include "fast_log.h"
22*6777b538SAndroid Build Coastguard Worker #include "find_match_length.h"
23*6777b538SAndroid Build Coastguard Worker #include "memory.h"
24*6777b538SAndroid Build Coastguard Worker #include "quality.h"
25*6777b538SAndroid Build Coastguard Worker #include "static_dict.h"
26*6777b538SAndroid Build Coastguard Worker
27*6777b538SAndroid Build Coastguard Worker #if defined(__cplusplus) || defined(c_plusplus)
28*6777b538SAndroid Build Coastguard Worker extern "C" {
29*6777b538SAndroid Build Coastguard Worker #endif
30*6777b538SAndroid Build Coastguard Worker
31*6777b538SAndroid Build Coastguard Worker typedef struct {
32*6777b538SAndroid Build Coastguard Worker /**
33*6777b538SAndroid Build Coastguard Worker * Dynamically allocated areas; regular hasher uses one or two allocations;
34*6777b538SAndroid Build Coastguard Worker * "composite" hasher uses up to 4 allocations.
35*6777b538SAndroid Build Coastguard Worker */
36*6777b538SAndroid Build Coastguard Worker void* extra[4];
37*6777b538SAndroid Build Coastguard Worker
38*6777b538SAndroid Build Coastguard Worker /**
39*6777b538SAndroid Build Coastguard Worker * False before the fisrt invocation of HasherSetup (where "extra" memory)
40*6777b538SAndroid Build Coastguard Worker * is allocated.
41*6777b538SAndroid Build Coastguard Worker */
42*6777b538SAndroid Build Coastguard Worker BROTLI_BOOL is_setup_;
43*6777b538SAndroid Build Coastguard Worker
44*6777b538SAndroid Build Coastguard Worker size_t dict_num_lookups;
45*6777b538SAndroid Build Coastguard Worker size_t dict_num_matches;
46*6777b538SAndroid Build Coastguard Worker
47*6777b538SAndroid Build Coastguard Worker BrotliHasherParams params;
48*6777b538SAndroid Build Coastguard Worker
49*6777b538SAndroid Build Coastguard Worker /**
50*6777b538SAndroid Build Coastguard Worker * False if hasher needs to be "prepared" before use (before the first
51*6777b538SAndroid Build Coastguard Worker * invocation of HasherSetup or after HasherReset). "preparation" is hasher
52*6777b538SAndroid Build Coastguard Worker * data initialization (using input ringbuffer).
53*6777b538SAndroid Build Coastguard Worker */
54*6777b538SAndroid Build Coastguard Worker BROTLI_BOOL is_prepared_;
55*6777b538SAndroid Build Coastguard Worker } HasherCommon;
56*6777b538SAndroid Build Coastguard Worker
57*6777b538SAndroid Build Coastguard Worker #define score_t size_t
58*6777b538SAndroid Build Coastguard Worker
59*6777b538SAndroid Build Coastguard Worker static const uint32_t kCutoffTransformsCount = 10;
60*6777b538SAndroid Build Coastguard Worker /* 0, 12, 27, 23, 42, 63, 56, 48, 59, 64 */
61*6777b538SAndroid Build Coastguard Worker /* 0+0, 4+8, 8+19, 12+11, 16+26, 20+43, 24+32, 28+20, 32+27, 36+28 */
62*6777b538SAndroid Build Coastguard Worker static const uint64_t kCutoffTransforms =
63*6777b538SAndroid Build Coastguard Worker BROTLI_MAKE_UINT64_T(0x071B520A, 0xDA2D3200);
64*6777b538SAndroid Build Coastguard Worker
65*6777b538SAndroid Build Coastguard Worker typedef struct HasherSearchResult {
66*6777b538SAndroid Build Coastguard Worker size_t len;
67*6777b538SAndroid Build Coastguard Worker size_t distance;
68*6777b538SAndroid Build Coastguard Worker score_t score;
69*6777b538SAndroid Build Coastguard Worker int len_code_delta; /* == len_code - len */
70*6777b538SAndroid Build Coastguard Worker } HasherSearchResult;
71*6777b538SAndroid Build Coastguard Worker
72*6777b538SAndroid Build Coastguard Worker /* kHashMul32 multiplier has these properties:
73*6777b538SAndroid Build Coastguard Worker * The multiplier must be odd. Otherwise we may lose the highest bit.
74*6777b538SAndroid Build Coastguard Worker * No long streaks of ones or zeros.
75*6777b538SAndroid Build Coastguard Worker * There is no effort to ensure that it is a prime, the oddity is enough
76*6777b538SAndroid Build Coastguard Worker for this use.
77*6777b538SAndroid Build Coastguard Worker * The number has been tuned heuristically against compression benchmarks. */
78*6777b538SAndroid Build Coastguard Worker static const uint32_t kHashMul32 = 0x1E35A7BD;
79*6777b538SAndroid Build Coastguard Worker static const uint64_t kHashMul64 = BROTLI_MAKE_UINT64_T(0x1E35A7BD, 0x1E35A7BD);
80*6777b538SAndroid Build Coastguard Worker static const uint64_t kHashMul64Long =
81*6777b538SAndroid Build Coastguard Worker BROTLI_MAKE_UINT64_T(0x1FE35A7Bu, 0xD3579BD3u);
82*6777b538SAndroid Build Coastguard Worker
Hash14(const uint8_t * data)83*6777b538SAndroid Build Coastguard Worker static BROTLI_INLINE uint32_t Hash14(const uint8_t* data) {
84*6777b538SAndroid Build Coastguard Worker uint32_t h = BROTLI_UNALIGNED_LOAD32LE(data) * kHashMul32;
85*6777b538SAndroid Build Coastguard Worker /* The higher bits contain more mixture from the multiplication,
86*6777b538SAndroid Build Coastguard Worker so we take our results from there. */
87*6777b538SAndroid Build Coastguard Worker return h >> (32 - 14);
88*6777b538SAndroid Build Coastguard Worker }
89*6777b538SAndroid Build Coastguard Worker
PrepareDistanceCache(int * BROTLI_RESTRICT distance_cache,const int num_distances)90*6777b538SAndroid Build Coastguard Worker static BROTLI_INLINE void PrepareDistanceCache(
91*6777b538SAndroid Build Coastguard Worker int* BROTLI_RESTRICT distance_cache, const int num_distances) {
92*6777b538SAndroid Build Coastguard Worker if (num_distances > 4) {
93*6777b538SAndroid Build Coastguard Worker int last_distance = distance_cache[0];
94*6777b538SAndroid Build Coastguard Worker distance_cache[4] = last_distance - 1;
95*6777b538SAndroid Build Coastguard Worker distance_cache[5] = last_distance + 1;
96*6777b538SAndroid Build Coastguard Worker distance_cache[6] = last_distance - 2;
97*6777b538SAndroid Build Coastguard Worker distance_cache[7] = last_distance + 2;
98*6777b538SAndroid Build Coastguard Worker distance_cache[8] = last_distance - 3;
99*6777b538SAndroid Build Coastguard Worker distance_cache[9] = last_distance + 3;
100*6777b538SAndroid Build Coastguard Worker if (num_distances > 10) {
101*6777b538SAndroid Build Coastguard Worker int next_last_distance = distance_cache[1];
102*6777b538SAndroid Build Coastguard Worker distance_cache[10] = next_last_distance - 1;
103*6777b538SAndroid Build Coastguard Worker distance_cache[11] = next_last_distance + 1;
104*6777b538SAndroid Build Coastguard Worker distance_cache[12] = next_last_distance - 2;
105*6777b538SAndroid Build Coastguard Worker distance_cache[13] = next_last_distance + 2;
106*6777b538SAndroid Build Coastguard Worker distance_cache[14] = next_last_distance - 3;
107*6777b538SAndroid Build Coastguard Worker distance_cache[15] = next_last_distance + 3;
108*6777b538SAndroid Build Coastguard Worker }
109*6777b538SAndroid Build Coastguard Worker }
110*6777b538SAndroid Build Coastguard Worker }
111*6777b538SAndroid Build Coastguard Worker
112*6777b538SAndroid Build Coastguard Worker #define BROTLI_LITERAL_BYTE_SCORE 135
113*6777b538SAndroid Build Coastguard Worker #define BROTLI_DISTANCE_BIT_PENALTY 30
114*6777b538SAndroid Build Coastguard Worker /* Score must be positive after applying maximal penalty. */
115*6777b538SAndroid Build Coastguard Worker #define BROTLI_SCORE_BASE (BROTLI_DISTANCE_BIT_PENALTY * 8 * sizeof(size_t))
116*6777b538SAndroid Build Coastguard Worker
117*6777b538SAndroid Build Coastguard Worker /* Usually, we always choose the longest backward reference. This function
118*6777b538SAndroid Build Coastguard Worker allows for the exception of that rule.
119*6777b538SAndroid Build Coastguard Worker
120*6777b538SAndroid Build Coastguard Worker If we choose a backward reference that is further away, it will
121*6777b538SAndroid Build Coastguard Worker usually be coded with more bits. We approximate this by assuming
122*6777b538SAndroid Build Coastguard Worker log2(distance). If the distance can be expressed in terms of the
123*6777b538SAndroid Build Coastguard Worker last four distances, we use some heuristic constants to estimate
124*6777b538SAndroid Build Coastguard Worker the bits cost. For the first up to four literals we use the bit
125*6777b538SAndroid Build Coastguard Worker cost of the literals from the literal cost model, after that we
126*6777b538SAndroid Build Coastguard Worker use the average bit cost of the cost model.
127*6777b538SAndroid Build Coastguard Worker
128*6777b538SAndroid Build Coastguard Worker This function is used to sometimes discard a longer backward reference
129*6777b538SAndroid Build Coastguard Worker when it is not much longer and the bit cost for encoding it is more
130*6777b538SAndroid Build Coastguard Worker than the saved literals.
131*6777b538SAndroid Build Coastguard Worker
132*6777b538SAndroid Build Coastguard Worker backward_reference_offset MUST be positive. */
BackwardReferenceScore(size_t copy_length,size_t backward_reference_offset)133*6777b538SAndroid Build Coastguard Worker static BROTLI_INLINE score_t BackwardReferenceScore(
134*6777b538SAndroid Build Coastguard Worker size_t copy_length, size_t backward_reference_offset) {
135*6777b538SAndroid Build Coastguard Worker return BROTLI_SCORE_BASE + BROTLI_LITERAL_BYTE_SCORE * (score_t)copy_length -
136*6777b538SAndroid Build Coastguard Worker BROTLI_DISTANCE_BIT_PENALTY * Log2FloorNonZero(backward_reference_offset);
137*6777b538SAndroid Build Coastguard Worker }
138*6777b538SAndroid Build Coastguard Worker
BackwardReferenceScoreUsingLastDistance(size_t copy_length)139*6777b538SAndroid Build Coastguard Worker static BROTLI_INLINE score_t BackwardReferenceScoreUsingLastDistance(
140*6777b538SAndroid Build Coastguard Worker size_t copy_length) {
141*6777b538SAndroid Build Coastguard Worker return BROTLI_LITERAL_BYTE_SCORE * (score_t)copy_length +
142*6777b538SAndroid Build Coastguard Worker BROTLI_SCORE_BASE + 15;
143*6777b538SAndroid Build Coastguard Worker }
144*6777b538SAndroid Build Coastguard Worker
BackwardReferencePenaltyUsingLastDistance(size_t distance_short_code)145*6777b538SAndroid Build Coastguard Worker static BROTLI_INLINE score_t BackwardReferencePenaltyUsingLastDistance(
146*6777b538SAndroid Build Coastguard Worker size_t distance_short_code) {
147*6777b538SAndroid Build Coastguard Worker return (score_t)39 + ((0x1CA10 >> (distance_short_code & 0xE)) & 0xE);
148*6777b538SAndroid Build Coastguard Worker }
149*6777b538SAndroid Build Coastguard Worker
TestStaticDictionaryItem(const BrotliEncoderDictionary * dictionary,size_t len,size_t word_idx,const uint8_t * data,size_t max_length,size_t max_backward,size_t max_distance,HasherSearchResult * out)150*6777b538SAndroid Build Coastguard Worker static BROTLI_INLINE BROTLI_BOOL TestStaticDictionaryItem(
151*6777b538SAndroid Build Coastguard Worker const BrotliEncoderDictionary* dictionary, size_t len, size_t word_idx,
152*6777b538SAndroid Build Coastguard Worker const uint8_t* data, size_t max_length, size_t max_backward,
153*6777b538SAndroid Build Coastguard Worker size_t max_distance, HasherSearchResult* out) {
154*6777b538SAndroid Build Coastguard Worker size_t offset;
155*6777b538SAndroid Build Coastguard Worker size_t matchlen;
156*6777b538SAndroid Build Coastguard Worker size_t backward;
157*6777b538SAndroid Build Coastguard Worker score_t score;
158*6777b538SAndroid Build Coastguard Worker offset = dictionary->words->offsets_by_length[len] + len * word_idx;
159*6777b538SAndroid Build Coastguard Worker if (len > max_length) {
160*6777b538SAndroid Build Coastguard Worker return BROTLI_FALSE;
161*6777b538SAndroid Build Coastguard Worker }
162*6777b538SAndroid Build Coastguard Worker
163*6777b538SAndroid Build Coastguard Worker matchlen =
164*6777b538SAndroid Build Coastguard Worker FindMatchLengthWithLimit(data, &dictionary->words->data[offset], len);
165*6777b538SAndroid Build Coastguard Worker if (matchlen + dictionary->cutoffTransformsCount <= len || matchlen == 0) {
166*6777b538SAndroid Build Coastguard Worker return BROTLI_FALSE;
167*6777b538SAndroid Build Coastguard Worker }
168*6777b538SAndroid Build Coastguard Worker {
169*6777b538SAndroid Build Coastguard Worker size_t cut = len - matchlen;
170*6777b538SAndroid Build Coastguard Worker size_t transform_id = (cut << 2) +
171*6777b538SAndroid Build Coastguard Worker (size_t)((dictionary->cutoffTransforms >> (cut * 6)) & 0x3F);
172*6777b538SAndroid Build Coastguard Worker backward = max_backward + 1 + word_idx +
173*6777b538SAndroid Build Coastguard Worker (transform_id << dictionary->words->size_bits_by_length[len]);
174*6777b538SAndroid Build Coastguard Worker }
175*6777b538SAndroid Build Coastguard Worker if (backward > max_distance) {
176*6777b538SAndroid Build Coastguard Worker return BROTLI_FALSE;
177*6777b538SAndroid Build Coastguard Worker }
178*6777b538SAndroid Build Coastguard Worker score = BackwardReferenceScore(matchlen, backward);
179*6777b538SAndroid Build Coastguard Worker if (score < out->score) {
180*6777b538SAndroid Build Coastguard Worker return BROTLI_FALSE;
181*6777b538SAndroid Build Coastguard Worker }
182*6777b538SAndroid Build Coastguard Worker out->len = matchlen;
183*6777b538SAndroid Build Coastguard Worker out->len_code_delta = (int)len - (int)matchlen;
184*6777b538SAndroid Build Coastguard Worker out->distance = backward;
185*6777b538SAndroid Build Coastguard Worker out->score = score;
186*6777b538SAndroid Build Coastguard Worker return BROTLI_TRUE;
187*6777b538SAndroid Build Coastguard Worker }
188*6777b538SAndroid Build Coastguard Worker
SearchInStaticDictionary(const BrotliEncoderDictionary * dictionary,HasherCommon * common,const uint8_t * data,size_t max_length,size_t max_backward,size_t max_distance,HasherSearchResult * out,BROTLI_BOOL shallow)189*6777b538SAndroid Build Coastguard Worker static BROTLI_INLINE void SearchInStaticDictionary(
190*6777b538SAndroid Build Coastguard Worker const BrotliEncoderDictionary* dictionary,
191*6777b538SAndroid Build Coastguard Worker HasherCommon* common, const uint8_t* data, size_t max_length,
192*6777b538SAndroid Build Coastguard Worker size_t max_backward, size_t max_distance,
193*6777b538SAndroid Build Coastguard Worker HasherSearchResult* out, BROTLI_BOOL shallow) {
194*6777b538SAndroid Build Coastguard Worker size_t key;
195*6777b538SAndroid Build Coastguard Worker size_t i;
196*6777b538SAndroid Build Coastguard Worker if (common->dict_num_matches < (common->dict_num_lookups >> 7)) {
197*6777b538SAndroid Build Coastguard Worker return;
198*6777b538SAndroid Build Coastguard Worker }
199*6777b538SAndroid Build Coastguard Worker key = Hash14(data) << 1;
200*6777b538SAndroid Build Coastguard Worker for (i = 0; i < (shallow ? 1u : 2u); ++i, ++key) {
201*6777b538SAndroid Build Coastguard Worker common->dict_num_lookups++;
202*6777b538SAndroid Build Coastguard Worker if (dictionary->hash_table_lengths[key] != 0) {
203*6777b538SAndroid Build Coastguard Worker BROTLI_BOOL item_matches = TestStaticDictionaryItem(
204*6777b538SAndroid Build Coastguard Worker dictionary, dictionary->hash_table_lengths[key],
205*6777b538SAndroid Build Coastguard Worker dictionary->hash_table_words[key], data,
206*6777b538SAndroid Build Coastguard Worker max_length, max_backward, max_distance, out);
207*6777b538SAndroid Build Coastguard Worker if (item_matches) {
208*6777b538SAndroid Build Coastguard Worker common->dict_num_matches++;
209*6777b538SAndroid Build Coastguard Worker }
210*6777b538SAndroid Build Coastguard Worker }
211*6777b538SAndroid Build Coastguard Worker }
212*6777b538SAndroid Build Coastguard Worker }
213*6777b538SAndroid Build Coastguard Worker
214*6777b538SAndroid Build Coastguard Worker typedef struct BackwardMatch {
215*6777b538SAndroid Build Coastguard Worker uint32_t distance;
216*6777b538SAndroid Build Coastguard Worker uint32_t length_and_code;
217*6777b538SAndroid Build Coastguard Worker } BackwardMatch;
218*6777b538SAndroid Build Coastguard Worker
InitBackwardMatch(BackwardMatch * self,size_t dist,size_t len)219*6777b538SAndroid Build Coastguard Worker static BROTLI_INLINE void InitBackwardMatch(BackwardMatch* self,
220*6777b538SAndroid Build Coastguard Worker size_t dist, size_t len) {
221*6777b538SAndroid Build Coastguard Worker self->distance = (uint32_t)dist;
222*6777b538SAndroid Build Coastguard Worker self->length_and_code = (uint32_t)(len << 5);
223*6777b538SAndroid Build Coastguard Worker }
224*6777b538SAndroid Build Coastguard Worker
InitDictionaryBackwardMatch(BackwardMatch * self,size_t dist,size_t len,size_t len_code)225*6777b538SAndroid Build Coastguard Worker static BROTLI_INLINE void InitDictionaryBackwardMatch(BackwardMatch* self,
226*6777b538SAndroid Build Coastguard Worker size_t dist, size_t len, size_t len_code) {
227*6777b538SAndroid Build Coastguard Worker self->distance = (uint32_t)dist;
228*6777b538SAndroid Build Coastguard Worker self->length_and_code =
229*6777b538SAndroid Build Coastguard Worker (uint32_t)((len << 5) | (len == len_code ? 0 : len_code));
230*6777b538SAndroid Build Coastguard Worker }
231*6777b538SAndroid Build Coastguard Worker
BackwardMatchLength(const BackwardMatch * self)232*6777b538SAndroid Build Coastguard Worker static BROTLI_INLINE size_t BackwardMatchLength(const BackwardMatch* self) {
233*6777b538SAndroid Build Coastguard Worker return self->length_and_code >> 5;
234*6777b538SAndroid Build Coastguard Worker }
235*6777b538SAndroid Build Coastguard Worker
BackwardMatchLengthCode(const BackwardMatch * self)236*6777b538SAndroid Build Coastguard Worker static BROTLI_INLINE size_t BackwardMatchLengthCode(const BackwardMatch* self) {
237*6777b538SAndroid Build Coastguard Worker size_t code = self->length_and_code & 31;
238*6777b538SAndroid Build Coastguard Worker return code ? code : BackwardMatchLength(self);
239*6777b538SAndroid Build Coastguard Worker }
240*6777b538SAndroid Build Coastguard Worker
241*6777b538SAndroid Build Coastguard Worker #define EXPAND_CAT(a, b) CAT(a, b)
242*6777b538SAndroid Build Coastguard Worker #define CAT(a, b) a ## b
243*6777b538SAndroid Build Coastguard Worker #define FN(X) EXPAND_CAT(X, HASHER())
244*6777b538SAndroid Build Coastguard Worker
245*6777b538SAndroid Build Coastguard Worker #define HASHER() H10
246*6777b538SAndroid Build Coastguard Worker #define BUCKET_BITS 17
247*6777b538SAndroid Build Coastguard Worker #define MAX_TREE_SEARCH_DEPTH 64
248*6777b538SAndroid Build Coastguard Worker #define MAX_TREE_COMP_LENGTH 128
249*6777b538SAndroid Build Coastguard Worker #include "hash_to_binary_tree_inc.h" /* NOLINT(build/include) */
250*6777b538SAndroid Build Coastguard Worker #undef MAX_TREE_SEARCH_DEPTH
251*6777b538SAndroid Build Coastguard Worker #undef MAX_TREE_COMP_LENGTH
252*6777b538SAndroid Build Coastguard Worker #undef BUCKET_BITS
253*6777b538SAndroid Build Coastguard Worker #undef HASHER
254*6777b538SAndroid Build Coastguard Worker /* MAX_NUM_MATCHES == 64 + MAX_TREE_SEARCH_DEPTH */
255*6777b538SAndroid Build Coastguard Worker #define MAX_NUM_MATCHES_H10 128
256*6777b538SAndroid Build Coastguard Worker
257*6777b538SAndroid Build Coastguard Worker /* For BUCKET_SWEEP_BITS == 0, enabling the dictionary lookup makes compression
258*6777b538SAndroid Build Coastguard Worker a little faster (0.5% - 1%) and it compresses 0.15% better on small text
259*6777b538SAndroid Build Coastguard Worker and HTML inputs. */
260*6777b538SAndroid Build Coastguard Worker
261*6777b538SAndroid Build Coastguard Worker #define HASHER() H2
262*6777b538SAndroid Build Coastguard Worker #define BUCKET_BITS 16
263*6777b538SAndroid Build Coastguard Worker #define BUCKET_SWEEP_BITS 0
264*6777b538SAndroid Build Coastguard Worker #define HASH_LEN 5
265*6777b538SAndroid Build Coastguard Worker #define USE_DICTIONARY 1
266*6777b538SAndroid Build Coastguard Worker #include "hash_longest_match_quickly_inc.h" /* NOLINT(build/include) */
267*6777b538SAndroid Build Coastguard Worker #undef BUCKET_SWEEP_BITS
268*6777b538SAndroid Build Coastguard Worker #undef USE_DICTIONARY
269*6777b538SAndroid Build Coastguard Worker #undef HASHER
270*6777b538SAndroid Build Coastguard Worker
271*6777b538SAndroid Build Coastguard Worker #define HASHER() H3
272*6777b538SAndroid Build Coastguard Worker #define BUCKET_SWEEP_BITS 1
273*6777b538SAndroid Build Coastguard Worker #define USE_DICTIONARY 0
274*6777b538SAndroid Build Coastguard Worker #include "hash_longest_match_quickly_inc.h" /* NOLINT(build/include) */
275*6777b538SAndroid Build Coastguard Worker #undef USE_DICTIONARY
276*6777b538SAndroid Build Coastguard Worker #undef BUCKET_SWEEP_BITS
277*6777b538SAndroid Build Coastguard Worker #undef BUCKET_BITS
278*6777b538SAndroid Build Coastguard Worker #undef HASHER
279*6777b538SAndroid Build Coastguard Worker
280*6777b538SAndroid Build Coastguard Worker #define HASHER() H4
281*6777b538SAndroid Build Coastguard Worker #define BUCKET_BITS 17
282*6777b538SAndroid Build Coastguard Worker #define BUCKET_SWEEP_BITS 2
283*6777b538SAndroid Build Coastguard Worker #define USE_DICTIONARY 1
284*6777b538SAndroid Build Coastguard Worker #include "hash_longest_match_quickly_inc.h" /* NOLINT(build/include) */
285*6777b538SAndroid Build Coastguard Worker #undef USE_DICTIONARY
286*6777b538SAndroid Build Coastguard Worker #undef HASH_LEN
287*6777b538SAndroid Build Coastguard Worker #undef BUCKET_SWEEP_BITS
288*6777b538SAndroid Build Coastguard Worker #undef BUCKET_BITS
289*6777b538SAndroid Build Coastguard Worker #undef HASHER
290*6777b538SAndroid Build Coastguard Worker
291*6777b538SAndroid Build Coastguard Worker #define HASHER() H5
292*6777b538SAndroid Build Coastguard Worker #include "hash_longest_match_inc.h" /* NOLINT(build/include) */
293*6777b538SAndroid Build Coastguard Worker #undef HASHER
294*6777b538SAndroid Build Coastguard Worker
295*6777b538SAndroid Build Coastguard Worker #define HASHER() H6
296*6777b538SAndroid Build Coastguard Worker #include "hash_longest_match64_inc.h" /* NOLINT(build/include) */
297*6777b538SAndroid Build Coastguard Worker #undef HASHER
298*6777b538SAndroid Build Coastguard Worker
299*6777b538SAndroid Build Coastguard Worker #define BUCKET_BITS 15
300*6777b538SAndroid Build Coastguard Worker
301*6777b538SAndroid Build Coastguard Worker #define NUM_LAST_DISTANCES_TO_CHECK 4
302*6777b538SAndroid Build Coastguard Worker #define NUM_BANKS 1
303*6777b538SAndroid Build Coastguard Worker #define BANK_BITS 16
304*6777b538SAndroid Build Coastguard Worker #define HASHER() H40
305*6777b538SAndroid Build Coastguard Worker #include "hash_forgetful_chain_inc.h" /* NOLINT(build/include) */
306*6777b538SAndroid Build Coastguard Worker #undef HASHER
307*6777b538SAndroid Build Coastguard Worker #undef NUM_LAST_DISTANCES_TO_CHECK
308*6777b538SAndroid Build Coastguard Worker
309*6777b538SAndroid Build Coastguard Worker #define NUM_LAST_DISTANCES_TO_CHECK 10
310*6777b538SAndroid Build Coastguard Worker #define HASHER() H41
311*6777b538SAndroid Build Coastguard Worker #include "hash_forgetful_chain_inc.h" /* NOLINT(build/include) */
312*6777b538SAndroid Build Coastguard Worker #undef HASHER
313*6777b538SAndroid Build Coastguard Worker #undef NUM_LAST_DISTANCES_TO_CHECK
314*6777b538SAndroid Build Coastguard Worker #undef NUM_BANKS
315*6777b538SAndroid Build Coastguard Worker #undef BANK_BITS
316*6777b538SAndroid Build Coastguard Worker
317*6777b538SAndroid Build Coastguard Worker #define NUM_LAST_DISTANCES_TO_CHECK 16
318*6777b538SAndroid Build Coastguard Worker #define NUM_BANKS 512
319*6777b538SAndroid Build Coastguard Worker #define BANK_BITS 9
320*6777b538SAndroid Build Coastguard Worker #define HASHER() H42
321*6777b538SAndroid Build Coastguard Worker #include "hash_forgetful_chain_inc.h" /* NOLINT(build/include) */
322*6777b538SAndroid Build Coastguard Worker #undef HASHER
323*6777b538SAndroid Build Coastguard Worker #undef NUM_LAST_DISTANCES_TO_CHECK
324*6777b538SAndroid Build Coastguard Worker #undef NUM_BANKS
325*6777b538SAndroid Build Coastguard Worker #undef BANK_BITS
326*6777b538SAndroid Build Coastguard Worker
327*6777b538SAndroid Build Coastguard Worker #undef BUCKET_BITS
328*6777b538SAndroid Build Coastguard Worker
329*6777b538SAndroid Build Coastguard Worker #define HASHER() H54
330*6777b538SAndroid Build Coastguard Worker #define BUCKET_BITS 20
331*6777b538SAndroid Build Coastguard Worker #define BUCKET_SWEEP_BITS 2
332*6777b538SAndroid Build Coastguard Worker #define HASH_LEN 7
333*6777b538SAndroid Build Coastguard Worker #define USE_DICTIONARY 0
334*6777b538SAndroid Build Coastguard Worker #include "hash_longest_match_quickly_inc.h" /* NOLINT(build/include) */
335*6777b538SAndroid Build Coastguard Worker #undef USE_DICTIONARY
336*6777b538SAndroid Build Coastguard Worker #undef HASH_LEN
337*6777b538SAndroid Build Coastguard Worker #undef BUCKET_SWEEP_BITS
338*6777b538SAndroid Build Coastguard Worker #undef BUCKET_BITS
339*6777b538SAndroid Build Coastguard Worker #undef HASHER
340*6777b538SAndroid Build Coastguard Worker
341*6777b538SAndroid Build Coastguard Worker /* fast large window hashers */
342*6777b538SAndroid Build Coastguard Worker
343*6777b538SAndroid Build Coastguard Worker #define HASHER() HROLLING_FAST
344*6777b538SAndroid Build Coastguard Worker #define CHUNKLEN 32
345*6777b538SAndroid Build Coastguard Worker #define JUMP 4
346*6777b538SAndroid Build Coastguard Worker #define NUMBUCKETS 16777216
347*6777b538SAndroid Build Coastguard Worker #define MASK ((NUMBUCKETS * 64) - 1)
348*6777b538SAndroid Build Coastguard Worker #include "hash_rolling_inc.h" /* NOLINT(build/include) */
349*6777b538SAndroid Build Coastguard Worker #undef JUMP
350*6777b538SAndroid Build Coastguard Worker #undef HASHER
351*6777b538SAndroid Build Coastguard Worker
352*6777b538SAndroid Build Coastguard Worker
353*6777b538SAndroid Build Coastguard Worker #define HASHER() HROLLING
354*6777b538SAndroid Build Coastguard Worker #define JUMP 1
355*6777b538SAndroid Build Coastguard Worker #include "hash_rolling_inc.h" /* NOLINT(build/include) */
356*6777b538SAndroid Build Coastguard Worker #undef MASK
357*6777b538SAndroid Build Coastguard Worker #undef NUMBUCKETS
358*6777b538SAndroid Build Coastguard Worker #undef JUMP
359*6777b538SAndroid Build Coastguard Worker #undef CHUNKLEN
360*6777b538SAndroid Build Coastguard Worker #undef HASHER
361*6777b538SAndroid Build Coastguard Worker
362*6777b538SAndroid Build Coastguard Worker #define HASHER() H35
363*6777b538SAndroid Build Coastguard Worker #define HASHER_A H3
364*6777b538SAndroid Build Coastguard Worker #define HASHER_B HROLLING_FAST
365*6777b538SAndroid Build Coastguard Worker #include "hash_composite_inc.h" /* NOLINT(build/include) */
366*6777b538SAndroid Build Coastguard Worker #undef HASHER_A
367*6777b538SAndroid Build Coastguard Worker #undef HASHER_B
368*6777b538SAndroid Build Coastguard Worker #undef HASHER
369*6777b538SAndroid Build Coastguard Worker
370*6777b538SAndroid Build Coastguard Worker #define HASHER() H55
371*6777b538SAndroid Build Coastguard Worker #define HASHER_A H54
372*6777b538SAndroid Build Coastguard Worker #define HASHER_B HROLLING_FAST
373*6777b538SAndroid Build Coastguard Worker #include "hash_composite_inc.h" /* NOLINT(build/include) */
374*6777b538SAndroid Build Coastguard Worker #undef HASHER_A
375*6777b538SAndroid Build Coastguard Worker #undef HASHER_B
376*6777b538SAndroid Build Coastguard Worker #undef HASHER
377*6777b538SAndroid Build Coastguard Worker
378*6777b538SAndroid Build Coastguard Worker #define HASHER() H65
379*6777b538SAndroid Build Coastguard Worker #define HASHER_A H6
380*6777b538SAndroid Build Coastguard Worker #define HASHER_B HROLLING
381*6777b538SAndroid Build Coastguard Worker #include "hash_composite_inc.h" /* NOLINT(build/include) */
382*6777b538SAndroid Build Coastguard Worker #undef HASHER_A
383*6777b538SAndroid Build Coastguard Worker #undef HASHER_B
384*6777b538SAndroid Build Coastguard Worker #undef HASHER
385*6777b538SAndroid Build Coastguard Worker
386*6777b538SAndroid Build Coastguard Worker #undef FN
387*6777b538SAndroid Build Coastguard Worker #undef CAT
388*6777b538SAndroid Build Coastguard Worker #undef EXPAND_CAT
389*6777b538SAndroid Build Coastguard Worker
390*6777b538SAndroid Build Coastguard Worker #define FOR_SIMPLE_HASHERS(H) H(2) H(3) H(4) H(5) H(6) H(40) H(41) H(42) H(54)
391*6777b538SAndroid Build Coastguard Worker #define FOR_COMPOSITE_HASHERS(H) H(35) H(55) H(65)
392*6777b538SAndroid Build Coastguard Worker #define FOR_GENERIC_HASHERS(H) FOR_SIMPLE_HASHERS(H) FOR_COMPOSITE_HASHERS(H)
393*6777b538SAndroid Build Coastguard Worker #define FOR_ALL_HASHERS(H) FOR_GENERIC_HASHERS(H) H(10)
394*6777b538SAndroid Build Coastguard Worker
395*6777b538SAndroid Build Coastguard Worker typedef struct {
396*6777b538SAndroid Build Coastguard Worker HasherCommon common;
397*6777b538SAndroid Build Coastguard Worker
398*6777b538SAndroid Build Coastguard Worker union {
399*6777b538SAndroid Build Coastguard Worker #define MEMBER_(N) \
400*6777b538SAndroid Build Coastguard Worker H ## N _H ## N;
401*6777b538SAndroid Build Coastguard Worker FOR_ALL_HASHERS(MEMBER_)
402*6777b538SAndroid Build Coastguard Worker #undef MEMBER_
403*6777b538SAndroid Build Coastguard Worker } privat;
404*6777b538SAndroid Build Coastguard Worker } Hasher;
405*6777b538SAndroid Build Coastguard Worker
406*6777b538SAndroid Build Coastguard Worker /* MUST be invoked before any other method. */
HasherInit(Hasher * hasher)407*6777b538SAndroid Build Coastguard Worker static BROTLI_INLINE void HasherInit(Hasher* hasher) {
408*6777b538SAndroid Build Coastguard Worker hasher->common.is_setup_ = BROTLI_FALSE;
409*6777b538SAndroid Build Coastguard Worker hasher->common.extra[0] = NULL;
410*6777b538SAndroid Build Coastguard Worker hasher->common.extra[1] = NULL;
411*6777b538SAndroid Build Coastguard Worker hasher->common.extra[2] = NULL;
412*6777b538SAndroid Build Coastguard Worker hasher->common.extra[3] = NULL;
413*6777b538SAndroid Build Coastguard Worker }
414*6777b538SAndroid Build Coastguard Worker
DestroyHasher(MemoryManager * m,Hasher * hasher)415*6777b538SAndroid Build Coastguard Worker static BROTLI_INLINE void DestroyHasher(MemoryManager* m, Hasher* hasher) {
416*6777b538SAndroid Build Coastguard Worker if (hasher->common.extra[0] != NULL) BROTLI_FREE(m, hasher->common.extra[0]);
417*6777b538SAndroid Build Coastguard Worker if (hasher->common.extra[1] != NULL) BROTLI_FREE(m, hasher->common.extra[1]);
418*6777b538SAndroid Build Coastguard Worker if (hasher->common.extra[2] != NULL) BROTLI_FREE(m, hasher->common.extra[2]);
419*6777b538SAndroid Build Coastguard Worker if (hasher->common.extra[3] != NULL) BROTLI_FREE(m, hasher->common.extra[3]);
420*6777b538SAndroid Build Coastguard Worker }
421*6777b538SAndroid Build Coastguard Worker
HasherReset(Hasher * hasher)422*6777b538SAndroid Build Coastguard Worker static BROTLI_INLINE void HasherReset(Hasher* hasher) {
423*6777b538SAndroid Build Coastguard Worker hasher->common.is_prepared_ = BROTLI_FALSE;
424*6777b538SAndroid Build Coastguard Worker }
425*6777b538SAndroid Build Coastguard Worker
HasherSize(const BrotliEncoderParams * params,BROTLI_BOOL one_shot,const size_t input_size,size_t * alloc_size)426*6777b538SAndroid Build Coastguard Worker static BROTLI_INLINE void HasherSize(const BrotliEncoderParams* params,
427*6777b538SAndroid Build Coastguard Worker BROTLI_BOOL one_shot, const size_t input_size, size_t* alloc_size) {
428*6777b538SAndroid Build Coastguard Worker switch (params->hasher.type) {
429*6777b538SAndroid Build Coastguard Worker #define SIZE_(N) \
430*6777b538SAndroid Build Coastguard Worker case N: \
431*6777b538SAndroid Build Coastguard Worker HashMemAllocInBytesH ## N(params, one_shot, input_size, alloc_size); \
432*6777b538SAndroid Build Coastguard Worker break;
433*6777b538SAndroid Build Coastguard Worker FOR_ALL_HASHERS(SIZE_)
434*6777b538SAndroid Build Coastguard Worker #undef SIZE_
435*6777b538SAndroid Build Coastguard Worker default:
436*6777b538SAndroid Build Coastguard Worker break;
437*6777b538SAndroid Build Coastguard Worker }
438*6777b538SAndroid Build Coastguard Worker }
439*6777b538SAndroid Build Coastguard Worker
HasherSetup(MemoryManager * m,Hasher * hasher,BrotliEncoderParams * params,const uint8_t * data,size_t position,size_t input_size,BROTLI_BOOL is_last)440*6777b538SAndroid Build Coastguard Worker static BROTLI_INLINE void HasherSetup(MemoryManager* m, Hasher* hasher,
441*6777b538SAndroid Build Coastguard Worker BrotliEncoderParams* params, const uint8_t* data, size_t position,
442*6777b538SAndroid Build Coastguard Worker size_t input_size, BROTLI_BOOL is_last) {
443*6777b538SAndroid Build Coastguard Worker BROTLI_BOOL one_shot = (position == 0 && is_last);
444*6777b538SAndroid Build Coastguard Worker if (!hasher->common.is_setup_) {
445*6777b538SAndroid Build Coastguard Worker size_t alloc_size[4] = {0};
446*6777b538SAndroid Build Coastguard Worker size_t i;
447*6777b538SAndroid Build Coastguard Worker ChooseHasher(params, ¶ms->hasher);
448*6777b538SAndroid Build Coastguard Worker hasher->common.params = params->hasher;
449*6777b538SAndroid Build Coastguard Worker hasher->common.dict_num_lookups = 0;
450*6777b538SAndroid Build Coastguard Worker hasher->common.dict_num_matches = 0;
451*6777b538SAndroid Build Coastguard Worker HasherSize(params, one_shot, input_size, alloc_size);
452*6777b538SAndroid Build Coastguard Worker for (i = 0; i < 4; ++i) {
453*6777b538SAndroid Build Coastguard Worker if (alloc_size[i] == 0) continue;
454*6777b538SAndroid Build Coastguard Worker hasher->common.extra[i] = BROTLI_ALLOC(m, uint8_t, alloc_size[i]);
455*6777b538SAndroid Build Coastguard Worker if (BROTLI_IS_OOM(m) || BROTLI_IS_NULL(hasher->common.extra[i])) return;
456*6777b538SAndroid Build Coastguard Worker }
457*6777b538SAndroid Build Coastguard Worker switch (hasher->common.params.type) {
458*6777b538SAndroid Build Coastguard Worker #define INITIALIZE_(N) \
459*6777b538SAndroid Build Coastguard Worker case N: \
460*6777b538SAndroid Build Coastguard Worker InitializeH ## N(&hasher->common, \
461*6777b538SAndroid Build Coastguard Worker &hasher->privat._H ## N, params); \
462*6777b538SAndroid Build Coastguard Worker break;
463*6777b538SAndroid Build Coastguard Worker FOR_ALL_HASHERS(INITIALIZE_);
464*6777b538SAndroid Build Coastguard Worker #undef INITIALIZE_
465*6777b538SAndroid Build Coastguard Worker default:
466*6777b538SAndroid Build Coastguard Worker break;
467*6777b538SAndroid Build Coastguard Worker }
468*6777b538SAndroid Build Coastguard Worker HasherReset(hasher);
469*6777b538SAndroid Build Coastguard Worker hasher->common.is_setup_ = BROTLI_TRUE;
470*6777b538SAndroid Build Coastguard Worker }
471*6777b538SAndroid Build Coastguard Worker
472*6777b538SAndroid Build Coastguard Worker if (!hasher->common.is_prepared_) {
473*6777b538SAndroid Build Coastguard Worker switch (hasher->common.params.type) {
474*6777b538SAndroid Build Coastguard Worker #define PREPARE_(N) \
475*6777b538SAndroid Build Coastguard Worker case N: \
476*6777b538SAndroid Build Coastguard Worker PrepareH ## N( \
477*6777b538SAndroid Build Coastguard Worker &hasher->privat._H ## N, \
478*6777b538SAndroid Build Coastguard Worker one_shot, input_size, data); \
479*6777b538SAndroid Build Coastguard Worker break;
480*6777b538SAndroid Build Coastguard Worker FOR_ALL_HASHERS(PREPARE_)
481*6777b538SAndroid Build Coastguard Worker #undef PREPARE_
482*6777b538SAndroid Build Coastguard Worker default: break;
483*6777b538SAndroid Build Coastguard Worker }
484*6777b538SAndroid Build Coastguard Worker hasher->common.is_prepared_ = BROTLI_TRUE;
485*6777b538SAndroid Build Coastguard Worker }
486*6777b538SAndroid Build Coastguard Worker }
487*6777b538SAndroid Build Coastguard Worker
InitOrStitchToPreviousBlock(MemoryManager * m,Hasher * hasher,const uint8_t * data,size_t mask,BrotliEncoderParams * params,size_t position,size_t input_size,BROTLI_BOOL is_last)488*6777b538SAndroid Build Coastguard Worker static BROTLI_INLINE void InitOrStitchToPreviousBlock(
489*6777b538SAndroid Build Coastguard Worker MemoryManager* m, Hasher* hasher, const uint8_t* data, size_t mask,
490*6777b538SAndroid Build Coastguard Worker BrotliEncoderParams* params, size_t position, size_t input_size,
491*6777b538SAndroid Build Coastguard Worker BROTLI_BOOL is_last) {
492*6777b538SAndroid Build Coastguard Worker HasherSetup(m, hasher, params, data, position, input_size, is_last);
493*6777b538SAndroid Build Coastguard Worker if (BROTLI_IS_OOM(m)) return;
494*6777b538SAndroid Build Coastguard Worker switch (hasher->common.params.type) {
495*6777b538SAndroid Build Coastguard Worker #define INIT_(N) \
496*6777b538SAndroid Build Coastguard Worker case N: \
497*6777b538SAndroid Build Coastguard Worker StitchToPreviousBlockH ## N( \
498*6777b538SAndroid Build Coastguard Worker &hasher->privat._H ## N, \
499*6777b538SAndroid Build Coastguard Worker input_size, position, data, mask); \
500*6777b538SAndroid Build Coastguard Worker break;
501*6777b538SAndroid Build Coastguard Worker FOR_ALL_HASHERS(INIT_)
502*6777b538SAndroid Build Coastguard Worker #undef INIT_
503*6777b538SAndroid Build Coastguard Worker default: break;
504*6777b538SAndroid Build Coastguard Worker }
505*6777b538SAndroid Build Coastguard Worker }
506*6777b538SAndroid Build Coastguard Worker
507*6777b538SAndroid Build Coastguard Worker /* NB: when seamless dictionary-ring-buffer copies are implemented, don't forget
508*6777b538SAndroid Build Coastguard Worker to add proper guards for non-zero-BROTLI_PARAM_STREAM_OFFSET. */
FindCompoundDictionaryMatch(const PreparedDictionary * self,const uint8_t * BROTLI_RESTRICT data,const size_t ring_buffer_mask,const int * BROTLI_RESTRICT distance_cache,const size_t cur_ix,const size_t max_length,const size_t distance_offset,const size_t max_distance,HasherSearchResult * BROTLI_RESTRICT out)509*6777b538SAndroid Build Coastguard Worker static BROTLI_INLINE void FindCompoundDictionaryMatch(
510*6777b538SAndroid Build Coastguard Worker const PreparedDictionary* self, const uint8_t* BROTLI_RESTRICT data,
511*6777b538SAndroid Build Coastguard Worker const size_t ring_buffer_mask, const int* BROTLI_RESTRICT distance_cache,
512*6777b538SAndroid Build Coastguard Worker const size_t cur_ix, const size_t max_length, const size_t distance_offset,
513*6777b538SAndroid Build Coastguard Worker const size_t max_distance, HasherSearchResult* BROTLI_RESTRICT out) {
514*6777b538SAndroid Build Coastguard Worker const uint32_t source_offset = self->source_offset;
515*6777b538SAndroid Build Coastguard Worker const uint32_t source_size = self->source_size;
516*6777b538SAndroid Build Coastguard Worker const size_t boundary = distance_offset - source_size;
517*6777b538SAndroid Build Coastguard Worker const uint32_t hash_bits = self->hash_bits;
518*6777b538SAndroid Build Coastguard Worker const uint32_t bucket_bits = self->bucket_bits;
519*6777b538SAndroid Build Coastguard Worker const uint32_t slot_bits = self->slot_bits;
520*6777b538SAndroid Build Coastguard Worker
521*6777b538SAndroid Build Coastguard Worker const uint32_t hash_shift = 64u - bucket_bits;
522*6777b538SAndroid Build Coastguard Worker const uint32_t slot_mask = (~((uint32_t)0U)) >> (32 - slot_bits);
523*6777b538SAndroid Build Coastguard Worker const uint64_t hash_mask = (~((uint64_t)0U)) >> (64 - hash_bits);
524*6777b538SAndroid Build Coastguard Worker
525*6777b538SAndroid Build Coastguard Worker const uint32_t* slot_offsets = (uint32_t*)(&self[1]);
526*6777b538SAndroid Build Coastguard Worker const uint16_t* heads = (uint16_t*)(&slot_offsets[1u << slot_bits]);
527*6777b538SAndroid Build Coastguard Worker const uint32_t* items = (uint32_t*)(&heads[1u << bucket_bits]);
528*6777b538SAndroid Build Coastguard Worker const uint8_t* source = (uint8_t*)(&items[source_offset]);
529*6777b538SAndroid Build Coastguard Worker
530*6777b538SAndroid Build Coastguard Worker const size_t cur_ix_masked = cur_ix & ring_buffer_mask;
531*6777b538SAndroid Build Coastguard Worker score_t best_score = out->score;
532*6777b538SAndroid Build Coastguard Worker size_t best_len = out->len;
533*6777b538SAndroid Build Coastguard Worker size_t i;
534*6777b538SAndroid Build Coastguard Worker const uint64_t h =
535*6777b538SAndroid Build Coastguard Worker (BROTLI_UNALIGNED_LOAD64LE(&data[cur_ix_masked]) & hash_mask) *
536*6777b538SAndroid Build Coastguard Worker kPreparedDictionaryHashMul64Long;
537*6777b538SAndroid Build Coastguard Worker const uint32_t key = (uint32_t)(h >> hash_shift);
538*6777b538SAndroid Build Coastguard Worker const uint32_t slot = key & slot_mask;
539*6777b538SAndroid Build Coastguard Worker const uint32_t head = heads[key];
540*6777b538SAndroid Build Coastguard Worker const uint32_t* BROTLI_RESTRICT chain = &items[slot_offsets[slot] + head];
541*6777b538SAndroid Build Coastguard Worker uint32_t item = (head == 0xFFFF) ? 1 : 0;
542*6777b538SAndroid Build Coastguard Worker for (i = 0; i < 4; ++i) {
543*6777b538SAndroid Build Coastguard Worker const size_t distance = (size_t)distance_cache[i];
544*6777b538SAndroid Build Coastguard Worker size_t offset;
545*6777b538SAndroid Build Coastguard Worker size_t limit;
546*6777b538SAndroid Build Coastguard Worker size_t len;
547*6777b538SAndroid Build Coastguard Worker if (distance <= boundary || distance > distance_offset) continue;
548*6777b538SAndroid Build Coastguard Worker offset = distance_offset - distance;
549*6777b538SAndroid Build Coastguard Worker limit = source_size - offset;
550*6777b538SAndroid Build Coastguard Worker limit = limit > max_length ? max_length : limit;
551*6777b538SAndroid Build Coastguard Worker len = FindMatchLengthWithLimit(&source[offset], &data[cur_ix_masked],
552*6777b538SAndroid Build Coastguard Worker limit);
553*6777b538SAndroid Build Coastguard Worker if (len >= 2) {
554*6777b538SAndroid Build Coastguard Worker score_t score = BackwardReferenceScoreUsingLastDistance(len);
555*6777b538SAndroid Build Coastguard Worker if (best_score < score) {
556*6777b538SAndroid Build Coastguard Worker if (i != 0) score -= BackwardReferencePenaltyUsingLastDistance(i);
557*6777b538SAndroid Build Coastguard Worker if (best_score < score) {
558*6777b538SAndroid Build Coastguard Worker best_score = score;
559*6777b538SAndroid Build Coastguard Worker if (len > best_len) best_len = len;
560*6777b538SAndroid Build Coastguard Worker out->len = len;
561*6777b538SAndroid Build Coastguard Worker out->len_code_delta = 0;
562*6777b538SAndroid Build Coastguard Worker out->distance = distance;
563*6777b538SAndroid Build Coastguard Worker out->score = best_score;
564*6777b538SAndroid Build Coastguard Worker }
565*6777b538SAndroid Build Coastguard Worker }
566*6777b538SAndroid Build Coastguard Worker }
567*6777b538SAndroid Build Coastguard Worker }
568*6777b538SAndroid Build Coastguard Worker while (item == 0) {
569*6777b538SAndroid Build Coastguard Worker size_t offset;
570*6777b538SAndroid Build Coastguard Worker size_t distance;
571*6777b538SAndroid Build Coastguard Worker size_t limit;
572*6777b538SAndroid Build Coastguard Worker item = *chain;
573*6777b538SAndroid Build Coastguard Worker chain++;
574*6777b538SAndroid Build Coastguard Worker offset = item & 0x7FFFFFFF;
575*6777b538SAndroid Build Coastguard Worker item &= 0x80000000;
576*6777b538SAndroid Build Coastguard Worker distance = distance_offset - offset;
577*6777b538SAndroid Build Coastguard Worker limit = source_size - offset;
578*6777b538SAndroid Build Coastguard Worker limit = (limit > max_length) ? max_length : limit;
579*6777b538SAndroid Build Coastguard Worker if (distance > max_distance) continue;
580*6777b538SAndroid Build Coastguard Worker if (cur_ix_masked + best_len > ring_buffer_mask ||
581*6777b538SAndroid Build Coastguard Worker best_len >= limit ||
582*6777b538SAndroid Build Coastguard Worker data[cur_ix_masked + best_len] != source[offset + best_len]) {
583*6777b538SAndroid Build Coastguard Worker continue;
584*6777b538SAndroid Build Coastguard Worker }
585*6777b538SAndroid Build Coastguard Worker {
586*6777b538SAndroid Build Coastguard Worker const size_t len = FindMatchLengthWithLimit(&source[offset],
587*6777b538SAndroid Build Coastguard Worker &data[cur_ix_masked],
588*6777b538SAndroid Build Coastguard Worker limit);
589*6777b538SAndroid Build Coastguard Worker if (len >= 4) {
590*6777b538SAndroid Build Coastguard Worker score_t score = BackwardReferenceScore(len, distance);
591*6777b538SAndroid Build Coastguard Worker if (best_score < score) {
592*6777b538SAndroid Build Coastguard Worker best_score = score;
593*6777b538SAndroid Build Coastguard Worker best_len = len;
594*6777b538SAndroid Build Coastguard Worker out->len = best_len;
595*6777b538SAndroid Build Coastguard Worker out->len_code_delta = 0;
596*6777b538SAndroid Build Coastguard Worker out->distance = distance;
597*6777b538SAndroid Build Coastguard Worker out->score = best_score;
598*6777b538SAndroid Build Coastguard Worker }
599*6777b538SAndroid Build Coastguard Worker }
600*6777b538SAndroid Build Coastguard Worker }
601*6777b538SAndroid Build Coastguard Worker }
602*6777b538SAndroid Build Coastguard Worker }
603*6777b538SAndroid Build Coastguard Worker
604*6777b538SAndroid Build Coastguard Worker /* NB: when seamless dictionary-ring-buffer copies are implemented, don't forget
605*6777b538SAndroid Build Coastguard Worker to add proper guards for non-zero-BROTLI_PARAM_STREAM_OFFSET. */
FindAllCompoundDictionaryMatches(const PreparedDictionary * self,const uint8_t * BROTLI_RESTRICT data,const size_t ring_buffer_mask,const size_t cur_ix,const size_t min_length,const size_t max_length,const size_t distance_offset,const size_t max_distance,BackwardMatch * matches,size_t match_limit)606*6777b538SAndroid Build Coastguard Worker static BROTLI_INLINE size_t FindAllCompoundDictionaryMatches(
607*6777b538SAndroid Build Coastguard Worker const PreparedDictionary* self, const uint8_t* BROTLI_RESTRICT data,
608*6777b538SAndroid Build Coastguard Worker const size_t ring_buffer_mask, const size_t cur_ix, const size_t min_length,
609*6777b538SAndroid Build Coastguard Worker const size_t max_length, const size_t distance_offset,
610*6777b538SAndroid Build Coastguard Worker const size_t max_distance, BackwardMatch* matches, size_t match_limit) {
611*6777b538SAndroid Build Coastguard Worker const uint32_t source_offset = self->source_offset;
612*6777b538SAndroid Build Coastguard Worker const uint32_t source_size = self->source_size;
613*6777b538SAndroid Build Coastguard Worker const uint32_t hash_bits = self->hash_bits;
614*6777b538SAndroid Build Coastguard Worker const uint32_t bucket_bits = self->bucket_bits;
615*6777b538SAndroid Build Coastguard Worker const uint32_t slot_bits = self->slot_bits;
616*6777b538SAndroid Build Coastguard Worker
617*6777b538SAndroid Build Coastguard Worker const uint32_t hash_shift = 64u - bucket_bits;
618*6777b538SAndroid Build Coastguard Worker const uint32_t slot_mask = (~((uint32_t)0U)) >> (32 - slot_bits);
619*6777b538SAndroid Build Coastguard Worker const uint64_t hash_mask = (~((uint64_t)0U)) >> (64 - hash_bits);
620*6777b538SAndroid Build Coastguard Worker
621*6777b538SAndroid Build Coastguard Worker const uint32_t* slot_offsets = (uint32_t*)(&self[1]);
622*6777b538SAndroid Build Coastguard Worker const uint16_t* heads = (uint16_t*)(&slot_offsets[1u << slot_bits]);
623*6777b538SAndroid Build Coastguard Worker const uint32_t* items = (uint32_t*)(&heads[1u << bucket_bits]);
624*6777b538SAndroid Build Coastguard Worker const uint8_t* source = (uint8_t*)(&items[source_offset]);
625*6777b538SAndroid Build Coastguard Worker
626*6777b538SAndroid Build Coastguard Worker const size_t cur_ix_masked = cur_ix & ring_buffer_mask;
627*6777b538SAndroid Build Coastguard Worker size_t best_len = min_length;
628*6777b538SAndroid Build Coastguard Worker const uint64_t h =
629*6777b538SAndroid Build Coastguard Worker (BROTLI_UNALIGNED_LOAD64LE(&data[cur_ix_masked]) & hash_mask) *
630*6777b538SAndroid Build Coastguard Worker kPreparedDictionaryHashMul64Long;
631*6777b538SAndroid Build Coastguard Worker const uint32_t key = (uint32_t)(h >> hash_shift);
632*6777b538SAndroid Build Coastguard Worker const uint32_t slot = key & slot_mask;
633*6777b538SAndroid Build Coastguard Worker const uint32_t head = heads[key];
634*6777b538SAndroid Build Coastguard Worker const uint32_t* BROTLI_RESTRICT chain = &items[slot_offsets[slot] + head];
635*6777b538SAndroid Build Coastguard Worker uint32_t item = (head == 0xFFFF) ? 1 : 0;
636*6777b538SAndroid Build Coastguard Worker size_t found = 0;
637*6777b538SAndroid Build Coastguard Worker while (item == 0) {
638*6777b538SAndroid Build Coastguard Worker size_t offset;
639*6777b538SAndroid Build Coastguard Worker size_t distance;
640*6777b538SAndroid Build Coastguard Worker size_t limit;
641*6777b538SAndroid Build Coastguard Worker size_t len;
642*6777b538SAndroid Build Coastguard Worker item = *chain;
643*6777b538SAndroid Build Coastguard Worker chain++;
644*6777b538SAndroid Build Coastguard Worker offset = item & 0x7FFFFFFF;
645*6777b538SAndroid Build Coastguard Worker item &= 0x80000000;
646*6777b538SAndroid Build Coastguard Worker distance = distance_offset - offset;
647*6777b538SAndroid Build Coastguard Worker limit = source_size - offset;
648*6777b538SAndroid Build Coastguard Worker limit = (limit > max_length) ? max_length : limit;
649*6777b538SAndroid Build Coastguard Worker if (distance > max_distance) continue;
650*6777b538SAndroid Build Coastguard Worker if (cur_ix_masked + best_len > ring_buffer_mask ||
651*6777b538SAndroid Build Coastguard Worker best_len >= limit ||
652*6777b538SAndroid Build Coastguard Worker data[cur_ix_masked + best_len] != source[offset + best_len]) {
653*6777b538SAndroid Build Coastguard Worker continue;
654*6777b538SAndroid Build Coastguard Worker }
655*6777b538SAndroid Build Coastguard Worker len = FindMatchLengthWithLimit(
656*6777b538SAndroid Build Coastguard Worker &source[offset], &data[cur_ix_masked], limit);
657*6777b538SAndroid Build Coastguard Worker if (len > best_len) {
658*6777b538SAndroid Build Coastguard Worker best_len = len;
659*6777b538SAndroid Build Coastguard Worker InitBackwardMatch(matches++, distance, len);
660*6777b538SAndroid Build Coastguard Worker found++;
661*6777b538SAndroid Build Coastguard Worker if (found == match_limit) break;
662*6777b538SAndroid Build Coastguard Worker }
663*6777b538SAndroid Build Coastguard Worker }
664*6777b538SAndroid Build Coastguard Worker return found;
665*6777b538SAndroid Build Coastguard Worker }
666*6777b538SAndroid Build Coastguard Worker
LookupCompoundDictionaryMatch(const CompoundDictionary * addon,const uint8_t * BROTLI_RESTRICT data,const size_t ring_buffer_mask,const int * BROTLI_RESTRICT distance_cache,const size_t cur_ix,const size_t max_length,const size_t max_ring_buffer_distance,const size_t max_distance,HasherSearchResult * sr)667*6777b538SAndroid Build Coastguard Worker static BROTLI_INLINE void LookupCompoundDictionaryMatch(
668*6777b538SAndroid Build Coastguard Worker const CompoundDictionary* addon, const uint8_t* BROTLI_RESTRICT data,
669*6777b538SAndroid Build Coastguard Worker const size_t ring_buffer_mask, const int* BROTLI_RESTRICT distance_cache,
670*6777b538SAndroid Build Coastguard Worker const size_t cur_ix, const size_t max_length,
671*6777b538SAndroid Build Coastguard Worker const size_t max_ring_buffer_distance, const size_t max_distance,
672*6777b538SAndroid Build Coastguard Worker HasherSearchResult* sr) {
673*6777b538SAndroid Build Coastguard Worker size_t base_offset = max_ring_buffer_distance + 1 + addon->total_size - 1;
674*6777b538SAndroid Build Coastguard Worker size_t d;
675*6777b538SAndroid Build Coastguard Worker for (d = 0; d < addon->num_chunks; ++d) {
676*6777b538SAndroid Build Coastguard Worker /* Only one prepared dictionary type is currently supported. */
677*6777b538SAndroid Build Coastguard Worker FindCompoundDictionaryMatch(
678*6777b538SAndroid Build Coastguard Worker (const PreparedDictionary*)addon->chunks[d], data, ring_buffer_mask,
679*6777b538SAndroid Build Coastguard Worker distance_cache, cur_ix, max_length,
680*6777b538SAndroid Build Coastguard Worker base_offset - addon->chunk_offsets[d], max_distance, sr);
681*6777b538SAndroid Build Coastguard Worker }
682*6777b538SAndroid Build Coastguard Worker }
683*6777b538SAndroid Build Coastguard Worker
LookupAllCompoundDictionaryMatches(const CompoundDictionary * addon,const uint8_t * BROTLI_RESTRICT data,const size_t ring_buffer_mask,const size_t cur_ix,size_t min_length,const size_t max_length,const size_t max_ring_buffer_distance,const size_t max_distance,BackwardMatch * matches,size_t match_limit)684*6777b538SAndroid Build Coastguard Worker static BROTLI_INLINE size_t LookupAllCompoundDictionaryMatches(
685*6777b538SAndroid Build Coastguard Worker const CompoundDictionary* addon, const uint8_t* BROTLI_RESTRICT data,
686*6777b538SAndroid Build Coastguard Worker const size_t ring_buffer_mask, const size_t cur_ix, size_t min_length,
687*6777b538SAndroid Build Coastguard Worker const size_t max_length, const size_t max_ring_buffer_distance,
688*6777b538SAndroid Build Coastguard Worker const size_t max_distance, BackwardMatch* matches,
689*6777b538SAndroid Build Coastguard Worker size_t match_limit) {
690*6777b538SAndroid Build Coastguard Worker size_t base_offset = max_ring_buffer_distance + 1 + addon->total_size - 1;
691*6777b538SAndroid Build Coastguard Worker size_t d;
692*6777b538SAndroid Build Coastguard Worker size_t total_found = 0;
693*6777b538SAndroid Build Coastguard Worker for (d = 0; d < addon->num_chunks; ++d) {
694*6777b538SAndroid Build Coastguard Worker /* Only one prepared dictionary type is currently supported. */
695*6777b538SAndroid Build Coastguard Worker total_found += FindAllCompoundDictionaryMatches(
696*6777b538SAndroid Build Coastguard Worker (const PreparedDictionary*)addon->chunks[d], data, ring_buffer_mask,
697*6777b538SAndroid Build Coastguard Worker cur_ix, min_length, max_length, base_offset - addon->chunk_offsets[d],
698*6777b538SAndroid Build Coastguard Worker max_distance, matches + total_found, match_limit - total_found);
699*6777b538SAndroid Build Coastguard Worker if (total_found == match_limit) break;
700*6777b538SAndroid Build Coastguard Worker if (total_found > 0) {
701*6777b538SAndroid Build Coastguard Worker min_length = BackwardMatchLength(&matches[total_found - 1]);
702*6777b538SAndroid Build Coastguard Worker }
703*6777b538SAndroid Build Coastguard Worker }
704*6777b538SAndroid Build Coastguard Worker return total_found;
705*6777b538SAndroid Build Coastguard Worker }
706*6777b538SAndroid Build Coastguard Worker
707*6777b538SAndroid Build Coastguard Worker #if defined(__cplusplus) || defined(c_plusplus)
708*6777b538SAndroid Build Coastguard Worker } /* extern "C" */
709*6777b538SAndroid Build Coastguard Worker #endif
710*6777b538SAndroid Build Coastguard Worker
711*6777b538SAndroid Build Coastguard Worker #endif /* BROTLI_ENC_HASH_H_ */
712