Line data Source code
1 : /******************************************************************************
2 : *
3 : * Project: Raster Matrix Format
4 : * Purpose: Implementation of the LZW compression algorithm as used in
5 : * GIS "Panorama"/"Integratsia" raster files. Based on implementation
6 : * of Kent Williams, but heavily modified over it. The key point
7 : * in the initial implementation is a hashing algorithm.
8 : * Author: Andrey Kiselev, dron@ak4719.spb.edu
9 : *
10 : ******************************************************************************
11 : * Copyright (c) 2007, Andrey Kiselev <dron@ak4719.spb.edu>
12 : *
13 : * SPDX-License-Identifier: MIT
14 : ******************************************************************************
15 : * COPYRIGHT NOTICE FROM THE INITIAL IMPLEMENTATION:
16 : *
17 : * The programs LZWCOM and LZWUNC, both in binary executable and source forms,
18 : * are in the public domain. No warranty is given or implied, and no
19 : * liability will be assumed by the author.
20 : *
21 : * Everyone on earth is hereby given permission to use, copy, distribute,
22 : * change, mangle, destroy or otherwise employ these programs, provided they
23 : * hurt no one but themselves in the process.
24 : *
25 : * Kent Williams
26 : * Norand Inc.
27 : * 550 2nd St S.E.
28 : * Cedar Rapids, Iowa 52401
29 : * (319) 369-3131
30 : ****************************************************************************/
31 :
32 : #include "cpl_conv.h"
33 :
34 : #include "rmfdataset.h"
35 :
36 : // Code marks that there is no predecessor in the string
37 : constexpr GUInt32 NO_PRED = 0xFFFF;
38 :
39 : // We are using 12-bit codes in this particular implementation
40 : constexpr GUInt32 TABSIZE = 4096U;
41 : constexpr GUInt32 STACKSIZE = TABSIZE;
42 :
43 : constexpr GUInt32 NOT_FND = 0xFFFF;
44 :
45 : /************************************************************************/
46 : /* LZWStringTab */
47 : /************************************************************************/
48 :
49 : typedef struct
50 : {
51 : bool bUsed;
52 : GUInt32 iNext; // hi bit is 'used' flag
53 : GUInt32 iPredecessor; // 12 bit code
54 : GByte iFollower;
55 : } LZWStringTab;
56 :
57 : /************************************************************************/
58 : /* LZWUpdateTab() */
59 : /************************************************************************/
60 :
61 : CPL_NOSANITIZE_UNSIGNED_INT_OVERFLOW
62 18983400 : static GUInt32 UnsanitizedMul(GUInt32 a, GUInt32 b)
63 : {
64 18983400 : return a * b;
65 : }
66 :
67 19349800 : static int UnsignedByteToSignedByte(GByte byVal)
68 : {
69 19349800 : return byVal >= 128 ? byVal - 256 : byVal;
70 : }
71 :
72 981204 : static void LZWUpdateTab(LZWStringTab *poCodeTab, GUInt32 iPred, GByte bFollow)
73 : {
74 : /* -------------------------------------------------------------------- */
75 : /* Hash uses the 'mid-square' algorithm. I.E. for a hash val of n bits */
76 : /* hash = middle binary digits of (key * key). Upon collision, hash */
77 : /* searches down linked list of keys that hashed to that key already. */
78 : /* It will NOT notice if the table is full. This must be handled */
79 : /* elsewhere */
80 : /* -------------------------------------------------------------------- */
81 981204 : const int iFollow = UnsignedByteToSignedByte(bFollow);
82 981430 : GUInt32 nLocal = CPLUnsanitizedAdd<GUInt32>(iPred, iFollow) | 0x0800;
83 981366 : nLocal = (UnsanitizedMul(nLocal, nLocal) >> 6) &
84 : 0x0FFF; // middle 12 bits of result
85 :
86 : // If string is not used
87 981381 : GUInt32 nNext = nLocal;
88 981381 : if (poCodeTab[nLocal].bUsed)
89 : {
90 : // If collision has occurred
91 1919700 : while ((nNext = poCodeTab[nLocal].iNext) != 0)
92 1257110 : nLocal = nNext;
93 :
94 : // Search for free entry from nLocal + 101
95 662590 : nNext = (nLocal + 101) & 0x0FFF;
96 34519700 : while (poCodeTab[nNext].bUsed)
97 : {
98 33857100 : if (++nNext >= TABSIZE)
99 2787 : nNext = 0;
100 : }
101 :
102 : // Put new tempnext into last element in collision list
103 662590 : poCodeTab[nLocal].iNext = nNext;
104 : }
105 :
106 981381 : poCodeTab[nNext].bUsed = true;
107 981381 : poCodeTab[nNext].iNext = 0;
108 981381 : poCodeTab[nNext].iPredecessor = iPred;
109 981381 : poCodeTab[nNext].iFollower = bFollow;
110 981381 : }
111 :
112 : /************************************************************************/
113 : /* LZWCreateTab() */
114 : /************************************************************************/
115 :
116 261 : static LZWStringTab *LZWCreateTab()
117 : {
118 : // Allocate space for the new table and pre-fill it
119 : LZWStringTab *poCodeTab =
120 261 : (LZWStringTab *)CPLMalloc(TABSIZE * sizeof(LZWStringTab));
121 :
122 261 : memset(poCodeTab, 0, TABSIZE * sizeof(LZWStringTab));
123 :
124 67077 : for (GUInt32 iCode = 0; iCode < 256; ++iCode)
125 66816 : LZWUpdateTab(poCodeTab, NO_PRED, static_cast<GByte>(iCode));
126 :
127 261 : return poCodeTab;
128 : }
129 :
130 : /************************************************************************/
131 : /* LZWFindIndex() */
132 : /************************************************************************/
133 :
134 17404400 : static GUInt32 LZWFindIndex(const LZWStringTab *poCodeTab, GUInt32 iPred,
135 : GByte bFollow)
136 : {
137 17404400 : const int iFollow = UnsignedByteToSignedByte(bFollow);
138 17684200 : GUInt32 nLocal = CPLUnsanitizedAdd<GUInt32>(iPred, iFollow) | 0x0800;
139 17690500 : nLocal = (UnsanitizedMul(nLocal, nLocal) >> 6) &
140 : 0x0FFF; // middle 12 bits of result
141 :
142 24712300 : do
143 : {
144 38888900 : CPLAssert(nLocal < TABSIZE);
145 38888900 : if (poCodeTab[nLocal].iPredecessor == iPred &&
146 14229900 : poCodeTab[nLocal].iFollower == bFollow)
147 : {
148 14140500 : return nLocal;
149 : }
150 24748400 : nLocal = poCodeTab[nLocal].iNext;
151 24748400 : } while (nLocal > 0);
152 :
153 36067 : return NOT_FND;
154 : }
155 :
156 : /************************************************************************/
157 : /* LZWPutCode() */
158 : /************************************************************************/
159 :
160 5058520 : static bool LZWPutCode(GUInt32 iCode, GUInt32 &iTmp, bool &bBitsleft,
161 : GByte *&pabyCurrent, const GByte *const pabyOutEnd)
162 : {
163 5058520 : if (bBitsleft)
164 : {
165 2635510 : if (pabyCurrent >= pabyOutEnd)
166 : {
167 0 : return false;
168 : }
169 2635510 : *(pabyCurrent++) = static_cast<GByte>((iCode >> 4) & 0xFF);
170 2635510 : iTmp = iCode & 0x000F;
171 2635510 : bBitsleft = false;
172 : }
173 : else
174 : {
175 2423020 : if (pabyCurrent + 1 >= pabyOutEnd)
176 : {
177 28 : return false;
178 : }
179 2422990 : *(pabyCurrent++) =
180 2422990 : static_cast<GByte>(((iTmp << 4) & 0xFF0) + ((iCode >> 8) & 0x00F));
181 2422990 : *(pabyCurrent++) = static_cast<GByte>(iCode & 0xFF);
182 2422990 : bBitsleft = true;
183 : }
184 5058500 : return true;
185 : }
186 :
187 : /************************************************************************/
188 : /* LZWReadStream() */
189 : /************************************************************************/
190 :
191 137 : static size_t LZWReadStream(const GByte *pabyIn, GUInt32 nSizeIn,
192 : GByte *pabyOut, GUInt32 nSizeOut,
193 : LZWStringTab *poCodeTab)
194 : {
195 137 : GByte *const pabyOutBegin = pabyOut;
196 :
197 : // The first code is always known
198 137 : GUInt32 iCode = (*pabyIn++ << 4) & 0xFF0;
199 137 : nSizeIn--;
200 137 : iCode += (*pabyIn >> 4) & 0x00F;
201 137 : GUInt32 iOldCode = iCode;
202 137 : bool bBitsleft = true;
203 :
204 137 : GByte iFinChar = poCodeTab[iCode].iFollower;
205 137 : nSizeOut--;
206 137 : *pabyOut++ = iFinChar;
207 :
208 137 : GUInt32 nCount = TABSIZE - 256;
209 :
210 : // Decompress the input buffer
211 4783380 : while (nSizeIn > 0)
212 : {
213 : // Fetch 12-bit code from input stream
214 4783300 : if (bBitsleft)
215 : {
216 2391720 : iCode = ((*pabyIn++ & 0x0F) << 8) & 0xF00;
217 2391720 : nSizeIn--;
218 2391720 : if (nSizeIn == 0)
219 57 : break;
220 2391660 : iCode += *pabyIn++;
221 2391660 : nSizeIn--;
222 2391660 : bBitsleft = FALSE;
223 : }
224 : else
225 : {
226 2391580 : iCode = (*pabyIn++ << 4) & 0xFF0;
227 2391580 : nSizeIn--;
228 2391580 : if (nSizeIn == 0)
229 0 : break;
230 2391580 : iCode += (*pabyIn >> 4) & 0x00F;
231 2391580 : bBitsleft = TRUE;
232 : }
233 :
234 4783250 : const GUInt32 iInCode = iCode;
235 4783250 : GByte bLastChar = 0; // TODO(schwehr): Why not nLastChar?
236 :
237 : // Do we have unknown code?
238 4783250 : bool bNewCode = false;
239 4783250 : if (!poCodeTab[iCode].bUsed)
240 : {
241 40101 : iCode = iOldCode;
242 40101 : bLastChar = iFinChar;
243 40101 : bNewCode = true;
244 : }
245 :
246 4783250 : GByte abyStack[STACKSIZE] = {};
247 4783250 : GByte *pabyTail = abyStack + STACKSIZE;
248 4783250 : GUInt32 nStackCount = 0;
249 :
250 16371000 : while (poCodeTab[iCode].iPredecessor != NO_PRED)
251 : {
252 : // Stack overrun
253 11587700 : if (nStackCount >= STACKSIZE)
254 0 : return 0;
255 : // Put the decoded character into stack
256 11587700 : *(--pabyTail) = poCodeTab[iCode].iFollower;
257 11587700 : nStackCount++;
258 11587700 : iCode = poCodeTab[iCode].iPredecessor;
259 : }
260 :
261 4783250 : if (!nSizeOut)
262 0 : return 0;
263 : // The first character
264 4783250 : iFinChar = poCodeTab[iCode].iFollower;
265 4783250 : nSizeOut--;
266 4783250 : *pabyOut++ = iFinChar;
267 :
268 : // Output buffer overrun
269 4783250 : if (nStackCount > nSizeOut)
270 0 : return 0;
271 :
272 : // Now copy the stack contents into output buffer. Our stack was
273 : // filled in reverse order, so no need in character reordering
274 4783250 : memcpy(pabyOut, pabyTail, nStackCount);
275 4783250 : nSizeOut -= nStackCount;
276 4783250 : pabyOut += nStackCount;
277 :
278 : // If code isn't known
279 4783250 : if (bNewCode)
280 : {
281 : // Output buffer overrun
282 40101 : if (!nSizeOut)
283 0 : return 0;
284 40101 : iFinChar = bLastChar; // the follower char of last
285 40101 : *pabyOut++ = iFinChar;
286 40101 : nSizeOut--;
287 : }
288 :
289 4783250 : if (nCount > 0)
290 : {
291 463213 : nCount--;
292 : // Add code to the table
293 463213 : LZWUpdateTab(poCodeTab, iOldCode, iFinChar);
294 : }
295 :
296 4783250 : iOldCode = iInCode;
297 : }
298 :
299 137 : return static_cast<size_t>(pabyOut - pabyOutBegin);
300 : }
301 :
302 : /************************************************************************/
303 : /* LZWDecompress() */
304 : /************************************************************************/
305 :
306 137 : size_t RMFDataset::LZWDecompress(const GByte *pabyIn, GUInt32 nSizeIn,
307 : GByte *pabyOut, GUInt32 nSizeOut, GUInt32,
308 : GUInt32)
309 : {
310 137 : if (pabyIn == nullptr || pabyOut == nullptr || nSizeIn < 2)
311 0 : return 0;
312 137 : LZWStringTab *poCodeTab = LZWCreateTab();
313 :
314 137 : size_t nRet = LZWReadStream(pabyIn, nSizeIn, pabyOut, nSizeOut, poCodeTab);
315 :
316 137 : CPLFree(poCodeTab);
317 :
318 137 : return nRet;
319 : }
320 :
321 : /************************************************************************/
322 : /* LZWWriteStream() */
323 : /************************************************************************/
324 :
325 124 : static size_t LZWWriteStream(const GByte *pabyIn, GUInt32 nSizeIn,
326 : GByte *pabyOut, GUInt32 nSizeOut,
327 : LZWStringTab *poCodeTab)
328 : {
329 : GUInt32 iCode;
330 124 : iCode = LZWFindIndex(poCodeTab, NO_PRED, *pabyIn++);
331 124 : nSizeIn--;
332 :
333 124 : GUInt32 nCount = TABSIZE - 256;
334 124 : GUInt32 iTmp = 0;
335 124 : bool bBitsleft = true;
336 124 : GByte *pabyCurrent = pabyOut;
337 124 : GByte *pabyOutEnd = pabyOut + nSizeOut;
338 :
339 14321400 : while (nSizeIn > 0)
340 : {
341 14321300 : const GByte bCurrentCode = *pabyIn++;
342 14321300 : nSizeIn--;
343 :
344 14321300 : GUInt32 iNextCode = LZWFindIndex(poCodeTab, iCode, bCurrentCode);
345 14373900 : if (iNextCode != NOT_FND)
346 : {
347 9771700 : iCode = iNextCode;
348 9771700 : continue;
349 : }
350 :
351 4602210 : if (!LZWPutCode(iCode, iTmp, bBitsleft, pabyCurrent, pabyOutEnd))
352 : {
353 28 : return 0;
354 : }
355 :
356 5211410 : if (nCount > 0)
357 : {
358 451343 : nCount--;
359 451343 : LZWUpdateTab(poCodeTab, iCode, bCurrentCode);
360 : }
361 :
362 5212320 : iCode = LZWFindIndex(poCodeTab, NO_PRED, bCurrentCode);
363 : }
364 :
365 96 : if (!LZWPutCode(iCode, iTmp, bBitsleft, pabyCurrent, pabyOutEnd))
366 : {
367 0 : return 0;
368 : }
369 :
370 96 : if (!bBitsleft)
371 : {
372 52 : if (pabyCurrent >= pabyOutEnd)
373 : {
374 0 : return 0;
375 : }
376 52 : *(pabyCurrent++) = static_cast<GByte>((iTmp << 4) & 0xFF0);
377 : }
378 :
379 96 : return static_cast<size_t>(pabyCurrent - pabyOut);
380 : }
381 :
382 : /************************************************************************/
383 : /* LZWCompress() */
384 : /************************************************************************/
385 :
386 124 : size_t RMFDataset::LZWCompress(const GByte *pabyIn, GUInt32 nSizeIn,
387 : GByte *pabyOut, GUInt32 nSizeOut, GUInt32,
388 : GUInt32, const RMFDataset *)
389 : {
390 124 : if (pabyIn == nullptr || pabyOut == nullptr || nSizeIn == 0)
391 0 : return 0;
392 :
393 : // Allocate space for the new table and pre-fill it
394 124 : LZWStringTab *poCodeTab = LZWCreateTab();
395 :
396 : size_t nWritten =
397 124 : LZWWriteStream(pabyIn, nSizeIn, pabyOut, nSizeOut, poCodeTab);
398 :
399 124 : CPLFree(poCodeTab);
400 :
401 124 : return nWritten;
402 : }
|