Line data Source code
1 : /*
2 : Copyright 2015 Esri
3 :
4 : Licensed under the Apache License, Version 2.0 (the "License");
5 : you may not use this file except in compliance with the License.
6 : You may obtain a copy of the License at
7 :
8 : http://www.apache.org/licenses/LICENSE-2.0
9 :
10 : Unless required by applicable law or agreed to in writing, software
11 : distributed under the License is distributed on an "AS IS" BASIS,
12 : WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 : See the License for the specific language governing permissions and
14 : limitations under the License.
15 :
16 : A local copy of the license and additional notices are located with the
17 : source distribution at:
18 :
19 : http://github.com/Esri/lerc/
20 :
21 : Contributors: Thomas Maurer
22 : Lucian Plesea (provided checksum code)
23 : */
24 :
25 : #include "Defines.h"
26 : #include "Lerc2.h"
27 :
28 : USING_NAMESPACE_LERC
29 : using namespace std;
30 :
31 0 : static void ignore_ret_val(bool) {}
32 :
33 : // -------------------------------------------------------------------------- ;
34 :
35 7807 : Lerc2::Lerc2()
36 : {
37 7807 : Init();
38 7807 : }
39 :
40 : // -------------------------------------------------------------------------- ;
41 :
42 0 : Lerc2::Lerc2(int nDim, int nCols, int nRows, const Byte* pMaskBits)
43 : {
44 0 : Init();
45 0 : ignore_ret_val(Set(nDim, nCols, nRows, pMaskBits));
46 0 : }
47 :
48 : // -------------------------------------------------------------------------- ;
49 :
50 5180 : bool Lerc2::SetEncoderToOldVersion(int version)
51 : {
52 5180 : if (version < 2 || version > kCurrVersion)
53 0 : return false;
54 :
55 5180 : if (version < 4 && m_headerInfo.nDim > 1)
56 0 : return false;
57 :
58 5180 : m_headerInfo.version = version;
59 :
60 5180 : return true;
61 : }
62 :
63 : // -------------------------------------------------------------------------- ;
64 :
65 7807 : void Lerc2::Init()
66 : {
67 7807 : m_microBlockSize = 8;
68 7807 : m_maxValToQuantize = 0;
69 7807 : m_encodeMask = true;
70 7807 : m_writeDataOneSweep = false;
71 7807 : m_imageEncodeMode = IEM_Tiling;
72 :
73 7807 : m_headerInfo.RawInit();
74 7807 : m_headerInfo.version = kCurrVersion;
75 7807 : m_headerInfo.microBlockSize = m_microBlockSize;
76 7807 : }
77 :
78 : // -------------------------------------------------------------------------- ;
79 :
80 5181 : bool Lerc2::Set(int nDim, int nCols, int nRows, const Byte* pMaskBits)
81 : {
82 5181 : if (nDim > 1 && m_headerInfo.version < 4)
83 0 : return false;
84 :
85 5181 : if (!m_bitMask.SetSize(nCols, nRows))
86 0 : return false;
87 :
88 5181 : if (pMaskBits)
89 : {
90 18 : memcpy(m_bitMask.Bits(), pMaskBits, m_bitMask.Size());
91 18 : m_headerInfo.numValidPixel = m_bitMask.CountValidBits();
92 : }
93 : else
94 : {
95 5163 : m_headerInfo.numValidPixel = nCols * nRows;
96 5163 : m_bitMask.SetAllValid();
97 : }
98 :
99 5181 : m_headerInfo.nDim = nDim;
100 5181 : m_headerInfo.nCols = nCols;
101 5181 : m_headerInfo.nRows = nRows;
102 :
103 5181 : return true;
104 : }
105 :
106 : // -------------------------------------------------------------------------- ;
107 :
108 : //// if the Lerc2 header should ever shrink in size to less than below, then update it (very unlikely)
109 : //
110 : //unsigned int Lerc2::MinNumBytesNeededToReadHeader()
111 : //{
112 : // unsigned int numBytes = (unsigned int)FileKey().length();
113 : // numBytes += 7 * sizeof(int);
114 : // numBytes += 3 * sizeof(double);
115 : // return numBytes;
116 : //}
117 :
118 : // -------------------------------------------------------------------------- ;
119 :
120 7282 : bool Lerc2::GetHeaderInfo(const Byte* pByte, size_t nBytesRemaining, struct HeaderInfo& hd)
121 : {
122 7282 : if (!pByte || !IsLittleEndianSystem())
123 0 : return false;
124 :
125 7282 : return ReadHeader(&pByte, nBytesRemaining, hd);
126 : }
127 :
128 : // -------------------------------------------------------------------------- ;
129 : // -------------------------------------------------------------------------- ;
130 :
131 5181 : unsigned int Lerc2::ComputeNumBytesHeaderToWrite(const struct HeaderInfo& hd)
132 : {
133 5181 : unsigned int numBytes = (unsigned int)FileKey().length();
134 5181 : numBytes += 1 * sizeof(int);
135 5181 : numBytes += (hd.version >= 3 ? 1 : 0) * sizeof(unsigned int);
136 5181 : numBytes += (hd.version >= 4 ? 7 : 6) * sizeof(int);
137 5181 : numBytes += 3 * sizeof(double);
138 5181 : return numBytes;
139 : }
140 :
141 : // -------------------------------------------------------------------------- ;
142 :
143 5181 : bool Lerc2::WriteHeader(Byte** ppByte, const struct HeaderInfo& hd)
144 : {
145 5181 : if (!ppByte)
146 0 : return false;
147 :
148 5181 : Byte* ptr = *ppByte;
149 :
150 10362 : string fileKey = FileKey();
151 5181 : size_t len = fileKey.length();
152 5181 : memcpy(ptr, fileKey.c_str(), len);
153 5181 : ptr += len;
154 :
155 5181 : memcpy(ptr, &hd.version, sizeof(int));
156 5181 : ptr += sizeof(int);
157 :
158 5181 : if (hd.version >= 3)
159 : {
160 370 : unsigned int checksum = 0;
161 370 : memcpy(ptr, &checksum, sizeof(unsigned int)); // place holder to be filled by the real check sum later
162 370 : ptr += sizeof(unsigned int);
163 : }
164 :
165 10362 : vector<int> intVec;
166 5181 : intVec.push_back(hd.nRows);
167 5181 : intVec.push_back(hd.nCols);
168 :
169 5181 : if (hd.version >= 4)
170 : {
171 370 : intVec.push_back(hd.nDim);
172 : }
173 :
174 5181 : intVec.push_back(hd.numValidPixel);
175 5181 : intVec.push_back(hd.microBlockSize);
176 5181 : intVec.push_back(hd.blobSize);
177 5181 : intVec.push_back((int)hd.dt);
178 :
179 5181 : len = intVec.size() * sizeof(int);
180 5181 : memcpy(ptr, &intVec[0], len);
181 5181 : ptr += len;
182 :
183 5181 : vector<double> dblVec;
184 5181 : dblVec.push_back(hd.maxZError);
185 5181 : dblVec.push_back(hd.zMin);
186 5181 : dblVec.push_back(hd.zMax);
187 :
188 5181 : len = dblVec.size() * sizeof(double);
189 5181 : memcpy(ptr, &dblVec[0], len);
190 5181 : ptr += len;
191 :
192 5181 : *ppByte = ptr;
193 5181 : return true;
194 : }
195 :
196 : // -------------------------------------------------------------------------- ;
197 :
198 9908 : bool Lerc2::ReadHeader(const Byte** ppByte, size_t& nBytesRemainingInOut, struct HeaderInfo& hd)
199 : {
200 9908 : if (!ppByte || !*ppByte)
201 0 : return false;
202 :
203 9908 : const Byte* ptr = *ppByte;
204 9908 : size_t nBytesRemaining = nBytesRemainingInOut;
205 :
206 19816 : string fileKey = FileKey();
207 9907 : size_t keyLen = fileKey.length();
208 :
209 9908 : hd.RawInit();
210 :
211 9908 : if (nBytesRemaining < keyLen || memcmp(ptr, fileKey.c_str(), keyLen))
212 1015 : return false;
213 :
214 8893 : ptr += keyLen;
215 8893 : nBytesRemaining -= keyLen;
216 :
217 8893 : if (nBytesRemaining < sizeof(int) || !memcpy(&(hd.version), ptr, sizeof(int)))
218 0 : return false;
219 :
220 8893 : ptr += sizeof(int);
221 8893 : nBytesRemaining -= sizeof(int);
222 :
223 8893 : if (hd.version > kCurrVersion) // this reader is outdated
224 0 : return false;
225 :
226 8893 : if (hd.version >= 3)
227 : {
228 4059 : if (nBytesRemaining < sizeof(unsigned int) || !memcpy(&(hd.checksum), ptr, sizeof(unsigned int)))
229 0 : return false;
230 :
231 4059 : ptr += sizeof(unsigned int);
232 4059 : nBytesRemaining -= sizeof(unsigned int);
233 : }
234 :
235 8893 : int nInts = (hd.version >= 4) ? 7 : 6;
236 17786 : vector<int> intVec(nInts, 0);
237 17785 : vector<double> dblVec(3, 0);
238 :
239 8893 : size_t len = sizeof(int) * intVec.size();
240 :
241 8893 : if (nBytesRemaining < len || !memcpy(&intVec[0], ptr, len))
242 0 : return false;
243 :
244 8893 : ptr += len;
245 8893 : nBytesRemaining -= len;
246 :
247 8893 : len = sizeof(double) * dblVec.size();
248 :
249 8893 : if (nBytesRemaining < len || !memcpy(&dblVec[0], ptr, len))
250 0 : return false;
251 :
252 8893 : ptr += len;
253 8893 : nBytesRemaining -= len;
254 :
255 8893 : int i = 0;
256 8893 : hd.nRows = intVec[i++];
257 8892 : hd.nCols = intVec[i++];
258 8892 : hd.nDim = (hd.version >= 4) ? intVec[i++] : 1;
259 8892 : hd.numValidPixel = intVec[i++];
260 8891 : hd.microBlockSize = intVec[i++];
261 8891 : hd.blobSize = intVec[i++];
262 8892 : const int dt = intVec[i++];
263 8892 : if( dt < DT_Char || dt > DT_Undefined )
264 0 : return false;
265 8892 : hd.dt = static_cast<DataType>(dt);
266 :
267 8892 : hd.maxZError = dblVec[0];
268 8892 : hd.zMin = dblVec[1];
269 8892 : hd.zMax = dblVec[2];
270 :
271 8892 : if (hd.nRows <= 0 || hd.nCols <= 0 || hd.nDim <= 0 || hd.numValidPixel < 0 || hd.microBlockSize <= 0 || hd.blobSize <= 0)
272 0 : return false;
273 :
274 8892 : *ppByte = ptr;
275 8892 : nBytesRemainingInOut = nBytesRemaining;
276 :
277 8892 : return true;
278 : }
279 :
280 : // -------------------------------------------------------------------------- ;
281 :
282 5181 : bool Lerc2::WriteMask(Byte** ppByte) const
283 : {
284 5181 : if (!ppByte)
285 0 : return false;
286 :
287 5181 : int numValid = m_headerInfo.numValidPixel;
288 5181 : int numTotal = m_headerInfo.nCols * m_headerInfo.nRows;
289 :
290 5181 : bool needMask = numValid > 0 && numValid < numTotal;
291 :
292 5181 : Byte* ptr = *ppByte;
293 :
294 5181 : if (needMask && m_encodeMask)
295 : {
296 : Byte* pArrRLE;
297 : size_t numBytesRLE;
298 9 : RLE rle;
299 9 : if (!rle.compress((const Byte*)m_bitMask.Bits(), m_bitMask.Size(), &pArrRLE, numBytesRLE, false))
300 0 : return false;
301 :
302 9 : int numBytesMask = (int)numBytesRLE;
303 9 : memcpy(ptr, &numBytesMask, sizeof(int)); // num bytes for compressed mask
304 9 : ptr += sizeof(int);
305 9 : memcpy(ptr, pArrRLE, numBytesRLE);
306 9 : ptr += numBytesRLE;
307 :
308 18 : delete[] pArrRLE;
309 : }
310 : else
311 : {
312 5172 : memset(ptr, 0, sizeof(int)); // indicates no mask stored
313 5172 : ptr += sizeof(int);
314 : }
315 :
316 5181 : *ppByte = ptr;
317 5181 : return true;
318 : }
319 :
320 : // -------------------------------------------------------------------------- ;
321 :
322 2626 : bool Lerc2::ReadMask(const Byte** ppByte, size_t& nBytesRemainingInOut)
323 : {
324 2626 : if (!ppByte)
325 0 : return false;
326 :
327 2626 : int numValid = m_headerInfo.numValidPixel;
328 2626 : int w = m_headerInfo.nCols;
329 2626 : int h = m_headerInfo.nRows;
330 :
331 2626 : const Byte* ptr = *ppByte;
332 2626 : size_t nBytesRemaining = nBytesRemainingInOut;
333 :
334 : int numBytesMask;
335 2626 : if (nBytesRemaining < sizeof(int) || !memcpy(&numBytesMask, ptr, sizeof(int)))
336 0 : return false;
337 :
338 2626 : ptr += sizeof(int);
339 2626 : nBytesRemaining -= sizeof(int);
340 :
341 2626 : if (numValid == 0 || numValid == w * h)
342 : {
343 2617 : if (numBytesMask != 0)
344 0 : return false;
345 : }
346 :
347 2626 : if (!m_bitMask.SetSize(w, h))
348 0 : return false;
349 :
350 2626 : if (numValid == 0)
351 9 : m_bitMask.SetAllInvalid();
352 2617 : else if (numValid == w * h)
353 2608 : m_bitMask.SetAllValid();
354 9 : else if (numBytesMask > 0) // read it in
355 : {
356 9 : if (nBytesRemaining < static_cast<size_t>(numBytesMask))
357 0 : return false;
358 :
359 9 : RLE rle;
360 9 : if (!rle.decompress(ptr, nBytesRemaining, m_bitMask.Bits(), m_bitMask.Size()))
361 0 : return false;
362 :
363 9 : ptr += numBytesMask;
364 9 : nBytesRemaining -= numBytesMask;
365 : }
366 : // else use previous mask
367 :
368 2626 : *ppByte = ptr;
369 2626 : nBytesRemainingInOut = nBytesRemaining;
370 :
371 2626 : return true;
372 : }
373 :
374 : // -------------------------------------------------------------------------- ;
375 :
376 5181 : bool Lerc2::DoChecksOnEncode(Byte* pBlobBegin, Byte* pBlobEnd) const
377 : {
378 5181 : if ((size_t)(pBlobEnd - pBlobBegin) != (size_t)m_headerInfo.blobSize)
379 0 : return false;
380 :
381 5181 : if (m_headerInfo.version >= 3)
382 : {
383 370 : int blobSize = (int)(pBlobEnd - pBlobBegin);
384 370 : int nBytes = (int)(FileKey().length() + sizeof(int) + sizeof(unsigned int)); // start right after the checksum entry
385 370 : if (blobSize < nBytes)
386 0 : return false;
387 370 : unsigned int checksum = ComputeChecksumFletcher32(pBlobBegin + nBytes, blobSize - nBytes);
388 :
389 370 : nBytes -= sizeof(unsigned int);
390 370 : memcpy(pBlobBegin + nBytes, &checksum, sizeof(unsigned int));
391 : }
392 :
393 5181 : return true;
394 : }
395 :
396 : // -------------------------------------------------------------------------- ;
397 :
398 : // from https://en.wikipedia.org/wiki/Fletcher's_checksum
399 : // modified from ushorts to bytes (by Lucian Plesea)
400 :
401 1385 : unsigned int Lerc2::ComputeChecksumFletcher32(const Byte* pByte, int len)
402 : {
403 1385 : unsigned int sum1 = 0xffff, sum2 = 0xffff;
404 1385 : unsigned int words = len / 2;
405 :
406 8801 : while (words)
407 : {
408 7416 : unsigned int tlen = (words >= 359) ? 359 : words;
409 7416 : words -= tlen;
410 2406280 : do {
411 2413690 : sum1 += (*pByte++ << 8);
412 2413690 : sum2 += sum1 += *pByte++;
413 2413690 : } while (--tlen);
414 :
415 7416 : sum1 = (sum1 & 0xffff) + (sum1 >> 16);
416 7416 : sum2 = (sum2 & 0xffff) + (sum2 >> 16);
417 : }
418 :
419 : // add the straggler byte if it exists
420 1385 : if (len & 1)
421 629 : sum2 += sum1 += (*pByte << 8);
422 :
423 : // second reduction step to reduce sums to 16 bits
424 1385 : sum1 = (sum1 & 0xffff) + (sum1 >> 16);
425 1385 : sum2 = (sum2 & 0xffff) + (sum2 >> 16);
426 :
427 1385 : return sum2 << 16 | sum1;
428 : }
429 :
430 : // -------------------------------------------------------------------------- ;
431 :
432 : //struct MyLessThanOp
433 : //{
434 : // inline bool operator() (const pair<unsigned int, unsigned int>& p0,
435 : // const pair<unsigned int, unsigned int>& p1) { return p0.first < p1.first; }
436 : //};
437 :
438 : // -------------------------------------------------------------------------- ;
439 :
440 709569 : void Lerc2::SortQuantArray(const vector<unsigned int>& quantVec, vector<pair<unsigned int, unsigned int> >& sortedQuantVec)
441 : {
442 709569 : int numElem = (int)quantVec.size();
443 709569 : sortedQuantVec.resize(numElem);
444 :
445 71881600 : for (int i = 0; i < numElem; i++)
446 71172000 : sortedQuantVec[i] = pair<unsigned int, unsigned int>(quantVec[i], i);
447 :
448 : //std::sort(sortedQuantVec.begin(), sortedQuantVec.end(), MyLessThanOp());
449 :
450 709569 : std::sort(sortedQuantVec.begin(), sortedQuantVec.end(),
451 371995000 : [](const pair<unsigned int, unsigned int>& p0,
452 371995000 : const pair<unsigned int, unsigned int>& p1) { return p0.first < p1.first; });
453 709569 : }
454 :
455 : // -------------------------------------------------------------------------- ;
456 :
|