OLD | NEW |
| (Empty) |
1 // Copyright 2015 PDFium Authors. All rights reserved. | |
2 // Use of this source code is governed by a BSD-style license that can be | |
3 // found in the LICENSE file. | |
4 | |
5 // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com | |
6 | |
7 #include "core/src/fxcodec/jbig2/JBig2_TrdProc.h" | |
8 | |
9 #include <memory> | |
10 | |
11 #include "core/src/fxcodec/jbig2/JBig2_ArithDecoder.h" | |
12 #include "core/src/fxcodec/jbig2/JBig2_ArithIntDecoder.h" | |
13 #include "core/src/fxcodec/jbig2/JBig2_GrrdProc.h" | |
14 #include "core/src/fxcodec/jbig2/JBig2_HuffmanDecoder.h" | |
15 | |
16 CJBig2_Image* CJBig2_TRDProc::decode_Huffman(CJBig2_BitStream* pStream, | |
17 JBig2ArithCtx* grContext) { | |
18 int32_t STRIPT, FIRSTS; | |
19 FX_DWORD NINSTANCES; | |
20 int32_t DT, DFS, CURS; | |
21 int32_t SI, TI; | |
22 CJBig2_Image* IBI; | |
23 FX_DWORD WI, HI; | |
24 int32_t IDS; | |
25 FX_BOOL RI; | |
26 int32_t RDWI, RDHI, RDXI, RDYI; | |
27 CJBig2_Image* IBOI; | |
28 FX_DWORD WOI, HOI; | |
29 FX_BOOL bFirst; | |
30 FX_DWORD nTmp; | |
31 int32_t nVal, nBits; | |
32 std::unique_ptr<CJBig2_HuffmanDecoder> pHuffmanDecoder( | |
33 new CJBig2_HuffmanDecoder(pStream)); | |
34 std::unique_ptr<CJBig2_Image> SBREG(new CJBig2_Image(SBW, SBH)); | |
35 SBREG->fill(SBDEFPIXEL); | |
36 if (pHuffmanDecoder->decodeAValue(SBHUFFDT, &STRIPT) != 0) | |
37 return nullptr; | |
38 | |
39 STRIPT *= SBSTRIPS; | |
40 STRIPT = -STRIPT; | |
41 FIRSTS = 0; | |
42 NINSTANCES = 0; | |
43 while (NINSTANCES < SBNUMINSTANCES) { | |
44 if (pHuffmanDecoder->decodeAValue(SBHUFFDT, &DT) != 0) | |
45 return nullptr; | |
46 | |
47 DT *= SBSTRIPS; | |
48 STRIPT = STRIPT + DT; | |
49 bFirst = TRUE; | |
50 for (;;) { | |
51 if (bFirst) { | |
52 if (pHuffmanDecoder->decodeAValue(SBHUFFFS, &DFS) != 0) | |
53 return nullptr; | |
54 | |
55 FIRSTS = FIRSTS + DFS; | |
56 CURS = FIRSTS; | |
57 bFirst = FALSE; | |
58 } else { | |
59 nVal = pHuffmanDecoder->decodeAValue(SBHUFFDS, &IDS); | |
60 if (nVal == JBIG2_OOB) { | |
61 break; | |
62 } else if (nVal != 0) { | |
63 return nullptr; | |
64 } else { | |
65 CURS = CURS + IDS + SBDSOFFSET; | |
66 } | |
67 } | |
68 uint8_t CURT = 0; | |
69 if (SBSTRIPS != 1) { | |
70 nTmp = 1; | |
71 while ((FX_DWORD)(1 << nTmp) < SBSTRIPS) { | |
72 nTmp++; | |
73 } | |
74 if (pStream->readNBits(nTmp, &nVal) != 0) | |
75 return nullptr; | |
76 | |
77 CURT = nVal; | |
78 } | |
79 TI = STRIPT + CURT; | |
80 nVal = 0; | |
81 nBits = 0; | |
82 FX_DWORD IDI; | |
83 for (;;) { | |
84 if (pStream->read1Bit(&nTmp) != 0) | |
85 return nullptr; | |
86 | |
87 nVal = (nVal << 1) | nTmp; | |
88 nBits++; | |
89 for (IDI = 0; IDI < SBNUMSYMS; IDI++) { | |
90 if ((nBits == SBSYMCODES[IDI].codelen) && | |
91 (nVal == SBSYMCODES[IDI].code)) { | |
92 break; | |
93 } | |
94 } | |
95 if (IDI < SBNUMSYMS) { | |
96 break; | |
97 } | |
98 } | |
99 if (SBREFINE == 0) { | |
100 RI = 0; | |
101 } else { | |
102 if (pStream->read1Bit(&RI) != 0) { | |
103 return nullptr; | |
104 } | |
105 } | |
106 if (RI == 0) { | |
107 IBI = SBSYMS[IDI]; | |
108 } else { | |
109 if ((pHuffmanDecoder->decodeAValue(SBHUFFRDW, &RDWI) != 0) || | |
110 (pHuffmanDecoder->decodeAValue(SBHUFFRDH, &RDHI) != 0) || | |
111 (pHuffmanDecoder->decodeAValue(SBHUFFRDX, &RDXI) != 0) || | |
112 (pHuffmanDecoder->decodeAValue(SBHUFFRDY, &RDYI) != 0) || | |
113 (pHuffmanDecoder->decodeAValue(SBHUFFRSIZE, &nVal) != 0)) { | |
114 return nullptr; | |
115 } | |
116 pStream->alignByte(); | |
117 nTmp = pStream->getOffset(); | |
118 IBOI = SBSYMS[IDI]; | |
119 if (!IBOI) | |
120 return nullptr; | |
121 | |
122 WOI = IBOI->m_nWidth; | |
123 HOI = IBOI->m_nHeight; | |
124 if ((int)(WOI + RDWI) < 0 || (int)(HOI + RDHI) < 0) | |
125 return nullptr; | |
126 | |
127 std::unique_ptr<CJBig2_GRRDProc> pGRRD(new CJBig2_GRRDProc()); | |
128 pGRRD->GRW = WOI + RDWI; | |
129 pGRRD->GRH = HOI + RDHI; | |
130 pGRRD->GRTEMPLATE = SBRTEMPLATE; | |
131 pGRRD->GRREFERENCE = IBOI; | |
132 pGRRD->GRREFERENCEDX = (RDWI >> 2) + RDXI; | |
133 pGRRD->GRREFERENCEDY = (RDHI >> 2) + RDYI; | |
134 pGRRD->TPGRON = 0; | |
135 pGRRD->GRAT[0] = SBRAT[0]; | |
136 pGRRD->GRAT[1] = SBRAT[1]; | |
137 pGRRD->GRAT[2] = SBRAT[2]; | |
138 pGRRD->GRAT[3] = SBRAT[3]; | |
139 | |
140 { | |
141 std::unique_ptr<CJBig2_ArithDecoder> pArithDecoder( | |
142 new CJBig2_ArithDecoder(pStream)); | |
143 IBI = pGRRD->decode(pArithDecoder.get(), grContext); | |
144 if (!IBI) | |
145 return nullptr; | |
146 } | |
147 | |
148 pStream->alignByte(); | |
149 pStream->offset(2); | |
150 if ((FX_DWORD)nVal != (pStream->getOffset() - nTmp)) { | |
151 delete IBI; | |
152 return nullptr; | |
153 } | |
154 } | |
155 if (!IBI) { | |
156 continue; | |
157 } | |
158 WI = IBI->m_nWidth; | |
159 HI = IBI->m_nHeight; | |
160 if (TRANSPOSED == 0 && ((REFCORNER == JBIG2_CORNER_TOPRIGHT) || | |
161 (REFCORNER == JBIG2_CORNER_BOTTOMRIGHT))) { | |
162 CURS = CURS + WI - 1; | |
163 } else if (TRANSPOSED == 1 && ((REFCORNER == JBIG2_CORNER_BOTTOMLEFT) || | |
164 (REFCORNER == JBIG2_CORNER_BOTTOMRIGHT))) { | |
165 CURS = CURS + HI - 1; | |
166 } | |
167 SI = CURS; | |
168 if (TRANSPOSED == 0) { | |
169 switch (REFCORNER) { | |
170 case JBIG2_CORNER_TOPLEFT: | |
171 SBREG->composeFrom(SI, TI, IBI, SBCOMBOP); | |
172 break; | |
173 case JBIG2_CORNER_TOPRIGHT: | |
174 SBREG->composeFrom(SI - WI + 1, TI, IBI, SBCOMBOP); | |
175 break; | |
176 case JBIG2_CORNER_BOTTOMLEFT: | |
177 SBREG->composeFrom(SI, TI - HI + 1, IBI, SBCOMBOP); | |
178 break; | |
179 case JBIG2_CORNER_BOTTOMRIGHT: | |
180 SBREG->composeFrom(SI - WI + 1, TI - HI + 1, IBI, SBCOMBOP); | |
181 break; | |
182 } | |
183 } else { | |
184 switch (REFCORNER) { | |
185 case JBIG2_CORNER_TOPLEFT: | |
186 SBREG->composeFrom(TI, SI, IBI, SBCOMBOP); | |
187 break; | |
188 case JBIG2_CORNER_TOPRIGHT: | |
189 SBREG->composeFrom(TI - WI + 1, SI, IBI, SBCOMBOP); | |
190 break; | |
191 case JBIG2_CORNER_BOTTOMLEFT: | |
192 SBREG->composeFrom(TI, SI - HI + 1, IBI, SBCOMBOP); | |
193 break; | |
194 case JBIG2_CORNER_BOTTOMRIGHT: | |
195 SBREG->composeFrom(TI - WI + 1, SI - HI + 1, IBI, SBCOMBOP); | |
196 break; | |
197 } | |
198 } | |
199 if (RI != 0) { | |
200 delete IBI; | |
201 } | |
202 if (TRANSPOSED == 0 && ((REFCORNER == JBIG2_CORNER_TOPLEFT) || | |
203 (REFCORNER == JBIG2_CORNER_BOTTOMLEFT))) { | |
204 CURS = CURS + WI - 1; | |
205 } else if (TRANSPOSED == 1 && ((REFCORNER == JBIG2_CORNER_TOPLEFT) || | |
206 (REFCORNER == JBIG2_CORNER_TOPRIGHT))) { | |
207 CURS = CURS + HI - 1; | |
208 } | |
209 NINSTANCES = NINSTANCES + 1; | |
210 } | |
211 } | |
212 return SBREG.release(); | |
213 } | |
214 | |
215 CJBig2_Image* CJBig2_TRDProc::decode_Arith(CJBig2_ArithDecoder* pArithDecoder, | |
216 JBig2ArithCtx* grContext, | |
217 JBig2IntDecoderState* pIDS) { | |
218 std::unique_ptr<CJBig2_ArithIntDecoder> IADT; | |
219 std::unique_ptr<CJBig2_ArithIntDecoder> IAFS; | |
220 std::unique_ptr<CJBig2_ArithIntDecoder> IADS; | |
221 std::unique_ptr<CJBig2_ArithIntDecoder> IAIT; | |
222 std::unique_ptr<CJBig2_ArithIntDecoder> IARI; | |
223 std::unique_ptr<CJBig2_ArithIntDecoder> IARDW; | |
224 std::unique_ptr<CJBig2_ArithIntDecoder> IARDH; | |
225 std::unique_ptr<CJBig2_ArithIntDecoder> IARDX; | |
226 std::unique_ptr<CJBig2_ArithIntDecoder> IARDY; | |
227 std::unique_ptr<CJBig2_ArithIaidDecoder> IAID; | |
228 CJBig2_ArithIntDecoder* pIADT; | |
229 CJBig2_ArithIntDecoder* pIAFS; | |
230 CJBig2_ArithIntDecoder* pIADS; | |
231 CJBig2_ArithIntDecoder* pIAIT; | |
232 CJBig2_ArithIntDecoder* pIARI; | |
233 CJBig2_ArithIntDecoder* pIARDW; | |
234 CJBig2_ArithIntDecoder* pIARDH; | |
235 CJBig2_ArithIntDecoder* pIARDX; | |
236 CJBig2_ArithIntDecoder* pIARDY; | |
237 CJBig2_ArithIaidDecoder* pIAID; | |
238 if (pIDS) { | |
239 pIADT = pIDS->IADT; | |
240 pIAFS = pIDS->IAFS; | |
241 pIADS = pIDS->IADS; | |
242 pIAIT = pIDS->IAIT; | |
243 pIARI = pIDS->IARI; | |
244 pIARDW = pIDS->IARDW; | |
245 pIARDH = pIDS->IARDH; | |
246 pIARDX = pIDS->IARDX; | |
247 pIARDY = pIDS->IARDY; | |
248 pIAID = pIDS->IAID; | |
249 } else { | |
250 IADT.reset(new CJBig2_ArithIntDecoder()); | |
251 IAFS.reset(new CJBig2_ArithIntDecoder()); | |
252 IADS.reset(new CJBig2_ArithIntDecoder()); | |
253 IAIT.reset(new CJBig2_ArithIntDecoder()); | |
254 IARI.reset(new CJBig2_ArithIntDecoder()); | |
255 IARDW.reset(new CJBig2_ArithIntDecoder()); | |
256 IARDH.reset(new CJBig2_ArithIntDecoder()); | |
257 IARDX.reset(new CJBig2_ArithIntDecoder()); | |
258 IARDY.reset(new CJBig2_ArithIntDecoder()); | |
259 IAID.reset(new CJBig2_ArithIaidDecoder(SBSYMCODELEN)); | |
260 pIADT = IADT.get(); | |
261 pIAFS = IAFS.get(); | |
262 pIADS = IADS.get(); | |
263 pIAIT = IAIT.get(); | |
264 pIARI = IARI.get(); | |
265 pIARDW = IARDW.get(); | |
266 pIARDH = IARDH.get(); | |
267 pIARDX = IARDX.get(); | |
268 pIARDY = IARDY.get(); | |
269 pIAID = IAID.get(); | |
270 } | |
271 std::unique_ptr<CJBig2_Image> SBREG(new CJBig2_Image(SBW, SBH)); | |
272 SBREG->fill(SBDEFPIXEL); | |
273 int32_t STRIPT; | |
274 pIADT->decode(pArithDecoder, &STRIPT); | |
275 STRIPT *= SBSTRIPS; | |
276 STRIPT = -STRIPT; | |
277 int32_t FIRSTS = 0; | |
278 FX_DWORD NINSTANCES = 0; | |
279 while (NINSTANCES < SBNUMINSTANCES) { | |
280 int32_t CURS; | |
281 int32_t DT; | |
282 pIADT->decode(pArithDecoder, &DT); | |
283 DT *= SBSTRIPS; | |
284 STRIPT += DT; | |
285 bool bFirst = true; | |
286 for (;;) { | |
287 if (bFirst) { | |
288 int32_t DFS; | |
289 pIAFS->decode(pArithDecoder, &DFS); | |
290 FIRSTS += DFS; | |
291 CURS = FIRSTS; | |
292 bFirst = false; | |
293 } else { | |
294 int32_t IDS; | |
295 if (!pIADS->decode(pArithDecoder, &IDS)) | |
296 break; | |
297 CURS += IDS + SBDSOFFSET; | |
298 } | |
299 if (NINSTANCES >= SBNUMINSTANCES) { | |
300 break; | |
301 } | |
302 int CURT = 0; | |
303 if (SBSTRIPS != 1) | |
304 pIAIT->decode(pArithDecoder, &CURT); | |
305 | |
306 int32_t TI = STRIPT + CURT; | |
307 FX_DWORD IDI; | |
308 pIAID->decode(pArithDecoder, &IDI); | |
309 if (IDI >= SBNUMSYMS) | |
310 return nullptr; | |
311 | |
312 int RI; | |
313 if (SBREFINE == 0) | |
314 RI = 0; | |
315 else | |
316 pIARI->decode(pArithDecoder, &RI); | |
317 | |
318 std::unique_ptr<CJBig2_Image> IBI; | |
319 CJBig2_Image* pIBI; | |
320 if (RI == 0) { | |
321 pIBI = SBSYMS[IDI]; | |
322 } else { | |
323 int32_t RDWI; | |
324 int32_t RDHI; | |
325 int32_t RDXI; | |
326 int32_t RDYI; | |
327 pIARDW->decode(pArithDecoder, &RDWI); | |
328 pIARDH->decode(pArithDecoder, &RDHI); | |
329 pIARDX->decode(pArithDecoder, &RDXI); | |
330 pIARDY->decode(pArithDecoder, &RDYI); | |
331 CJBig2_Image* IBOI = SBSYMS[IDI]; | |
332 FX_DWORD WOI = IBOI->m_nWidth; | |
333 FX_DWORD HOI = IBOI->m_nHeight; | |
334 if ((int)(WOI + RDWI) < 0 || (int)(HOI + RDHI) < 0) { | |
335 return nullptr; | |
336 } | |
337 std::unique_ptr<CJBig2_GRRDProc> pGRRD(new CJBig2_GRRDProc()); | |
338 pGRRD->GRW = WOI + RDWI; | |
339 pGRRD->GRH = HOI + RDHI; | |
340 pGRRD->GRTEMPLATE = SBRTEMPLATE; | |
341 pGRRD->GRREFERENCE = IBOI; | |
342 pGRRD->GRREFERENCEDX = (RDWI >> 1) + RDXI; | |
343 pGRRD->GRREFERENCEDY = (RDHI >> 1) + RDYI; | |
344 pGRRD->TPGRON = 0; | |
345 pGRRD->GRAT[0] = SBRAT[0]; | |
346 pGRRD->GRAT[1] = SBRAT[1]; | |
347 pGRRD->GRAT[2] = SBRAT[2]; | |
348 pGRRD->GRAT[3] = SBRAT[3]; | |
349 IBI.reset(pGRRD->decode(pArithDecoder, grContext)); | |
350 pIBI = IBI.get(); | |
351 } | |
352 if (!pIBI) | |
353 return nullptr; | |
354 | |
355 FX_DWORD WI = pIBI->m_nWidth; | |
356 FX_DWORD HI = pIBI->m_nHeight; | |
357 if (TRANSPOSED == 0 && ((REFCORNER == JBIG2_CORNER_TOPRIGHT) || | |
358 (REFCORNER == JBIG2_CORNER_BOTTOMRIGHT))) { | |
359 CURS += WI - 1; | |
360 } else if (TRANSPOSED == 1 && ((REFCORNER == JBIG2_CORNER_BOTTOMLEFT) || | |
361 (REFCORNER == JBIG2_CORNER_BOTTOMRIGHT))) { | |
362 CURS += HI - 1; | |
363 } | |
364 int32_t SI = CURS; | |
365 if (TRANSPOSED == 0) { | |
366 switch (REFCORNER) { | |
367 case JBIG2_CORNER_TOPLEFT: | |
368 SBREG->composeFrom(SI, TI, pIBI, SBCOMBOP); | |
369 break; | |
370 case JBIG2_CORNER_TOPRIGHT: | |
371 SBREG->composeFrom(SI - WI + 1, TI, pIBI, SBCOMBOP); | |
372 break; | |
373 case JBIG2_CORNER_BOTTOMLEFT: | |
374 SBREG->composeFrom(SI, TI - HI + 1, pIBI, SBCOMBOP); | |
375 break; | |
376 case JBIG2_CORNER_BOTTOMRIGHT: | |
377 SBREG->composeFrom(SI - WI + 1, TI - HI + 1, pIBI, SBCOMBOP); | |
378 break; | |
379 } | |
380 } else { | |
381 switch (REFCORNER) { | |
382 case JBIG2_CORNER_TOPLEFT: | |
383 SBREG->composeFrom(TI, SI, pIBI, SBCOMBOP); | |
384 break; | |
385 case JBIG2_CORNER_TOPRIGHT: | |
386 SBREG->composeFrom(TI - WI + 1, SI, pIBI, SBCOMBOP); | |
387 break; | |
388 case JBIG2_CORNER_BOTTOMLEFT: | |
389 SBREG->composeFrom(TI, SI - HI + 1, pIBI, SBCOMBOP); | |
390 break; | |
391 case JBIG2_CORNER_BOTTOMRIGHT: | |
392 SBREG->composeFrom(TI - WI + 1, SI - HI + 1, pIBI, SBCOMBOP); | |
393 break; | |
394 } | |
395 } | |
396 if (TRANSPOSED == 0 && ((REFCORNER == JBIG2_CORNER_TOPLEFT) || | |
397 (REFCORNER == JBIG2_CORNER_BOTTOMLEFT))) { | |
398 CURS += WI - 1; | |
399 } else if (TRANSPOSED == 1 && ((REFCORNER == JBIG2_CORNER_TOPLEFT) || | |
400 (REFCORNER == JBIG2_CORNER_TOPRIGHT))) { | |
401 CURS += HI - 1; | |
402 } | |
403 ++NINSTANCES; | |
404 } | |
405 } | |
406 return SBREG.release(); | |
407 } | |
OLD | NEW |