Coverage Report

Created: 2026-04-09 07:06

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/ghostpdl/brotli/c/enc/block_splitter_inc.h
Line
Count
Source
1
/* NOLINT(build/header_guard) */
2
/* Copyright 2013 Google Inc. All Rights Reserved.
3
4
   Distributed under MIT license.
5
   See file LICENSE for detail or copy at https://opensource.org/licenses/MIT
6
*/
7
8
/* template parameters: FN, DataType */
9
10
0
#define HistogramType FN(Histogram)
11
12
static void FN(InitialEntropyCodes)(const DataType* data, size_t length,
13
                                    size_t stride,
14
                                    size_t num_histograms,
15
0
                                    HistogramType* histograms) {
16
0
  uint32_t seed = 7;
17
0
  size_t block_length = length / num_histograms;
18
0
  size_t i;
19
0
  FN(ClearHistograms)(histograms, num_histograms);
20
0
  for (i = 0; i < num_histograms; ++i) {
21
0
    size_t pos = length * i / num_histograms;
22
0
    if (i != 0) {
23
0
      pos += MyRand(&seed) % block_length;
24
0
    }
25
0
    if (pos + stride >= length) {
26
0
      pos = length - stride - 1;
27
0
    }
28
0
    FN(HistogramAddVector)(&histograms[i], data + pos, stride);
29
0
  }
30
0
}
Unexecuted instantiation: block_splitter.c:InitialEntropyCodesLiteral
Unexecuted instantiation: block_splitter.c:InitialEntropyCodesCommand
Unexecuted instantiation: block_splitter.c:InitialEntropyCodesDistance
31
32
static void FN(RandomSample)(uint32_t* seed,
33
                             const DataType* data,
34
                             size_t length,
35
                             size_t stride,
36
0
                             HistogramType* sample) {
37
0
  size_t pos = 0;
38
0
  if (stride >= length) {
39
0
    stride = length;
40
0
  } else {
41
0
    pos = MyRand(seed) % (length - stride + 1);
42
0
  }
43
0
  FN(HistogramAddVector)(sample, data + pos, stride);
44
0
}
Unexecuted instantiation: block_splitter.c:RandomSampleLiteral
Unexecuted instantiation: block_splitter.c:RandomSampleCommand
Unexecuted instantiation: block_splitter.c:RandomSampleDistance
45
46
static void FN(RefineEntropyCodes)(const DataType* data, size_t length,
47
                                   size_t stride,
48
                                   size_t num_histograms,
49
                                   HistogramType* histograms,
50
0
                                   HistogramType* tmp) {
51
0
  size_t iters =
52
0
      kIterMulForRefining * length / stride + kMinItersForRefining;
53
0
  uint32_t seed = 7;
54
0
  size_t iter;
55
0
  iters = ((iters + num_histograms - 1) / num_histograms) * num_histograms;
56
0
  for (iter = 0; iter < iters; ++iter) {
57
0
    FN(HistogramClear)(tmp);
58
0
    FN(RandomSample)(&seed, data, length, stride, tmp);
59
0
    FN(HistogramAddHistogram)(&histograms[iter % num_histograms], tmp);
60
0
  }
61
0
}
Unexecuted instantiation: block_splitter.c:RefineEntropyCodesLiteral
Unexecuted instantiation: block_splitter.c:RefineEntropyCodesCommand
Unexecuted instantiation: block_splitter.c:RefineEntropyCodesDistance
62
63
/* Assigns a block id from the range [0, num_histograms) to each data element
64
   in data[0..length) and fills in block_id[0..length) with the assigned values.
65
   Returns the number of blocks, i.e. one plus the number of block switches. */
66
static size_t FN(FindBlocks)(const DataType* data, const size_t length,
67
                             const double block_switch_bitcost,
68
                             const size_t num_histograms,
69
                             const HistogramType* histograms,
70
                             double* insert_cost,
71
                             double* cost,
72
                             uint8_t* switch_signal,
73
0
                             uint8_t* block_id) {
74
0
  const size_t alphabet_size = FN(HistogramDataSize)();
75
0
  const size_t bitmap_len = (num_histograms + 7) >> 3;
76
0
  size_t num_blocks = 1;
77
0
  size_t byte_ix;
78
0
  size_t i;
79
0
  size_t j;
80
0
  BROTLI_DCHECK(num_histograms <= 256);
81
82
  /* Trivial case: single histogram -> single block type. */
83
0
  if (num_histograms <= 1) {
84
0
    for (i = 0; i < length; ++i) {
85
0
      block_id[i] = 0;
86
0
    }
87
0
    return 1;
88
0
  }
89
90
  /* Fill bitcost for each symbol of all histograms.
91
   * Non-existing symbol cost: 2 + log2(total_count).
92
   * Regular symbol cost: -log2(symbol_count / total_count). */
93
0
  memset(insert_cost, 0,
94
0
         sizeof(insert_cost[0]) * alphabet_size * num_histograms);
95
0
  for (i = 0; i < num_histograms; ++i) {
96
0
    insert_cost[i] = FastLog2((uint32_t)histograms[i].total_count_);
97
0
  }
98
0
  for (i = alphabet_size; i != 0;) {
99
    /* Reverse order to use the 0-th row as a temporary storage. */
100
0
    --i;
101
0
    for (j = 0; j < num_histograms; ++j) {
102
0
      insert_cost[i * num_histograms + j] =
103
0
          insert_cost[j] - BitCost(histograms[j].data_[i]);
104
0
    }
105
0
  }
106
107
  /* After each iteration of this loop, cost[k] will contain the difference
108
     between the minimum cost of arriving at the current byte position using
109
     entropy code k, and the minimum cost of arriving at the current byte
110
     position. This difference is capped at the block switch cost, and if it
111
     reaches block switch cost, it means that when we trace back from the last
112
     position, we need to switch here. */
113
0
  memset(cost, 0, sizeof(cost[0]) * num_histograms);
114
0
  memset(switch_signal, 0, sizeof(switch_signal[0]) * length * bitmap_len);
115
0
  for (byte_ix = 0; byte_ix < length; ++byte_ix) {
116
0
    size_t ix = byte_ix * bitmap_len;
117
0
    size_t symbol = data[byte_ix];
118
0
    size_t insert_cost_ix = symbol * num_histograms;
119
0
    double min_cost = 1e99;
120
0
    double block_switch_cost = block_switch_bitcost;
121
0
    static const size_t prologue_length = 2000;
122
0
    static const double multiplier = 0.07 / 2000;
123
0
    size_t k;
124
0
    for (k = 0; k < num_histograms; ++k) {
125
      /* We are coding the symbol with entropy code k. */
126
0
      cost[k] += insert_cost[insert_cost_ix + k];
127
0
      if (cost[k] < min_cost) {
128
0
        min_cost = cost[k];
129
0
        block_id[byte_ix] = (uint8_t)k;
130
0
      }
131
0
    }
132
    /* More blocks for the beginning. */
133
0
    if (byte_ix < prologue_length) {
134
0
      block_switch_cost *= 0.77 + multiplier * (double)byte_ix;
135
0
    }
136
0
    for (k = 0; k < num_histograms; ++k) {
137
0
      cost[k] -= min_cost;
138
0
      if (cost[k] >= block_switch_cost) {
139
0
        const uint8_t mask = (uint8_t)(1u << (k & 7));
140
0
        cost[k] = block_switch_cost;
141
0
        BROTLI_DCHECK((k >> 3) < bitmap_len);
142
0
        switch_signal[ix + (k >> 3)] |= mask;
143
0
      }
144
0
    }
145
0
  }
146
147
0
  byte_ix = length - 1;
148
0
  {  /* Trace back from the last position and switch at the marked places. */
149
0
    size_t ix = byte_ix * bitmap_len;
150
0
    uint8_t cur_id = block_id[byte_ix];
151
0
    while (byte_ix > 0) {
152
0
      const uint8_t mask = (uint8_t)(1u << (cur_id & 7));
153
0
      BROTLI_DCHECK(((size_t)cur_id >> 3) < bitmap_len);
154
0
      --byte_ix;
155
0
      ix -= bitmap_len;
156
0
      if (switch_signal[ix + (cur_id >> 3)] & mask) {
157
0
        if (cur_id != block_id[byte_ix]) {
158
0
          cur_id = block_id[byte_ix];
159
0
          ++num_blocks;
160
0
        }
161
0
      }
162
0
      block_id[byte_ix] = cur_id;
163
0
    }
164
0
  }
165
0
  return num_blocks;
166
0
}
Unexecuted instantiation: block_splitter.c:FindBlocksLiteral
Unexecuted instantiation: block_splitter.c:FindBlocksCommand
Unexecuted instantiation: block_splitter.c:FindBlocksDistance
167
168
static size_t FN(RemapBlockIds)(uint8_t* block_ids, const size_t length,
169
0
                                uint16_t* new_id, const size_t num_histograms) {
170
0
  static const uint16_t kInvalidId = 256;
171
0
  uint16_t next_id = 0;
172
0
  size_t i;
173
0
  for (i = 0; i < num_histograms; ++i) {
174
0
    new_id[i] = kInvalidId;
175
0
  }
176
0
  for (i = 0; i < length; ++i) {
177
0
    BROTLI_DCHECK(block_ids[i] < num_histograms);
178
0
    if (new_id[block_ids[i]] == kInvalidId) {
179
0
      new_id[block_ids[i]] = next_id++;
180
0
    }
181
0
  }
182
0
  for (i = 0; i < length; ++i) {
183
0
    block_ids[i] = (uint8_t)new_id[block_ids[i]];
184
0
    BROTLI_DCHECK(block_ids[i] < num_histograms);
185
0
  }
186
0
  BROTLI_DCHECK(next_id <= num_histograms);
187
0
  return next_id;
188
0
}
Unexecuted instantiation: block_splitter.c:RemapBlockIdsLiteral
Unexecuted instantiation: block_splitter.c:RemapBlockIdsCommand
Unexecuted instantiation: block_splitter.c:RemapBlockIdsDistance
189
190
static void FN(BuildBlockHistograms)(const DataType* data, const size_t length,
191
                                     const uint8_t* block_ids,
192
                                     const size_t num_histograms,
193
0
                                     HistogramType* histograms) {
194
0
  size_t i;
195
0
  FN(ClearHistograms)(histograms, num_histograms);
196
0
  for (i = 0; i < length; ++i) {
197
0
    FN(HistogramAdd)(&histograms[block_ids[i]], data[i]);
198
0
  }
199
0
}
Unexecuted instantiation: block_splitter.c:BuildBlockHistogramsLiteral
Unexecuted instantiation: block_splitter.c:BuildBlockHistogramsCommand
Unexecuted instantiation: block_splitter.c:BuildBlockHistogramsDistance
200
201
/* Given the initial partitioning build partitioning with limited number
202
 * of histograms (and block types). */
203
static void FN(ClusterBlocks)(MemoryManager* m,
204
                              const DataType* data, const size_t length,
205
                              const size_t num_blocks,
206
                              uint8_t* block_ids,
207
0
                              BlockSplit* split) {
208
0
  uint32_t* histogram_symbols = BROTLI_ALLOC(m, uint32_t, num_blocks);
209
0
  uint32_t* u32 =
210
0
      BROTLI_ALLOC(m, uint32_t, num_blocks + 4 * HISTOGRAMS_PER_BATCH);
211
0
  const size_t expected_num_clusters = CLUSTERS_PER_BATCH *
212
0
      (num_blocks + HISTOGRAMS_PER_BATCH - 1) / HISTOGRAMS_PER_BATCH;
213
0
  size_t all_histograms_size = 0;
214
0
  size_t all_histograms_capacity = expected_num_clusters;
215
0
  HistogramType* all_histograms =
216
0
      BROTLI_ALLOC(m, HistogramType, all_histograms_capacity);
217
0
  size_t cluster_size_size = 0;
218
0
  size_t cluster_size_capacity = expected_num_clusters;
219
0
  uint32_t* cluster_size = BROTLI_ALLOC(m, uint32_t, cluster_size_capacity);
220
0
  size_t num_clusters = 0;
221
0
  HistogramType* histograms = BROTLI_ALLOC(m, HistogramType,
222
0
      BROTLI_MIN(size_t, num_blocks, HISTOGRAMS_PER_BATCH));
223
0
  size_t max_num_pairs =
224
0
      HISTOGRAMS_PER_BATCH * HISTOGRAMS_PER_BATCH / 2;
225
0
  size_t pairs_capacity = max_num_pairs + 1;
226
0
  HistogramPair* pairs = BROTLI_ALLOC(m, HistogramPair, pairs_capacity);
227
0
  size_t pos = 0;
228
0
  uint32_t* clusters;
229
0
  size_t num_final_clusters;
230
0
  static const uint32_t kInvalidIndex = BROTLI_UINT32_MAX;
231
0
  uint32_t* new_index;
232
0
  size_t i;
233
0
  uint32_t* BROTLI_RESTRICT const sizes =
234
0
      u32 ? (u32 + 0 * HISTOGRAMS_PER_BATCH) : NULL;
235
0
  uint32_t* BROTLI_RESTRICT const new_clusters =
236
0
      u32 ? (u32 + 1 * HISTOGRAMS_PER_BATCH) : NULL;
237
0
  uint32_t* BROTLI_RESTRICT const symbols =
238
0
      u32 ? (u32 + 2 * HISTOGRAMS_PER_BATCH) : NULL;
239
0
  uint32_t* BROTLI_RESTRICT const remap =
240
0
      u32 ? (u32 + 3 * HISTOGRAMS_PER_BATCH) : NULL;
241
0
  uint32_t* BROTLI_RESTRICT const block_lengths =
242
0
      u32 ? (u32 + 4 * HISTOGRAMS_PER_BATCH) : NULL;
243
  /* TODO(eustas): move to arena? */
244
0
  HistogramType* tmp = BROTLI_ALLOC(m, HistogramType, 2);
245
246
0
  if (BROTLI_IS_OOM(m) || BROTLI_IS_NULL(histogram_symbols) ||
247
0
      BROTLI_IS_NULL(u32) || BROTLI_IS_NULL(all_histograms) ||
248
0
      BROTLI_IS_NULL(cluster_size) || BROTLI_IS_NULL(histograms) ||
249
0
      BROTLI_IS_NULL(pairs) || BROTLI_IS_NULL(tmp)) {
250
0
    return;
251
0
  }
252
253
0
  memset(u32, 0, (num_blocks + 4 * HISTOGRAMS_PER_BATCH) * sizeof(uint32_t));
254
255
  /* Calculate block lengths (convert repeating values -> series length). */
256
0
  {
257
0
    size_t block_idx = 0;
258
0
    for (i = 0; i < length; ++i) {
259
0
      BROTLI_DCHECK(block_idx < num_blocks);
260
0
      ++block_lengths[block_idx];
261
0
      if (i + 1 == length || block_ids[i] != block_ids[i + 1]) {
262
0
        ++block_idx;
263
0
      }
264
0
    }
265
0
    BROTLI_DCHECK(block_idx == num_blocks);
266
0
  }
267
268
  /* Pre-cluster blocks (cluster batches). */
269
0
  for (i = 0; i < num_blocks; i += HISTOGRAMS_PER_BATCH) {
270
0
    const size_t num_to_combine =
271
0
        BROTLI_MIN(size_t, num_blocks - i, HISTOGRAMS_PER_BATCH);
272
0
    size_t num_new_clusters;
273
0
    size_t j;
274
0
    for (j = 0; j < num_to_combine; ++j) {
275
0
      size_t k;
276
0
      size_t block_length = block_lengths[i + j];
277
0
      FN(HistogramClear)(&histograms[j]);
278
0
      for (k = 0; k < block_length; ++k) {
279
0
        FN(HistogramAdd)(&histograms[j], data[pos++]);
280
0
      }
281
0
      histograms[j].bit_cost_ = FN(BrotliPopulationCost)(&histograms[j]);
282
0
      new_clusters[j] = (uint32_t)j;
283
0
      symbols[j] = (uint32_t)j;
284
0
      sizes[j] = 1;
285
0
    }
286
0
    num_new_clusters = FN(BrotliHistogramCombine)(
287
0
        histograms, tmp, sizes, symbols, new_clusters, pairs, num_to_combine,
288
0
        num_to_combine, HISTOGRAMS_PER_BATCH, max_num_pairs);
289
0
    BROTLI_ENSURE_CAPACITY(m, HistogramType, all_histograms,
290
0
        all_histograms_capacity, all_histograms_size + num_new_clusters);
291
0
    BROTLI_ENSURE_CAPACITY(m, uint32_t, cluster_size,
292
0
        cluster_size_capacity, cluster_size_size + num_new_clusters);
293
0
    if (BROTLI_IS_OOM(m)) return;
294
0
    for (j = 0; j < num_new_clusters; ++j) {
295
0
      all_histograms[all_histograms_size++] = histograms[new_clusters[j]];
296
0
      cluster_size[cluster_size_size++] = sizes[new_clusters[j]];
297
0
      remap[new_clusters[j]] = (uint32_t)j;
298
0
    }
299
0
    for (j = 0; j < num_to_combine; ++j) {
300
0
      histogram_symbols[i + j] = (uint32_t)num_clusters + remap[symbols[j]];
301
0
    }
302
0
    num_clusters += num_new_clusters;
303
0
    BROTLI_DCHECK(num_clusters == cluster_size_size);
304
0
    BROTLI_DCHECK(num_clusters == all_histograms_size);
305
0
  }
306
0
  BROTLI_FREE(m, histograms);
307
308
  /* Final clustering. */
309
0
  max_num_pairs =
310
0
      BROTLI_MIN(size_t, 64 * num_clusters, (num_clusters / 2) * num_clusters);
311
0
  if (pairs_capacity < max_num_pairs + 1) {
312
0
    BROTLI_FREE(m, pairs);
313
0
    pairs = BROTLI_ALLOC(m, HistogramPair, max_num_pairs + 1);
314
0
    if (BROTLI_IS_OOM(m) || BROTLI_IS_NULL(pairs)) return;
315
0
  }
316
0
  clusters = BROTLI_ALLOC(m, uint32_t, num_clusters);
317
0
  if (BROTLI_IS_OOM(m) || BROTLI_IS_NULL(clusters)) return;
318
0
  for (i = 0; i < num_clusters; ++i) {
319
0
    clusters[i] = (uint32_t)i;
320
0
  }
321
0
  num_final_clusters = FN(BrotliHistogramCombine)(
322
0
      all_histograms, tmp, cluster_size, histogram_symbols, clusters, pairs,
323
0
      num_clusters, num_blocks, BROTLI_MAX_NUMBER_OF_BLOCK_TYPES,
324
0
      max_num_pairs);
325
0
  BROTLI_FREE(m, pairs);
326
0
  BROTLI_FREE(m, cluster_size);
327
328
  /* Assign blocks to final histograms. */
329
0
  new_index = BROTLI_ALLOC(m, uint32_t, num_clusters);
330
0
  if (BROTLI_IS_OOM(m) || BROTLI_IS_NULL(new_index)) return;
331
0
  for (i = 0; i < num_clusters; ++i) new_index[i] = kInvalidIndex;
332
0
  pos = 0;
333
0
  {
334
0
    uint32_t next_index = 0;
335
0
    for (i = 0; i < num_blocks; ++i) {
336
0
      size_t j;
337
0
      uint32_t best_out;
338
0
      double best_bits;
339
0
      FN(HistogramClear)(tmp);
340
0
      for (j = 0; j < block_lengths[i]; ++j) {
341
0
        FN(HistogramAdd)(tmp, data[pos++]);
342
0
      }
343
      /* Among equally good histograms prefer last used. */
344
      /* TODO(eustas): should we give a block-switch discount here? */
345
0
      best_out = (i == 0) ? histogram_symbols[0] : histogram_symbols[i - 1];
346
0
      best_bits = FN(BrotliHistogramBitCostDistance)(
347
0
          tmp, &all_histograms[best_out], tmp + 1);
348
0
      for (j = 0; j < num_final_clusters; ++j) {
349
0
        const double cur_bits = FN(BrotliHistogramBitCostDistance)(
350
0
            tmp, &all_histograms[clusters[j]], tmp + 1);
351
0
        if (cur_bits < best_bits) {
352
0
          best_bits = cur_bits;
353
0
          best_out = clusters[j];
354
0
        }
355
0
      }
356
0
      histogram_symbols[i] = best_out;
357
0
      if (new_index[best_out] == kInvalidIndex) {
358
0
        new_index[best_out] = next_index++;
359
0
      }
360
0
    }
361
0
  }
362
0
  BROTLI_FREE(m, tmp);
363
0
  BROTLI_FREE(m, clusters);
364
0
  BROTLI_FREE(m, all_histograms);
365
0
  BROTLI_ENSURE_CAPACITY(
366
0
      m, uint8_t, split->types, split->types_alloc_size, num_blocks);
367
0
  BROTLI_ENSURE_CAPACITY(
368
0
      m, uint32_t, split->lengths, split->lengths_alloc_size, num_blocks);
369
0
  if (BROTLI_IS_OOM(m)) return;
370
371
  /* Rewrite final assignment to block-split. There might be less blocks
372
   * than |num_blocks| due to clustering. */
373
0
  {
374
0
    uint32_t cur_length = 0;
375
0
    size_t block_idx = 0;
376
0
    uint8_t max_type = 0;
377
0
    for (i = 0; i < num_blocks; ++i) {
378
0
      cur_length += block_lengths[i];
379
0
      if (i + 1 == num_blocks ||
380
0
          histogram_symbols[i] != histogram_symbols[i + 1]) {
381
0
        const uint8_t id = (uint8_t)new_index[histogram_symbols[i]];
382
0
        split->types[block_idx] = id;
383
0
        split->lengths[block_idx] = cur_length;
384
0
        max_type = BROTLI_MAX(uint8_t, max_type, id);
385
0
        cur_length = 0;
386
0
        ++block_idx;
387
0
      }
388
0
    }
389
0
    split->num_blocks = block_idx;
390
0
    split->num_types = (size_t)max_type + 1;
391
0
  }
392
0
  BROTLI_FREE(m, new_index);
393
0
  BROTLI_FREE(m, u32);
394
0
  BROTLI_FREE(m, histogram_symbols);
395
0
}
Unexecuted instantiation: block_splitter.c:ClusterBlocksLiteral
Unexecuted instantiation: block_splitter.c:ClusterBlocksCommand
Unexecuted instantiation: block_splitter.c:ClusterBlocksDistance
396
397
/* Create BlockSplit (partitioning) given the limits, estimates and "effort"
398
 * parameters.
399
 *
400
 * NB: max_histograms is often less than number of histograms allowed by format;
401
 *     this is done intentionally, to save some "space" for context-aware
402
 *     clustering (here entropy is estimated for context-free symbols). */
403
static void FN(SplitByteVector)(MemoryManager* m,
404
                                const DataType* data, const size_t length,
405
                                const size_t symbols_per_histogram,
406
                                const size_t max_histograms,
407
                                const size_t sampling_stride_length,
408
                                const double block_switch_cost,
409
                                const BrotliEncoderParams* params,
410
0
                                BlockSplit* split) {
411
0
  const size_t data_size = FN(HistogramDataSize)();
412
0
  HistogramType* histograms;
413
0
  HistogramType* tmp;
414
  /* Calculate number of histograms; initial estimate is one histogram per
415
   * specified amount of symbols; however, this value is capped. */
416
0
  size_t num_histograms = length / symbols_per_histogram + 1;
417
0
  if (num_histograms > max_histograms) {
418
0
    num_histograms = max_histograms;
419
0
  }
420
421
  /* Corner case: no input. */
422
0
  if (length == 0) {
423
0
    split->num_types = 1;
424
0
    return;
425
0
  }
426
427
0
  if (length < kMinLengthForBlockSplitting) {
428
0
    BROTLI_ENSURE_CAPACITY(m, uint8_t,
429
0
        split->types, split->types_alloc_size, split->num_blocks + 1);
430
0
    BROTLI_ENSURE_CAPACITY(m, uint32_t,
431
0
        split->lengths, split->lengths_alloc_size, split->num_blocks + 1);
432
0
    if (BROTLI_IS_OOM(m)) return;
433
0
    split->num_types = 1;
434
0
    split->types[split->num_blocks] = 0;
435
0
    split->lengths[split->num_blocks] = (uint32_t)length;
436
0
    split->num_blocks++;
437
0
    return;
438
0
  }
439
0
  histograms = BROTLI_ALLOC(m, HistogramType, num_histograms + 1);
440
0
  tmp = histograms + num_histograms;
441
0
  if (BROTLI_IS_OOM(m) || BROTLI_IS_NULL(histograms)) return;
442
  /* Find good entropy codes. */
443
0
  FN(InitialEntropyCodes)(data, length,
444
0
                          sampling_stride_length,
445
0
                          num_histograms, histograms);
446
0
  FN(RefineEntropyCodes)(data, length,
447
0
                         sampling_stride_length,
448
0
                         num_histograms, histograms, tmp);
449
0
  {
450
    /* Find a good path through literals with the good entropy codes. */
451
0
    uint8_t* block_ids = BROTLI_ALLOC(m, uint8_t, length);
452
0
    size_t num_blocks = 0;
453
0
    const size_t bitmaplen = (num_histograms + 7) >> 3;
454
0
    double* insert_cost = BROTLI_ALLOC(m, double, data_size * num_histograms);
455
0
    double* cost = BROTLI_ALLOC(m, double, num_histograms);
456
0
    uint8_t* switch_signal = BROTLI_ALLOC(m, uint8_t, length * bitmaplen);
457
0
    uint16_t* new_id = BROTLI_ALLOC(m, uint16_t, num_histograms);
458
0
    const size_t iters = params->quality < HQ_ZOPFLIFICATION_QUALITY ? 3 : 10;
459
0
    size_t i;
460
0
    if (BROTLI_IS_OOM(m) || BROTLI_IS_NULL(block_ids) ||
461
0
        BROTLI_IS_NULL(insert_cost) || BROTLI_IS_NULL(cost) ||
462
0
        BROTLI_IS_NULL(switch_signal) || BROTLI_IS_NULL(new_id)) {
463
0
      return;
464
0
    }
465
0
    for (i = 0; i < iters; ++i) {
466
0
      num_blocks = FN(FindBlocks)(data, length,
467
0
                                  block_switch_cost,
468
0
                                  num_histograms, histograms,
469
0
                                  insert_cost, cost, switch_signal,
470
0
                                  block_ids);
471
0
      num_histograms = FN(RemapBlockIds)(block_ids, length,
472
0
                                         new_id, num_histograms);
473
0
      FN(BuildBlockHistograms)(data, length, block_ids,
474
0
                               num_histograms, histograms);
475
0
    }
476
0
    BROTLI_FREE(m, insert_cost);
477
0
    BROTLI_FREE(m, cost);
478
0
    BROTLI_FREE(m, switch_signal);
479
0
    BROTLI_FREE(m, new_id);
480
0
    BROTLI_FREE(m, histograms);
481
0
    FN(ClusterBlocks)(m, data, length, num_blocks, block_ids, split);
482
0
    if (BROTLI_IS_OOM(m)) return;
483
0
    BROTLI_FREE(m, block_ids);
484
0
  }
485
0
}
Unexecuted instantiation: block_splitter.c:SplitByteVectorLiteral
Unexecuted instantiation: block_splitter.c:SplitByteVectorCommand
Unexecuted instantiation: block_splitter.c:SplitByteVectorDistance
486
487
#undef HistogramType