forked from ScanNet/ScanNet
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathuplinksimple_image-codecs.h
executable file
·398 lines (307 loc) · 9.52 KB
/
uplinksimple_image-codecs.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
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
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
//
// image/image-codecs.h
// Uplink
//
// Copyright (c) 2013 Occipital, Inc. All rights reserved.
//
# pragma once
#include "memory.h"
#include <functional>
# include <stdint.h>
# include <cstdio>
# include <cassert>
# include <cstdlib>
# include <cstdio>
# include <cstring>
# define BITSTREAM_SUCCESS 1
# define BITSTREAM_FAILURE 0
namespace uplinksimple {
//------------------------------------------------------------------------------
typedef struct s_bitstream {
uint8_t *buf; /* head of bitstream */
uint8_t *pos; /* current byte in bitstream */
unsigned int remain; /* bits remaining */
unsigned int len; /* length of bitstream in bytes */
uint8_t cur_bits;
} bitstream_t;
int bs_init(bitstream_t *bp);
int bs_destroy(bitstream_t **ppb);
int bs_attach(bitstream_t *b, uint8_t *buf, int blen);
uint8_t bs_get(bitstream_t * b, uint8_t nbits);
int bs_bytes_used(bitstream_t *b);
static inline int
bs_put(bitstream_t * b,
uint8_t bits,
uint8_t nbits)
{
assert(nbits != 0 && nbits <= 8);
if (nbits > b->remain) {
unsigned int over = nbits - b->remain;
*(b->pos++) = b->cur_bits | (bits >> over);
b->remain = 8 - over;
b->cur_bits = bits << b->remain;
}
else {
b->cur_bits |= bits << (b->remain - nbits);
b->remain -= nbits;
// we've exhausted the byte. move to the next byte.
if (b->remain == 0) {
*(b->pos++) = b->cur_bits;
b->remain = 8;
b->cur_bits = 0;
}
}
assert((unsigned int)(b->pos - b->buf) <= b->len);
return BITSTREAM_SUCCESS;
}
static inline int
bs_flush(bitstream_t * b)
{
*(b->pos) = b->cur_bits;
return BITSTREAM_SUCCESS;
}
//------------------------------------------------------------------------------
}
namespace uplinksimple {
//------------------------------------------------------------------------------
inline int
bs_init(bitstream_t *bp)
{
if (bp) {
memset(bp, 0, sizeof(bitstream_t));
return BITSTREAM_SUCCESS;
}
return BITSTREAM_FAILURE;
}
inline int
bs_destroy(bitstream_t * b)
{
free(b);
return BITSTREAM_SUCCESS;
}
inline int
bs_attach(bitstream_t *b,
uint8_t *buf,
int blen)
{
b->buf = b->pos = buf;
b->remain = 8;
b->len = blen;
return BITSTREAM_SUCCESS;
}
inline int
bs_bytes_used(bitstream_t *b)
{
unsigned int used = (unsigned int)(b->pos - b->buf);
return b->remain != 8 ? used + 1 : used;
}
inline uint8_t
bs_get(bitstream_t * b,
uint8_t nbits)
{
uint8_t out;
if (b->remain == 0) {
b->pos++;
b->remain = 8;
}
if (nbits > b->remain) {
/* Get high bits */
out = *b->pos;
out <<= (8 - b->remain);
out >>= (8 - nbits);
b->pos++;
b->remain += 8 - nbits;
out |= (*b->pos) >> b->remain;
}
else {
out = *b->pos;
out <<= (8 - b->remain);
out >>= (8 - nbits);
b->remain -= nbits;
}
assert((unsigned int)(b->pos - b->buf) <= b->len);
return out;
}
//------------------------------------------------------------------------------
}
namespace uplinksimple {
//------------------------------------------------------------------------------
// -----------------------------------------------
// OCCIPITAL DEPTH FRAME COMPRESSION ALGORITHM 0.1
// -----------------------------------------------
// TYPICAL COMPRESSION RATIO ON 640x480 test image: 0.17
// DECODE:
// Step 0. Last value is initialized to 0. Frame size is known in advance.
// Step 1. Proceed by decoding following bitstream until all pixels are decoded.
// 00 - Next value is same as last value.
// 11 - Next value is last value + 1.
// 10 - Next value is last value - 1.
// 010 - bbbbb - Next N values are same as last value. (N encoded w/ 5 bits)
// 0111 - bbbbbbbbbbb - Next value is X. (X encoded w/ 11 bits)
// 01101 - Next value is last value + 2.
// 01100 - Next value is last value - 2.
inline uint16_t * decode(const uint8_t * bitstream_data, unsigned int bitstream_length_bytes, int numelements, uint16_t* output)
{
uint16_t lastVal = 0;
uint16_t curVal = 0;
bitstream_t bs;
bs_init(&bs);
bs_attach(&bs, const_cast<uint8_t*>(bitstream_data), bitstream_length_bytes);
uint16_t * depthimage = 0 == output
? (uint16_t*)malloc(numelements * sizeof(uint16_t))
: output
;
uint16_t * depth_ptr = depthimage;
while(numelements > 0)
{
uint8_t bit0 = bs_get(&bs, 1);
uint8_t bit1 = bs_get(&bs, 1);
if(bit0 == 0 && bit1 == 0) // 00
{
curVal = lastVal;
*(depth_ptr++) = curVal; lastVal = curVal;
numelements-=1;
}
else if(bit0 == 1) // 1 prefix
{
if(bit1 == 0) {
curVal = lastVal - 1;
}
else {
curVal = lastVal + 1;
}
*(depth_ptr++) = curVal; lastVal = curVal;
numelements-=1;
}
else // must be 01 prefix
{
uint8_t bit2 = bs_get(&bs, 1);
if(bit2 == 0) // 010 --> multiple zeros!
{
uint16_t numZeros = bs_get(&bs, 5);
numZeros += 5; // We never encode less than 5.
for(int i = 0; i < numZeros; i++) {
*(depth_ptr++) = curVal;
}
numelements-=numZeros;
}
else
{
uint8_t bit3 = bs_get(&bs, 1);
if(bit3 == 0) // 0110 -- DELTA!
{
uint8_t delta_bit = bs_get(&bs, 1);
if(delta_bit == 0) {
curVal = lastVal - 2;
}
else {
curVal = lastVal + 2;
}
*(depth_ptr++) = curVal; lastVal = curVal;
numelements-=1;
}
else // 0111 -- RESET!
{
uint16_t value = (bs_get(&bs, 3) << 8) | bs_get(&bs, 8); // 11 bits total.
curVal = value;
*(depth_ptr++) = curVal; lastVal = curVal;
numelements-=1;
}
}
}
}
return depthimage;
}
//------------------------------------------------------------------------------
inline uint32_t encode(const uint16_t * data_in, int numelements,
uint8_t* out_buffer, uint32_t out_buffer_size)
{
int numZeros = 0;
int lastVal = 0;
bitstream_t bs;
bs_init(&bs);
bs_attach(&bs, out_buffer, out_buffer_size);
// Loop over pixels.
while (numelements > 0) {
int curVal = *(data_in++);
int delta = curVal - lastVal;
if(delta == 0)
{
numZeros++;
}
else
{
if(numZeros > 0)
{
// MUST BURN ZEROS!
while( numZeros > 0 )
{
if(numZeros <= 4)
{
// Ternary is fastest way of deciding how many zeros to encode (2 * numZeros)
bs_put(&bs, 0x0000, numZeros == 1 ? 2 : numZeros == 2 ? 4 : numZeros == 3 ? 6 : 8);
numZeros = 0;
}
else
{
bs_put(&bs, 0x2, 3); // 010bbbbb
// We never encode less than 5 because in that case
// we'll just use multiple 2-bit single zeros.
unsigned int numberToEncode = numZeros - 5;
// We're only using 5 bits, so we can't encode greater than 31.
if(numberToEncode > 31) numberToEncode = 31;
bs_put(&bs, numberToEncode, 5); // 0b 010
numZeros -= (numberToEncode+5);
}
}
// numZeros is now zero.
}
if(delta == 1 || delta == -1)
{
bs_put(&bs, delta == 1 ? 0x3 : 0x2, 2); // 0b 11
}
else if (delta >= -2 && delta <= 2)
{
bs_put(&bs, delta == 2 ? 0xD : 0xC, 5);
}
else // Reset == 1111 bbbbbbbbbbb
{
bs_put(&bs, 0x7, 4); // 0111
bs_put(&bs, curVal >> 8, 3);
bs_put(&bs, curVal , 8);
}
} // end else block of if (delta == 0)
lastVal = curVal;
numelements--;
}
// FINISH Up -- repeat zeros check.
if(numZeros > 0)
{
// MUST BURN ZEROS!
while(numZeros > 0)
{
if(numZeros <= 4)
{
// Ternary is fastest way of deciding how many zeros to encode (2 * numZeros)
bs_put(&bs, 0x0000, numZeros == 1 ? 2 : numZeros == 2 ? 4 : numZeros == 3 ? 6 : 8);
numZeros = 0;
}
else
{
bs_put(&bs, 0x2, 3); // 010bbbbb
// We never encode less than 5 because in that case
// we'll just use multiple 2-bit single zeros.
unsigned int numberToEncode = numZeros - 5;
// We're only using 5 bits, so we can't encode greater than 31.
if(numberToEncode > 31) numberToEncode = 31;
bs_put(&bs, numberToEncode, 5); // 0b 010
numZeros -= (numberToEncode+5);
}
}
}
// numZeros is now zero.
// END FINISH UP
bs_flush(&bs);
return bs_bytes_used(&bs);
}
} // uplink namespace