-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathdecode_generic.h
178 lines (144 loc) · 5.51 KB
/
decode_generic.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
/**
Copyright (c) 2019 Vicente Romero Calero. All rights reserved.
Licensed under the MIT License.
See LICENSE file in the project root for full license information.
*/
#include <stddef.h>
#include <stdint.h>
#include "bitstream.h"
#include "common.h"
#include "internals.h"
#define decctx_(_width_) BITWIDTH_SUFFIX(decctx, _width_)
#define decctx decctx_(BITWIDTH)
#define decctx_init_(_width_) BITWIDTH_SUFFIX(decctx_init, _width_)
#define decctx_init decctx_init_(BITWIDTH)
#define decode_lower_bits_step_(_width_) BITWIDTH_SUFFIX(decode_lower_bits_step, _width_)
#define decode_lower_bits_step decode_lower_bits_step_(BITWIDTH)
#define decode_lower_bits_(_width_) BITWIDTH_SUFFIX(decode_lower_bits, _width_)
#define decode_lower_bits decode_lower_bits_(BITWIDTH)
#define decode_full_subtree_(_width_) BITWIDTH_SUFFIX(decode_full_subtree, _width_)
#define decode_full_subtree decode_full_subtree_(BITWIDTH)
#define bcltree_add_(_width_) BITWIDTH_SUFFIX(bcltree_add, _width_)
#define bcltree_add bcltree_add_(BITWIDTH)
#define bcltree_has_more_(_width_) BITWIDTH_SUFFIX(bcltree_has_more, _width_)
#define bcltree_has_more bcltree_has_more_(BITWIDTH)
#define bcltree_next_(_width_) BITWIDTH_SUFFIX(bcltree_next, _width_)
#define bcltree_next bcltree_next_(BITWIDTH)
#define decode_bit_cluster_tree_(_width_) BITWIDTH_SUFFIX(decode_bit_cluster_tree, _width_)
#define decode_bit_cluster_tree decode_bit_cluster_tree_(BITWIDTH)
#define vtenc_decode_(_width_) BITWIDTH_SUFFIX(vtenc_decode, _width_)
#define vtenc_decode vtenc_decode_(BITWIDTH)
struct decctx {
TYPE *values;
size_t values_len;
int reconstruct_full_subtrees;
size_t min_cluster_length;
struct dec_stack stack;
struct bsreader bits_reader;
};
static int decctx_init(struct decctx *ctx, const vtenc *dec,
const uint8_t *in, size_t in_len, TYPE *out, size_t out_len)
{
ctx->values = out;
ctx->values_len = out_len;
/**
* `skip_full_subtrees` parameter is only applicable to sets, i.e. sequences
* with no repeated values.
*/
ctx->reconstruct_full_subtrees = !dec->params.allow_repeated_values &&
dec->params.skip_full_subtrees;
ctx->min_cluster_length = dec->params.min_cluster_length;
dec_stack_init(&ctx->stack);
bsreader_init(&ctx->bits_reader, in, in_len);
return VTENC_OK;
}
static inline TYPE decode_lower_bits_step(struct decctx *ctx,
unsigned int n_bits)
{
#if BITWIDTH > BIT_STREAM_MAX_READ
uint64_t value = 0;
unsigned int shift = 0;
if (n_bits > BIT_STREAM_MAX_READ) {
value = bsreader_read(&ctx->bits_reader, BIT_STREAM_MAX_READ);
shift = BIT_STREAM_MAX_READ;
n_bits -= BIT_STREAM_MAX_READ;
}
return (TYPE)(value | (bsreader_read(&ctx->bits_reader, n_bits) << shift));
#else
return (TYPE)bsreader_read(&ctx->bits_reader, n_bits);
#endif
}
static inline void decode_lower_bits(struct decctx *ctx,
TYPE *values, size_t values_len, unsigned int n_bits, TYPE higher_bits)
{
for (size_t i = 0; i < values_len; ++i) {
values[i] = higher_bits | decode_lower_bits_step(ctx, n_bits);
}
}
static inline void decode_full_subtree(TYPE *values, size_t values_len, TYPE higher_bits)
{
for (size_t i = 0; i < values_len; ++i) {
values[i] = higher_bits | (TYPE)i;
}
}
static inline void bcltree_add(struct decctx *ctx,
const struct dec_bit_cluster *cluster)
{
if (cluster->length == 0)
return;
dec_stack_push(&ctx->stack, cluster);
}
static inline int bcltree_has_more(struct decctx *ctx)
{
return !dec_stack_empty(&ctx->stack);
}
static inline struct dec_bit_cluster *bcltree_next(struct decctx *ctx)
{
return dec_stack_pop(&ctx->stack);
}
static int decode_bit_cluster_tree(struct decctx *ctx)
{
bcltree_add(ctx, &(struct dec_bit_cluster){0, ctx->values_len, BITWIDTH, 0});
while (bcltree_has_more(ctx)) {
struct dec_bit_cluster *cluster = bcltree_next(ctx);
size_t cl_from = cluster->from;
size_t cl_len = cluster->length;
unsigned int cl_bit_pos = cluster->bit_pos;
uint64_t cl_higher_bits = cluster->higher_bits;
if (cl_bit_pos == 0) {
for (size_t i = 0; i < cl_len; i++) {
ctx->values[cl_from + i] = (TYPE)cl_higher_bits;
}
continue;
}
if (ctx->reconstruct_full_subtrees && is_full_subtree(cl_len, cl_bit_pos)) {
decode_full_subtree(ctx->values + cl_from, cl_len, cl_higher_bits);
continue;
}
if (cl_len <= ctx->min_cluster_length) {
decode_lower_bits(ctx, ctx->values + cl_from, cl_len, cl_bit_pos, cl_higher_bits);
continue;
}
unsigned int enc_len = bits_len_u64(cl_len);
uint64_t n_zeros = bsreader_read(&ctx->bits_reader, enc_len);
if (n_zeros > (uint64_t)cl_len) return VTENC_ERR_WRONG_FORMAT;
unsigned int next_bit_pos = cl_bit_pos - 1;
struct dec_bit_cluster zeros_cluster = {cl_from, n_zeros, next_bit_pos, cl_higher_bits};
struct dec_bit_cluster ones_cluster = {cl_from + n_zeros, cl_len - n_zeros, next_bit_pos, cl_higher_bits | (1LL << (next_bit_pos))};
bcltree_add(ctx, &ones_cluster);
bcltree_add(ctx, &zeros_cluster);
}
return VTENC_OK;
}
int vtenc_decode(vtenc *dec, const uint8_t *in, size_t in_len, TYPE *out, size_t out_len)
{
struct decctx ctx;
uint64_t max_values = dec->params.allow_repeated_values ? LIST_MAX_VALUES : SET_MAX_VALUES;
if ((uint64_t)out_len > max_values)
return VTENC_ERR_OUTPUT_TOO_BIG;
int rc = decctx_init(&ctx, dec, in, in_len, out, out_len);
if (rc != VTENC_OK)
return rc;
memset(out, 0, out_len * sizeof(*out));
return decode_bit_cluster_tree(&ctx);
}