forked from simongog/sdsl-lite
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcoder_elias_delta.hpp
258 lines (236 loc) · 10.8 KB
/
coder_elias_delta.hpp
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
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
/* sdsl - succinct data structures library
Copyright (C) 2009 Simon Gog
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see http://www.gnu.org/licenses/ .
*/
/*! \file coder_elias_delta.hpp
\brief coder_elias_delta.hpp contains the class sdsl::coder::elias_delta
\author Simon Gog
*/
#ifndef SDSL_CODER_ELIAS_DELTA
#define SDSL_CODER_ELIAS_DELTA
#include "int_vector.hpp"
namespace sdsl
{
namespace coder
{
//! A class to encode and decode between Elias-\f$\delta\f$ and binary code.
class elias_delta
{
public:
typedef uint64_t size_type;
static struct impl {
//! Array contains precomputed values for the decoding of the prefix sum of Elias-Delta encoded numbers.
/*! The 8 most significant bits contain the length of decoded bits.
* The following 8 bits contain the number of decoded values.
* The last 16 bits contain the sum of the decoded values.
*/
uint32_t prefixsum[1<<16];
uint16_t prefixsum_8bit[(1<<8)*8];
impl() {
// initialize prefixsum
for (uint64_t x=0; x < (1<<16); ++x) {
const uint64_t* w = &x; // copy of x
uint64_t value = 0;
uint16_t numbers = 0, offset=0, offset2=0;
while ((x >> offset) !=0) {
uint64_t len_1_len = bits::read_unary(w, offset), len = 0;
if (len_1_len == 0) {
offset += 1;
value += 1;
++numbers;
} else {
offset2 = offset + len_1_len +1;
len = bits::read_int(w, offset2, len_1_len) + (1ULL << len_1_len);
offset2 += len_1_len;
if (offset2 + len-1 <= 16) {
value += bits::read_int(w, offset2, len-1) + (1ULL << (len-1));
offset = offset2 + len - 1;
++numbers;
} else break;
}
}
uint32_t result=0;// the highest 8 bit equal the shift/offset, the second highest 8 bit equal the number of decoded values,
// and the last 16 bit equals value of decoded prefix sum
result = (offset << 24) | (numbers<<16) | value;
if (value>0)
assert(offset > 0 and numbers > 0 and offset<=16 and numbers <= 16);
prefixsum[x] = result;
}
// initialize prefixsum_8bit
for (uint32_t maxi=1, idx=0; maxi<=8; ++maxi) {
for (uint64_t x=0; x < (1<<8); ++x) {
const uint64_t* w = &x; // copy of x
uint64_t value = 0;
uint32_t numbers = 0, offset=0, offset2=0;
while ((x >> offset) !=0 and numbers < maxi) {
uint64_t len_1_len = bits::read_unary(w, offset), len = 0;
if (len_1_len == 0) {
offset += 1;
value += 1;
++numbers;
} else {
offset2 = offset + len_1_len +1;
len = bits::read_int(w, offset2, len_1_len) + (1ULL << len_1_len);
offset2 += len_1_len;
if (offset2 + len-1 <= 8) {
value += bits::read_int(w, offset2, len-1) + (1ULL << (len-1));
offset = offset2 + len - 1;
++numbers;
} else break;
}
}
uint16_t result=0;// the highest 8 bit equal the shift/offset, the second highest 8 bit equal the number of decoded values,
// and the last 16 bit equals value of decoded prefix sum
result = (offset << 8) | (numbers<<4) | value;
prefixsum_8bit[idx++] = result;
}
}
}
} data;
static const uint8_t min_codeword_length = 1; // 1 represents 1 and is the code word with minimum length
static uint8_t encoding_length(uint64_t);
//! Decode n Elias-delta encoded bits beginning at start_idx in the bitstring "data"
/* \param data Bitstring
\param start_idx Starting index of the decoding.
\param n Number of values to decode from the bitstring.
\param it Iterator to decode the values.
*/
template<bool sumup, bool increment,class Iterator>
static uint64_t decode(const uint64_t* data, const size_type start_idx, size_type n, Iterator it=(Iterator)nullptr);
//! Decode n EliasDelta encoded integers beginning at start_idx in the bitstring "data" and return the sum of these values.
/*! \param data Pointer to the beginning of the EliasDelta encoded bitstring.
\param start_idx Index of the first bit to endcode the values from.
\param n Number of values to decode from the bitstring. Attention: There have to be at least n encoded values in the bitstring.
*/
static uint64_t decode_prefix_sum(const uint64_t* data, const size_type start_idx, size_type n);
static uint64_t decode_prefix_sum(const uint64_t* data, const size_type start_idx, const size_type end_idx, size_type n);
template<class int_vector>
static bool encode(const int_vector& v, int_vector& z);
template<class int_vector>
static bool decode(const int_vector& z, int_vector& v);
//! Encode one positive integer x to an int_vector at bit position start_idx.
/* \param x Positive integer to encode.
\param z Raw data of vector to write the encoded form of x.
\param start_idx Beginning bit index to write the encoded form ox x in z.
*/
static void encode(uint64_t x, uint64_t*& z, uint8_t& offset);
template<class int_vector>
static uint64_t* raw_data(int_vector& v) {
return v.m_data;
}
};
// \sa coder::elias_delta::encoding_length
inline uint8_t elias_delta::encoding_length(uint64_t w)
{
uint8_t len_1 = w ? bits::hi(w) : 64;
return len_1 + (bits::hi(len_1+1)<<1) + 1;
}
template<class int_vector>
bool elias_delta::encode(const int_vector& v, int_vector& z)
{
typedef typename int_vector::size_type size_type;
z.width(v.width());
size_type z_bit_size = 0;
uint64_t w;
const uint64_t zero_val = v.width() < 64 ? (1ULL)<<v.width() : 0;
for (typename int_vector::const_iterator it = v.begin(), end = v.end(); it != end; ++it) {
if ((w=*it) == 0) {
w = zero_val;
}
z_bit_size += encoding_length(w);
}
z.bit_resize(z_bit_size); // Initial size of z
if (z_bit_size & 0x3F) { // if z_bit_size % 64 != 0
*(z.m_data + (z_bit_size>>6)) = 0; // initialize last word
}
z_bit_size = 0;
uint64_t* z_data = z.m_data;
uint8_t offset=0;
size_type len, len_1_len; // TODO: change to uint8_t and test it
for (typename int_vector::const_iterator it = v.begin(), end=v.end(); it != end; ++it) {
w = *it;
if (w == 0) {
w = zero_val;
}
// (number of bits to represent w)
len = w ? bits::hi(w)+1 : 65;
// (number of bits to represent the length of w) -1
len_1_len = bits::hi(len);
// Write unary representation for the length of the length of w
bits::write_int_and_move(z_data, 1ULL << len_1_len, offset, len_1_len+1);
if (len_1_len) {
bits::write_int_and_move(z_data, len, offset, len_1_len);
bits::write_int_and_move(z_data, w, offset, len-1);
}
}
return true;
}
inline void elias_delta::encode(uint64_t x, uint64_t*& z, uint8_t& offset)
{
// if (x == 0) {
// throw std::logic_error("elias_delta::encode(uint64_t x, uint64_t* &z, uint8_t &offset); x equals 0 that cannot be encoded!");
// }
uint8_t len, len_1_len;
// (number of bits to represent w)
len = x ? bits::hi(x)+1 : 65;
// (number of bits to represent the length of w) - 1
len_1_len = bits::hi(len);
// Write unary representation for the length of the length of w
bits::write_int_and_move(z, 1ULL << len_1_len, offset, len_1_len+1);
if (len_1_len) {
bits::write_int_and_move(z, len, offset, len_1_len);
bits::write_int_and_move(z, x, offset, len-1);
}
}
template<class int_vector>
bool elias_delta::decode(const int_vector& z, int_vector& v)
{
typename int_vector::size_type len_1_len, len, n = 0;
const uint64_t* z_data = z.data();
const uint64_t* z_end = z.data()+(z.bit_size()>>6);
uint8_t offset = 0;
while ((z_data < z_end) or (z_data==z_end and offset < (z.bit_size()&0x3F))) {
len_1_len = bits::read_unary_and_move(z_data, offset);
if (len_1_len) {
len = bits::read_int_and_move(z_data, offset, len_1_len) + (1 << len_1_len);
bits::move_right(z_data, offset, len-1);
}
++n;
}
v.width(z.width());
v.resize(n);
return decode<false, true>(z.data(), 0, n, v.begin());
}
template<bool sumup, bool increment, class Iterator>
inline uint64_t elias_delta::decode(const uint64_t* data, const size_type start_idx, size_type n, Iterator it)
{
data += (start_idx >> 6);
uint64_t value = 0;
size_type i = 0;
size_type len_1_len, len;
uint8_t offset = start_idx & 0x3F;
while (i++ < n) {// while not all values are decoded
if (!sumup) value = 0;
len_1_len = bits::read_unary_and_move(data, offset); // read length of length of x
if (!len_1_len) {
value += 1;
} else {
len = bits::read_int_and_move(data, offset, len_1_len) + (1ULL << len_1_len);
value += bits::read_int_and_move(data, offset, len-1) + (len-1<64) * (1ULL << (len-1));
}
if (increment) *(it++) = value;
}
return value;
}
} // end namespace coder
} // end namespace sdsl
#endif