Coverage Report

Created: 2022-08-24 06:33

/src/libjxl/lib/jxl/huffman_table.cc
Line
Count
Source (jump to first uncovered line)
1
// Copyright (c) the JPEG XL Project Authors. All rights reserved.
2
//
3
// Use of this source code is governed by a BSD-style
4
// license that can be found in the LICENSE file.
5
6
#include "lib/jxl/huffman_table.h"
7
8
#include <cstring> /* for memcpy */
9
#include <vector>
10
11
#include "lib/jxl/ans_params.h"
12
#include "lib/jxl/dec_huffman.h"
13
14
namespace jxl {
15
16
/* Returns reverse(reverse(key, len) + 1, len), where reverse(key, len) is the
17
   bit-wise reversal of the len least significant bits of key. */
18
47.8M
static inline int GetNextKey(int key, int len) {
19
47.8M
  int step = 1u << (len - 1);
20
95.7M
  while (key & step) {
21
47.8M
    step >>= 1;
22
47.8M
  }
23
47.8M
  return (key & (step - 1)) + step;
24
47.8M
}
25
26
/* Stores code in table[0], table[step], table[2*step], ..., table[end] */
27
/* Assumes that end is an integer multiple of step */
28
static inline void ReplicateValue(HuffmanCode* table, int step, int end,
29
47.8M
                                  HuffmanCode code) {
30
48.3M
  do {
31
48.3M
    end -= step;
32
48.3M
    table[end] = code;
33
48.3M
  } while (end > 0);
34
47.8M
}
35
36
/* Returns the table width of the next 2nd level table. count is the histogram
37
   of bit lengths for the remaining symbols, len is the code length of the next
38
   processed symbol */
39
static inline size_t NextTableBitSize(const uint16_t* const count, size_t len,
40
1.04M
                                      int root_bits) {
41
1.04M
  size_t left = 1u << (len - root_bits);
42
1.05M
  while (len < PREFIX_MAX_BITS) {
43
797k
    if (left <= count[len]) break;
44
10.8k
    left -= count[len];
45
10.8k
    ++len;
46
10.8k
    left <<= 1;
47
10.8k
  }
48
1.04M
  return len - root_bits;
49
1.04M
}
50
51
uint32_t BuildHuffmanTable(HuffmanCode* root_table, int root_bits,
52
                           const uint8_t* const code_lengths,
53
29.6k
                           size_t code_lengths_size, uint16_t* count) {
54
29.6k
  HuffmanCode code;   /* current table entry */
55
29.6k
  HuffmanCode* table; /* next available space in table */
56
29.6k
  size_t len;         /* current code length */
57
29.6k
  size_t symbol;      /* symbol index in original or sorted table */
58
29.6k
  int key;            /* reversed prefix code */
59
29.6k
  int step;           /* step size to replicate values in current table */
60
29.6k
  int low;            /* low bits for current root entry */
61
29.6k
  int mask;           /* mask for low bits */
62
29.6k
  size_t table_bits;  /* key length of current table */
63
29.6k
  int table_size;     /* size of current table */
64
29.6k
  int total_size;     /* sum of root table size and 2nd level table sizes */
65
  /* offsets in sorted table for each length */
66
29.6k
  uint16_t offset[PREFIX_MAX_BITS + 1];
67
29.6k
  size_t max_length = 1;
68
69
29.6k
  if (code_lengths_size > 1u << PREFIX_MAX_BITS) return 0;
70
71
  /* symbols sorted by code length */
72
29.6k
  std::vector<uint16_t> sorted_storage(code_lengths_size);
73
29.6k
  uint16_t* sorted = sorted_storage.data();
74
75
  /* generate offsets into sorted symbol table by code length */
76
29.6k
  {
77
29.6k
    uint16_t sum = 0;
78
474k
    for (len = 1; len <= PREFIX_MAX_BITS; len++) {
79
444k
      offset[len] = sum;
80
444k
      if (count[len]) {
81
79.7k
        sum = static_cast<uint16_t>(sum + count[len]);
82
79.7k
        max_length = len;
83
79.7k
      }
84
444k
    }
85
29.6k
  }
86
87
  /* sort symbols by length, by symbol order within each length */
88
106M
  for (symbol = 0; symbol < code_lengths_size; symbol++) {
89
106M
    if (code_lengths[symbol] != 0) {
90
47.8M
      sorted[offset[code_lengths[symbol]]++] = symbol;
91
47.8M
    }
92
106M
  }
93
94
29.6k
  table = root_table;
95
29.6k
  table_bits = root_bits;
96
29.6k
  table_size = 1u << table_bits;
97
29.6k
  total_size = table_size;
98
99
  /* special case code with only one value */
100
29.6k
  if (offset[PREFIX_MAX_BITS] == 1) {
101
1.64k
    code.bits = 0;
102
1.64k
    code.value = static_cast<uint16_t>(sorted[0]);
103
54.2k
    for (key = 0; key < total_size; ++key) {
104
52.5k
      table[key] = code;
105
52.5k
    }
106
1.64k
    return total_size;
107
1.64k
  }
108
109
  /* fill in root table */
110
  /* let's reduce the table size to a smaller size if possible, and */
111
  /* create the repetitions by memcpy if possible in the coming loop */
112
27.9k
  if (table_bits > max_length) {
113
19.8k
    table_bits = max_length;
114
19.8k
    table_size = 1u << table_bits;
115
19.8k
  }
116
27.9k
  key = 0;
117
27.9k
  symbol = 0;
118
27.9k
  code.bits = 1;
119
27.9k
  step = 2;
120
112k
  do {
121
490k
    for (; count[code.bits] != 0; --count[code.bits]) {
122
377k
      code.value = static_cast<uint16_t>(sorted[symbol++]);
123
377k
      ReplicateValue(&table[key], step, table_size, code);
124
377k
      key = GetNextKey(key, code.bits);
125
377k
    }
126
112k
    step <<= 1;
127
112k
  } while (++code.bits <= table_bits);
128
129
  /* if root_bits != table_bits we only created one fraction of the */
130
  /* table, and we need to replicate it now. */
131
95.5k
  while (total_size != table_size) {
132
67.5k
    memcpy(&table[table_size], &table[0], table_size * sizeof(table[0]));
133
67.5k
    table_size <<= 1;
134
67.5k
  }
135
136
  /* fill in 2nd level tables and add pointers to root table */
137
27.9k
  mask = total_size - 1;
138
27.9k
  low = -1;
139
52.6k
  for (len = root_bits + 1, step = 2; len <= max_length; ++len, step <<= 1) {
140
47.5M
    for (; count[len] != 0; --count[len]) {
141
47.5M
      if ((key & mask) != low) {
142
1.04M
        table += table_size;
143
1.04M
        table_bits = NextTableBitSize(count, len, root_bits);
144
1.04M
        table_size = 1u << table_bits;
145
1.04M
        total_size += table_size;
146
1.04M
        low = key & mask;
147
1.04M
        root_table[low].bits = static_cast<uint8_t>(table_bits + root_bits);
148
1.04M
        root_table[low].value =
149
1.04M
            static_cast<uint16_t>((table - root_table) - low);
150
1.04M
      }
151
47.5M
      code.bits = static_cast<uint8_t>(len - root_bits);
152
47.5M
      code.value = static_cast<uint16_t>(sorted[symbol++]);
153
47.5M
      ReplicateValue(&table[key >> root_bits], step, table_size, code);
154
47.5M
      key = GetNextKey(key, len);
155
47.5M
    }
156
24.6k
  }
157
158
27.9k
  return total_size;
159
29.6k
}
160
161
}  // namespace jxl