-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathStretch.hpp
63 lines (54 loc) · 1.35 KB
/
Stretch.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
#ifndef PAQ8GEN_STRETCH_HPP
#define PAQ8GEN_STRETCH_HPP
#include "Squash.hpp"
#include <cassert>
#include <cstdint>
/**
* Inverse of @ref squash. d = ln(p/(1-p)), d scaled by 8 bits, p by 12 bits.
* d has range -2047 to 2047 representing -8 to 8. p has range 0 to 4095.
*/
#ifndef CHALLENGE
struct Stretch {
constexpr Stretch() {
int pi = 0;
for( int x = -2047; x <= 2047; ++x ) {
int i = squash(x);
for( int j = pi; j <= i; ++j ) {
t[j] = static_cast<short>(x);
}
pi = i + 1;
}
t[4095] = 2047;
}
short t[4096] {};
};
constexpr auto stretch = Stretch();
#define stretch(x) stretch.t[x]
#else
#include "Array.hpp"
class Stretch {
Array<short> t;
private:
Stretch() : t( 4096 ) {
int pi = 0;
for( int x = -2047; x <= 2047; ++x ) { // invert squash()
int i = squash( x );
for( int j = pi; j <= i; ++j )
t[j] = x;
pi = i + 1;
}
t[4095] = 2047;
}
Stretch(Stretch const & /*unused*/);
auto operator=(Stretch const & /*unused*/) -> Stretch & { return *this; }
static Stretch * instance;
public:
static auto getInstance() -> Stretch&;
int operator()( int p ) const {
assert( p >= 0 && p < 4096 );
return t[p];
}
};
#define stretch(x) stretch->operator()(x)
#endif
#endif //PAQ8GEN_STRETCH_HPP