-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathBitStuffer2.h
137 lines (106 loc) · 4.6 KB
/
BitStuffer2.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
/*
Copyright 2015 Esri
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
A local copy of the license and additional notices are located with the
source distribution at:
http://github.com/Esri/lerc/
Contributors: Thomas Maurer
*/
#ifndef BITSTUFFER2_H
#define BITSTUFFER2_H
#include <vector>
#include <cstring>
#include <utility>
#include "Defines.h"
NAMESPACE_LERC_START
/** Bit stuffer, for writing unsigned int arrays compressed lossless
*
*/
class BitStuffer2
{
public:
BitStuffer2() {}
virtual ~BitStuffer2() {}
// dst buffer is already allocated. byte ptr is moved like a file pointer.
bool EncodeSimple(Byte** ppByte, const std::vector<unsigned int>& dataVec, int lerc2Version) const;
bool EncodeLut(Byte** ppByte, const std::vector<std::pair<unsigned int, unsigned int> >& sortedDataVec, int lerc2Version) const;
bool Decode(const Byte** ppByte, size_t& nBytesRemaining, std::vector<unsigned int>& dataVec, size_t maxElementCount, int lerc2Version) const;
static unsigned int ComputeNumBytesNeededSimple(unsigned int numElem, unsigned int maxElem);
static unsigned int ComputeNumBytesNeededLut(const std::vector<std::pair<unsigned int, unsigned int> >& sortedDataVec, bool& doLut);
private:
mutable std::vector<unsigned int> m_tmpLutVec, m_tmpIndexVec, m_tmpBitStuffVec;
static void BitStuff_Before_Lerc2v3(Byte** ppByte, const std::vector<unsigned int>& dataVec, int numBits);
static bool BitUnStuff_Before_Lerc2v3(const Byte** ppByte, size_t& nBytesRemaining, std::vector<unsigned int>& dataVec, unsigned int numElements, int numBits);
void BitStuff(Byte** ppByte, const std::vector<unsigned int>& dataVec, int numBits) const;
bool BitUnStuff(const Byte** ppByte, size_t& nBytesRemaining, std::vector<unsigned int>& dataVec, unsigned int numElements, int numBits) const;
static bool EncodeUInt(Byte** ppByte, unsigned int k, int numBytes); // numBytes = 1, 2, or 4
static bool DecodeUInt(const Byte** ppByte, size_t& nBytesRemaining, unsigned int& k, int numBytes);
static int NumBytesUInt(unsigned int k) { return (k < 256) ? 1 : (k < (1 << 16)) ? 2 : 4; }
static unsigned int NumTailBytesNotNeeded(unsigned int numElem, int numBits);
};
// -------------------------------------------------------------------------- ;
inline unsigned int BitStuffer2::ComputeNumBytesNeededSimple(unsigned int numElem, unsigned int maxElem)
{
int numBits = 0;
while ((numBits < 32) && (maxElem >> numBits))
numBits++;
return 1 + NumBytesUInt(numElem) + ((numElem * numBits + 7) >> 3);
}
// -------------------------------------------------------------------------- ;
inline bool BitStuffer2::EncodeUInt(Byte** ppByte, unsigned int k, int numBytes)
{
Byte* ptr = *ppByte;
if (numBytes == 1)
*ptr = (Byte)k;
else if (numBytes == 2)
{
unsigned short kShort = (unsigned short)k;
memcpy(ptr, &kShort, sizeof(unsigned short));
}
else if (numBytes == 4)
memcpy(ptr, &k, sizeof(unsigned int));
else
return false;
*ppByte += numBytes;
return true;
}
// -------------------------------------------------------------------------- ;
inline bool BitStuffer2::DecodeUInt(const Byte** ppByte, size_t& nBytesRemaining, unsigned int& k, int numBytes)
{
if (nBytesRemaining < (size_t)numBytes)
return false;
const Byte* ptr = *ppByte;
if (numBytes == 1)
k = *ptr;
else if (numBytes == 2)
{
unsigned short s;
memcpy(&s, ptr, sizeof(unsigned short));
k = s;
}
else if (numBytes == 4)
memcpy(&k, ptr, sizeof(unsigned int));
else
return false;
*ppByte += numBytes;
nBytesRemaining -= numBytes;
return true;
}
// -------------------------------------------------------------------------- ;
inline unsigned int BitStuffer2::NumTailBytesNotNeeded(unsigned int numElem, int numBits)
{
int numBitsTail = ((unsigned long long)numElem * numBits) & 31;
int numBytesTail = (numBitsTail + 7) >> 3;
return (numBytesTail > 0) ? 4 - numBytesTail : 0;
}
// -------------------------------------------------------------------------- ;
NAMESPACE_LERC_END
#endif