Update the trunk to the OpenCV's CVS (2008-07-14)
[opencv] / otherlibs / _graphics / src / libjasper / jpc_t1dec.c
1 /*
2  * Copyright (c) 1999-2000 Image Power, Inc. and the University of
3  *   British Columbia.
4  * Copyright (c) 2001-2003 Michael David Adams.
5  * All rights reserved.
6  */
7
8 /* __START_OF_JASPER_LICENSE__
9  * 
10  * JasPer License Version 2.0
11  * 
12  * Copyright (c) 2001-2006 Michael David Adams
13  * Copyright (c) 1999-2000 Image Power, Inc.
14  * Copyright (c) 1999-2000 The University of British Columbia
15  * 
16  * All rights reserved.
17  * 
18  * Permission is hereby granted, free of charge, to any person (the
19  * "User") obtaining a copy of this software and associated documentation
20  * files (the "Software"), to deal in the Software without restriction,
21  * including without limitation the rights to use, copy, modify, merge,
22  * publish, distribute, and/or sell copies of the Software, and to permit
23  * persons to whom the Software is furnished to do so, subject to the
24  * following conditions:
25  * 
26  * 1.  The above copyright notices and this permission notice (which
27  * includes the disclaimer below) shall be included in all copies or
28  * substantial portions of the Software.
29  * 
30  * 2.  The name of a copyright holder shall not be used to endorse or
31  * promote products derived from the Software without specific prior
32  * written permission.
33  * 
34  * THIS DISCLAIMER OF WARRANTY CONSTITUTES AN ESSENTIAL PART OF THIS
35  * LICENSE.  NO USE OF THE SOFTWARE IS AUTHORIZED HEREUNDER EXCEPT UNDER
36  * THIS DISCLAIMER.  THE SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS
37  * "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING
38  * BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A
39  * PARTICULAR PURPOSE AND NONINFRINGEMENT OF THIRD PARTY RIGHTS.  IN NO
40  * EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, OR ANY SPECIAL
41  * INDIRECT OR CONSEQUENTIAL DAMAGES, OR ANY DAMAGES WHATSOEVER RESULTING
42  * FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT,
43  * NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION
44  * WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.  NO ASSURANCES ARE
45  * PROVIDED BY THE COPYRIGHT HOLDERS THAT THE SOFTWARE DOES NOT INFRINGE
46  * THE PATENT OR OTHER INTELLECTUAL PROPERTY RIGHTS OF ANY OTHER ENTITY.
47  * EACH COPYRIGHT HOLDER DISCLAIMS ANY LIABILITY TO THE USER FOR CLAIMS
48  * BROUGHT BY ANY OTHER ENTITY BASED ON INFRINGEMENT OF INTELLECTUAL
49  * PROPERTY RIGHTS OR OTHERWISE.  AS A CONDITION TO EXERCISING THE RIGHTS
50  * GRANTED HEREUNDER, EACH USER HEREBY ASSUMES SOLE RESPONSIBILITY TO SECURE
51  * ANY OTHER INTELLECTUAL PROPERTY RIGHTS NEEDED, IF ANY.  THE SOFTWARE
52  * IS NOT FAULT-TOLERANT AND IS NOT INTENDED FOR USE IN MISSION-CRITICAL
53  * SYSTEMS, SUCH AS THOSE USED IN THE OPERATION OF NUCLEAR FACILITIES,
54  * AIRCRAFT NAVIGATION OR COMMUNICATION SYSTEMS, AIR TRAFFIC CONTROL
55  * SYSTEMS, DIRECT LIFE SUPPORT MACHINES, OR WEAPONS SYSTEMS, IN WHICH
56  * THE FAILURE OF THE SOFTWARE OR SYSTEM COULD LEAD DIRECTLY TO DEATH,
57  * PERSONAL INJURY, OR SEVERE PHYSICAL OR ENVIRONMENTAL DAMAGE ("HIGH
58  * RISK ACTIVITIES").  THE COPYRIGHT HOLDERS SPECIFICALLY DISCLAIM ANY
59  * EXPRESS OR IMPLIED WARRANTY OF FITNESS FOR HIGH RISK ACTIVITIES.
60  * 
61  * __END_OF_JASPER_LICENSE__
62  */
63
64 /*
65  * Tier 1 Decoder
66  *
67  * $Id: jpc_t1dec.c,v 1.2 2008/05/26 09:40:52 vp153 Exp $
68  */
69
70 /******************************************************************************\
71 * Includes.
72 \******************************************************************************/
73
74 #include <stdio.h>
75 #include <stdlib.h>
76 #include <assert.h>
77
78 #include "jasper/jas_fix.h"
79 #include "jasper/jas_stream.h"
80 #include "jasper/jas_math.h"
81
82 #include "jpc_bs.h"
83 #include "jpc_mqdec.h"
84 #include "jpc_t1dec.h"
85 #include "jpc_t1cod.h"
86 #include "jpc_dec.h"
87
88 /******************************************************************************\
89 *
90 \******************************************************************************/
91
92 static int jpc_dec_decodecblk(jpc_dec_t *dec, jpc_dec_tile_t *tile, jpc_dec_tcomp_t *tcomp, jpc_dec_band_t *band,
93   jpc_dec_cblk_t *cblk, int dopartial, int maxlyrs);
94 static int dec_sigpass(jpc_dec_t *dec, jpc_mqdec_t *mqdec, int bitpos, int orient,
95   int vcausalflag, jas_matrix_t *flags, jas_matrix_t *data);
96 static int dec_rawsigpass(jpc_dec_t *dec, jpc_bitstream_t *in, int bitpos,
97   int vcausalflag, jas_matrix_t *flags, jas_matrix_t *data);
98 static int dec_refpass(jpc_dec_t *dec, jpc_mqdec_t *mqdec, int bitpos, int vcausalflag,
99   jas_matrix_t *flags, jas_matrix_t *data);
100 static int dec_rawrefpass(jpc_dec_t *dec, jpc_bitstream_t *in, int bitpos,
101   int vcausalflag, jas_matrix_t *flags, jas_matrix_t *data);
102 static int dec_clnpass(jpc_dec_t *dec, jpc_mqdec_t *mqdec, int bitpos, int orient,
103   int vcausalflag, int segsymflag, jas_matrix_t *flags, jas_matrix_t *data);
104
105 #if defined(DEBUG)
106 static long t1dec_cnt = 0;
107 #endif
108
109 #if !defined(DEBUG)
110 #define JPC_T1D_GETBIT(mqdec, v, passtypename, symtypename) \
111         ((v) = jpc_mqdec_getbit(mqdec))
112 #else
113 #define JPC_T1D_GETBIT(mqdec, v, passtypename, symtypename) \
114 { \
115         (v) = jpc_mqdec_getbit(mqdec); \
116         if (jas_getdbglevel() >= 100) { \
117                 jas_eprintf("index = %ld; passtype = %s; symtype = %s; sym = %d\n", t1dec_cnt, passtypename, symtypename, v); \
118                 ++t1dec_cnt; \
119         } \
120 }
121 #endif
122 #define JPC_T1D_GETBITNOSKEW(mqdec, v, passtypename, symtypename) \
123         JPC_T1D_GETBIT(mqdec, v, passtypename, symtypename)
124
125 #if !defined(DEBUG)
126 #define JPC_T1D_RAWGETBIT(bitstream, v, passtypename, symtypename) \
127         ((v) = jpc_bitstream_getbit(bitstream))
128 #else
129 #define JPC_T1D_RAWGETBIT(bitstream, v, passtypename, symtypename) \
130 { \
131         (v) = jpc_bitstream_getbit(bitstream); \
132         if (jas_getdbglevel() >= 100) { \
133                 jas_eprintf("index = %ld; passtype = %s; symtype = %s; sym = %d\n", t1dec_cnt, passtypename, symtypename, v); \
134                 ++t1dec_cnt; \
135         } \
136 }
137 #endif
138
139 /******************************************************************************\
140 * Code.
141 \******************************************************************************/
142
143 int jpc_dec_decodecblks(jpc_dec_t *dec, jpc_dec_tile_t *tile)
144 {
145         jpc_dec_tcomp_t *tcomp;
146         int compcnt;
147         jpc_dec_rlvl_t *rlvl;
148         int rlvlcnt;
149         jpc_dec_band_t *band;
150         int bandcnt;
151         jpc_dec_prc_t *prc;
152         int prccnt;
153         jpc_dec_cblk_t *cblk;
154         int cblkcnt;
155
156         for (compcnt = dec->numcomps, tcomp = tile->tcomps; compcnt > 0;
157           --compcnt, ++tcomp) {
158                 for (rlvlcnt = tcomp->numrlvls, rlvl = tcomp->rlvls;
159                   rlvlcnt > 0; --rlvlcnt, ++rlvl) {
160                         if (!rlvl->bands) {
161                                 continue;
162                         }
163                         for (bandcnt = rlvl->numbands, band = rlvl->bands;
164                           bandcnt > 0; --bandcnt, ++band) {
165                                 if (!band->data) {
166                                         continue;
167                                 }
168                                 for (prccnt = rlvl->numprcs, prc = band->prcs;
169                                   prccnt > 0; --prccnt, ++prc) {
170                                         if (!prc->cblks) {
171                                                 continue;
172                                         }
173                                         for (cblkcnt = prc->numcblks,
174                                           cblk = prc->cblks; cblkcnt > 0;
175                                           --cblkcnt, ++cblk) {
176                                                 if (jpc_dec_decodecblk(dec, tile, tcomp,
177                                                   band, cblk, 1, JPC_MAXLYRS)) {
178                                                         return -1;
179                                                 }
180                                         }
181                                 }
182
183                         }
184                 }
185         }
186
187         return 0;
188 }
189
190 static int jpc_dec_decodecblk(jpc_dec_t *dec, jpc_dec_tile_t *tile, jpc_dec_tcomp_t *tcomp, jpc_dec_band_t *band,
191   jpc_dec_cblk_t *cblk, int dopartial, int maxlyrs)
192 {
193         jpc_dec_seg_t *seg;
194         int i;
195         int bpno;
196         int passtype;
197         int ret;
198         int compno;
199         int filldata;
200         int fillmask;
201         jpc_dec_ccp_t *ccp;
202
203         compno = tcomp - tile->tcomps;
204
205         if (!cblk->flags) {
206                 /* Note: matrix is assumed to be zeroed */
207                 if (!(cblk->flags = jas_matrix_create(jas_matrix_numrows(cblk->data) +
208                   2, jas_matrix_numcols(cblk->data) + 2))) {
209                         return -1;
210                 }
211         }
212
213         seg = cblk->segs.head;
214         while (seg && (seg != cblk->curseg || dopartial) && (maxlyrs < 0 ||
215           seg->lyrno < maxlyrs)) {
216                 assert(seg->numpasses >= seg->maxpasses || dopartial);
217                 assert(seg->stream);
218                 jas_stream_rewind(seg->stream);
219                 jas_stream_setrwcount(seg->stream, 0);
220                 if (seg->type == JPC_SEG_MQ) {
221                         if (!cblk->mqdec) {
222                                 if (!(cblk->mqdec = jpc_mqdec_create(JPC_NUMCTXS, 0))) {
223                                         return -1;
224                                 }
225                                 jpc_mqdec_setctxs(cblk->mqdec, JPC_NUMCTXS, jpc_mqctxs);
226                         }
227                         jpc_mqdec_setinput(cblk->mqdec, seg->stream);
228                         jpc_mqdec_init(cblk->mqdec);
229                 } else {
230                         assert(seg->type == JPC_SEG_RAW);
231                         if (!cblk->nulldec) {
232                                 if (!(cblk->nulldec = jpc_bitstream_sopen(seg->stream, "r"))) {
233                                         assert(0);
234                                 }
235                         }
236                 }
237
238
239                 for (i = 0; i < seg->numpasses; ++i) {
240                         if (cblk->numimsbs > band->numbps) {
241                                 ccp = &tile->cp->ccps[compno];
242                                 if (ccp->roishift <= 0) {
243                                         jas_eprintf("warning: corrupt code stream\n");
244                                 } else {
245                                         if (cblk->numimsbs < ccp->roishift - band->numbps) {
246                                                 jas_eprintf("warning: corrupt code stream\n");
247                                         }
248                                 }
249                         }
250                         bpno = band->roishift + band->numbps - 1 - (cblk->numimsbs +
251                           (seg->passno + i - cblk->firstpassno + 2) / 3);
252 if (bpno < 0) {
253         goto premature_exit;
254 }
255 #if 1
256                         passtype = (seg->passno + i + 2) % 3;
257 #else
258                         passtype = JPC_PASSTYPE(seg->passno + i + 2);
259 #endif
260                         assert(bpno >= 0 && bpno < 31);
261                         switch (passtype) {
262                         case JPC_SIGPASS:
263                                 ret = (seg->type == JPC_SEG_MQ) ? dec_sigpass(dec,
264                                   cblk->mqdec, bpno, band->orient,
265                                   (tile->cp->ccps[compno].cblkctx & JPC_COX_VSC) != 0,
266                                   cblk->flags, cblk->data) :
267                                   dec_rawsigpass(dec, cblk->nulldec, bpno,
268                                   (tile->cp->ccps[compno].cblkctx & JPC_COX_VSC) != 0,
269                                   cblk->flags, cblk->data);
270                                 break;
271                         case JPC_REFPASS:
272                                 ret = (seg->type == JPC_SEG_MQ) ?
273                                   dec_refpass(dec, cblk->mqdec, bpno,
274                                   (tile->cp->ccps[compno].cblkctx & JPC_COX_VSC) != 0,
275                                   cblk->flags, cblk->data) :
276                                   dec_rawrefpass(dec, cblk->nulldec, bpno,
277                                   (tile->cp->ccps[compno].cblkctx & JPC_COX_VSC) != 0,
278                                   cblk->flags, cblk->data);
279                                 break;
280                         case JPC_CLNPASS:
281                                 assert(seg->type == JPC_SEG_MQ);
282                                 ret = dec_clnpass(dec, cblk->mqdec, bpno,
283                                   band->orient, (tile->cp->ccps[compno].cblkctx &
284                                   JPC_COX_VSC) != 0, (tile->cp->ccps[compno].cblkctx &
285                                   JPC_COX_SEGSYM) != 0, cblk->flags,
286                                   cblk->data);
287                                 break;
288                         default:
289                                 ret = -1;
290                                 break;
291                         }
292                         /* Do we need to reset after each coding pass? */
293                         if (tile->cp->ccps[compno].cblkctx & JPC_COX_RESET) {
294                                 jpc_mqdec_setctxs(cblk->mqdec, JPC_NUMCTXS, jpc_mqctxs);
295                         }
296
297                         if (ret) {
298                                 jas_eprintf("coding pass failed passtype=%d segtype=%d\n", passtype, seg->type);
299                                 return -1;
300                         }
301
302                 }
303
304                 if (seg->type == JPC_SEG_MQ) {
305 /* Note: dont destroy mq decoder because context info will be lost */
306                 } else {
307                         assert(seg->type == JPC_SEG_RAW);
308                         if (tile->cp->ccps[compno].cblkctx & JPC_COX_PTERM) {
309                                 fillmask = 0x7f;
310                                 filldata = 0x2a;
311                         } else {
312                                 fillmask = 0;
313                                 filldata = 0;
314                         }
315                         if ((ret = jpc_bitstream_inalign(cblk->nulldec, fillmask,
316                           filldata)) < 0) {
317                                 return -1;
318                         } else if (ret > 0) {
319                                 jas_eprintf("warning: bad termination pattern detected\n");
320                         }
321                         jpc_bitstream_close(cblk->nulldec);
322                         cblk->nulldec = 0;
323                 }
324
325                 cblk->curseg = seg->next;
326                 jpc_seglist_remove(&cblk->segs, seg);
327                 jpc_seg_destroy(seg);
328                 seg = cblk->curseg;
329         }
330
331         assert(dopartial ? (!cblk->curseg) : 1);
332
333 premature_exit:
334         return 0;
335 }
336
337 /******************************************************************************\
338 * Code for significance pass.
339 \******************************************************************************/
340
341 #define jpc_sigpass_step(fp, frowstep, dp, bitpos, oneplushalf, orient, mqdec, vcausalflag) \
342 { \
343         int f; \
344         int v; \
345         f = *(fp); \
346         if ((f & JPC_OTHSIGMSK) && !(f & (JPC_SIG | JPC_VISIT))) { \
347                 jpc_mqdec_setcurctx((mqdec), JPC_GETZCCTXNO(f, (orient))); \
348                 JPC_T1D_GETBIT((mqdec), v, "SIG", "ZC"); \
349                 if (v) { \
350                         jpc_mqdec_setcurctx((mqdec), JPC_GETSCCTXNO(f)); \
351                         JPC_T1D_GETBIT((mqdec), v, "SIG", "SC"); \
352                         v ^= JPC_GETSPB(f); \
353                         JPC_UPDATEFLAGS4((fp), (frowstep), v, (vcausalflag)); \
354                         *(fp) |= JPC_SIG; \
355                         *(dp) = (v) ? (-(oneplushalf)) : (oneplushalf); \
356                 } \
357                 *(fp) |= JPC_VISIT; \
358         } \
359 }
360
361 static int dec_sigpass(jpc_dec_t *dec, register jpc_mqdec_t *mqdec, int bitpos, int orient,
362   int vcausalflag, jas_matrix_t *flags, jas_matrix_t *data)
363 {
364         int i;
365         int j;
366         int one;
367         int half;
368         int oneplushalf;
369         int vscanlen;
370         int width;
371         int height;
372         jpc_fix_t *fp;
373         int frowstep;
374         int fstripestep;
375         jpc_fix_t *fstripestart;
376         jpc_fix_t *fvscanstart;
377         jpc_fix_t *dp;
378         int drowstep;
379         int dstripestep;
380         jpc_fix_t *dstripestart;
381         jpc_fix_t *dvscanstart;
382         int k;
383
384         /* Avoid compiler warning about unused parameters. */
385         dec = 0;
386
387         width = jas_matrix_numcols(data);
388         height = jas_matrix_numrows(data);
389         frowstep = jas_matrix_rowstep(flags);
390         drowstep = jas_matrix_rowstep(data);
391         fstripestep = frowstep << 2;
392         dstripestep = drowstep << 2;
393
394         one = 1 << bitpos;
395         half = one >> 1;
396         oneplushalf = one | half;
397
398         fstripestart = jas_matrix_getref(flags, 1, 1);
399         dstripestart = jas_matrix_getref(data, 0, 0);
400         for (i = height; i > 0; i -= 4, fstripestart += fstripestep,
401           dstripestart += dstripestep) {
402                 fvscanstart = fstripestart;
403                 dvscanstart = dstripestart;
404                 vscanlen = JAS_MIN(i, 4);
405                 for (j = width; j > 0; --j, ++fvscanstart, ++dvscanstart) {
406                         fp = fvscanstart;
407                         dp = dvscanstart;
408                         k = vscanlen;
409
410                         /* Process first sample in vertical scan. */
411                         jpc_sigpass_step(fp, frowstep, dp, bitpos, oneplushalf,
412                           orient, mqdec, vcausalflag);
413                         if (--k <= 0) {
414                                 continue;
415                         }
416                         fp += frowstep;
417                         dp += drowstep;
418
419                         /* Process second sample in vertical scan. */
420                         jpc_sigpass_step(fp, frowstep, dp, bitpos, oneplushalf,
421                           orient, mqdec, 0);
422                         if (--k <= 0) {
423                                 continue;
424                         }
425                         fp += frowstep;
426                         dp += drowstep;
427
428                         /* Process third sample in vertical scan. */
429                         jpc_sigpass_step(fp, frowstep, dp, bitpos, oneplushalf,
430                           orient, mqdec, 0);
431                         if (--k <= 0) {
432                                 continue;
433                         }
434                         fp += frowstep;
435                         dp += drowstep;
436
437                         /* Process fourth sample in vertical scan. */
438                         jpc_sigpass_step(fp, frowstep, dp, bitpos, oneplushalf,
439                           orient, mqdec, 0);
440                 }
441         }
442         return 0;
443 }
444
445 #define jpc_rawsigpass_step(fp, frowstep, dp, oneplushalf, in, vcausalflag) \
446 { \
447         jpc_fix_t f = *(fp); \
448         jpc_fix_t v; \
449         if ((f & JPC_OTHSIGMSK) && !(f & (JPC_SIG | JPC_VISIT))) { \
450                 JPC_T1D_RAWGETBIT(in, v, "SIG", "ZC"); \
451                 if (v < 0) { \
452                         return -1; \
453                 } \
454                 if (v) { \
455                         JPC_T1D_RAWGETBIT(in, v, "SIG", "SC"); \
456                         if (v < 0) { \
457                                 return -1; \
458                         } \
459                         JPC_UPDATEFLAGS4((fp), (frowstep), v, (vcausalflag)); \
460                         *(fp) |= JPC_SIG; \
461                         *(dp) = v ? (-oneplushalf) : (oneplushalf); \
462                 } \
463                 *(fp) |= JPC_VISIT; \
464         } \
465 }
466
467 static int dec_rawsigpass(jpc_dec_t *dec, jpc_bitstream_t *in, int bitpos, int vcausalflag,
468   jas_matrix_t *flags, jas_matrix_t *data)
469 {
470         int i;
471         int j;
472         int k;
473         int one;
474         int half;
475         int oneplushalf;
476         int vscanlen;
477         int width;
478         int height;
479         jpc_fix_t *fp;
480         int frowstep;
481         int fstripestep;
482         jpc_fix_t *fstripestart;
483         jpc_fix_t *fvscanstart;
484         jpc_fix_t *dp;
485         int drowstep;
486         int dstripestep;
487         jpc_fix_t *dstripestart;
488         jpc_fix_t *dvscanstart;
489
490         /* Avoid compiler warning about unused parameters. */
491         dec = 0;
492
493         width = jas_matrix_numcols(data);
494         height = jas_matrix_numrows(data);
495         frowstep = jas_matrix_rowstep(flags);
496         drowstep = jas_matrix_rowstep(data);
497         fstripestep = frowstep << 2;
498         dstripestep = drowstep << 2;
499
500         one = 1 << bitpos;
501         half = one >> 1;
502         oneplushalf = one | half;
503
504         fstripestart = jas_matrix_getref(flags, 1, 1);
505         dstripestart = jas_matrix_getref(data, 0, 0);
506         for (i = height; i > 0; i -= 4, fstripestart += fstripestep,
507           dstripestart += dstripestep) {
508                 fvscanstart = fstripestart;
509                 dvscanstart = dstripestart;
510                 vscanlen = JAS_MIN(i, 4);
511                 for (j = width; j > 0; --j, ++fvscanstart, ++dvscanstart) {
512                         fp = fvscanstart;
513                         dp = dvscanstart;
514                         k = vscanlen;
515
516                         /* Process first sample in vertical scan. */
517                         jpc_rawsigpass_step(fp, frowstep, dp, oneplushalf,
518                           in, vcausalflag);
519                         if (--k <= 0) {
520                                 continue;
521                         }
522                         fp += frowstep;
523                         dp += drowstep;
524
525                         /* Process second sample in vertical scan. */
526                         jpc_rawsigpass_step(fp, frowstep, dp, oneplushalf,
527                           in, 0);
528                         if (--k <= 0) {
529                                 continue;
530                         }
531                         fp += frowstep;
532                         dp += drowstep;
533
534                         /* Process third sample in vertical scan. */
535                         jpc_rawsigpass_step(fp, frowstep, dp, oneplushalf,
536                           in, 0);
537                         if (--k <= 0) {
538                                 continue;
539                         }
540                         fp += frowstep;
541                         dp += drowstep;
542
543                         /* Process fourth sample in vertical scan. */
544                         jpc_rawsigpass_step(fp, frowstep, dp, oneplushalf,
545                           in, 0);
546
547                 }
548         }
549         return 0;
550 }
551
552 /******************************************************************************\
553 * Code for refinement pass.
554 \******************************************************************************/
555
556 #define jpc_refpass_step(fp, dp, poshalf, neghalf, mqdec, vcausalflag) \
557 { \
558         int v; \
559         int t; \
560         if (((*(fp)) & (JPC_SIG | JPC_VISIT)) == JPC_SIG) { \
561                 jpc_mqdec_setcurctx((mqdec), JPC_GETMAGCTXNO(*(fp))); \
562                 JPC_T1D_GETBITNOSKEW((mqdec), v, "REF", "MR"); \
563                 t = (v ? (poshalf) : (neghalf)); \
564                 *(dp) += (*(dp) < 0) ? (-t) : t; \
565                 *(fp) |= JPC_REFINE; \
566         } \
567 }
568
569 static int dec_refpass(jpc_dec_t *dec, register jpc_mqdec_t *mqdec, int bitpos,
570   int vcausalflag, jas_matrix_t *flags, jas_matrix_t *data)
571 {
572         int i;
573         int j;
574         int vscanlen;
575         int width;
576         int height;
577         int one;
578         int poshalf;
579         int neghalf;
580         jpc_fix_t *fp;
581         int frowstep;
582         int fstripestep;
583         jpc_fix_t *fstripestart;
584         jpc_fix_t *fvscanstart;
585         jpc_fix_t *dp;
586         int drowstep;
587         int dstripestep;
588         jpc_fix_t *dstripestart;
589         jpc_fix_t *dvscanstart;
590         int k;
591
592         /* Avoid compiler warning about unused parameters. */
593         dec = 0;
594         vcausalflag = 0;
595
596         width = jas_matrix_numcols(data);
597         height = jas_matrix_numrows(data);
598         frowstep = jas_matrix_rowstep(flags);
599         drowstep = jas_matrix_rowstep(data);
600         fstripestep = frowstep << 2;
601         dstripestep = drowstep << 2;
602
603         one = 1 << bitpos;
604         poshalf = one >> 1;
605         neghalf = (bitpos > 0) ? (-poshalf) : (-1);
606
607         fstripestart = jas_matrix_getref(flags, 1, 1);
608         dstripestart = jas_matrix_getref(data, 0, 0);
609         for (i = height; i > 0; i -= 4, fstripestart += fstripestep,
610           dstripestart += dstripestep) {
611                 fvscanstart = fstripestart;
612                 dvscanstart = dstripestart;
613                 vscanlen = JAS_MIN(i, 4);
614                 for (j = width; j > 0; --j, ++fvscanstart, ++dvscanstart) {
615                         fp = fvscanstart;
616                         dp = dvscanstart;
617                         k = vscanlen;
618
619                         /* Process first sample in vertical scan. */
620                         jpc_refpass_step(fp, dp, poshalf, neghalf, mqdec,
621                           vcausalflag);
622                         if (--k <= 0) {
623                                 continue;
624                         }
625                         fp += frowstep;
626                         dp += drowstep;
627
628                         /* Process second sample in vertical scan. */
629                         jpc_refpass_step(fp, dp, poshalf, neghalf, mqdec, 0);
630                         if (--k <= 0) {
631                                 continue;
632                         }
633                         fp += frowstep;
634                         dp += drowstep;
635
636                         /* Process third sample in vertical scan. */
637                         jpc_refpass_step(fp, dp, poshalf, neghalf, mqdec, 0);
638                         if (--k <= 0) {
639                                 continue;
640                         }
641                         fp += frowstep;
642                         dp += drowstep;
643
644                         /* Process fourth sample in vertical scan. */
645                         jpc_refpass_step(fp, dp, poshalf, neghalf, mqdec, 0);
646                 }
647         }
648
649         return 0;
650 }
651
652 #define jpc_rawrefpass_step(fp, dp, poshalf, neghalf, in, vcausalflag) \
653 { \
654         jpc_fix_t v; \
655         jpc_fix_t t; \
656         if (((*(fp)) & (JPC_SIG | JPC_VISIT)) == JPC_SIG) { \
657                 JPC_T1D_RAWGETBIT(in, v, "REF", "MAGREF"); \
658                 if (v < 0) { \
659                         return -1; \
660                 } \
661                 t = (v ? poshalf : neghalf); \
662                 *(dp) += (*(dp) < 0) ? (-t) : t; \
663                 *(fp) |= JPC_REFINE; \
664         } \
665 }
666
667 static int dec_rawrefpass(jpc_dec_t *dec, jpc_bitstream_t *in, int bitpos, int vcausalflag,
668   jas_matrix_t *flags, jas_matrix_t *data)
669 {
670         int i;
671         int j;
672         int k;
673         int vscanlen;
674         int width;
675         int height;
676         int one;
677         int poshalf;
678         int neghalf;
679         jpc_fix_t *fp;
680         int frowstep;
681         int fstripestep;
682         jpc_fix_t *fstripestart;
683         jpc_fix_t *fvscanstart;
684         jpc_fix_t *dp;
685         int drowstep;
686         int dstripestep;
687         jpc_fix_t *dstripestart;
688         jpc_fix_t *dvscanstart;
689
690         /* Avoid compiler warning about unused parameters. */
691         dec = 0;
692         vcausalflag = 0;
693
694         width = jas_matrix_numcols(data);
695         height = jas_matrix_numrows(data);
696         frowstep = jas_matrix_rowstep(flags);
697         drowstep = jas_matrix_rowstep(data);
698         fstripestep = frowstep << 2;
699         dstripestep = drowstep << 2;
700
701         one = 1 << bitpos;
702         poshalf = one >> 1;
703         neghalf = (bitpos > 0) ? (-poshalf) : (-1);
704
705         fstripestart = jas_matrix_getref(flags, 1, 1);
706         dstripestart = jas_matrix_getref(data, 0, 0);
707         for (i = height; i > 0; i -= 4, fstripestart += fstripestep,
708           dstripestart += dstripestep) {
709                 fvscanstart = fstripestart;
710                 dvscanstart = dstripestart;
711                 vscanlen = JAS_MIN(i, 4);
712                 for (j = width; j > 0; --j, ++fvscanstart, ++dvscanstart) {
713                         fp = fvscanstart;
714                         dp = dvscanstart;
715                         k = vscanlen;
716
717                         /* Process first sample in vertical scan. */
718                         jpc_rawrefpass_step(fp, dp, poshalf, neghalf, in,
719                           vcausalflag);
720                         if (--k <= 0) {
721                                 continue;
722                         }
723                         fp += frowstep;
724                         dp += drowstep;
725
726                         /* Process second sample in vertical scan. */
727                         jpc_rawrefpass_step(fp, dp, poshalf, neghalf, in, 0);
728                         if (--k <= 0) {
729                                 continue;
730                         }
731                         fp += frowstep;
732                         dp += drowstep;
733
734                         /* Process third sample in vertical scan. */
735                         jpc_rawrefpass_step(fp, dp, poshalf, neghalf, in, 0);
736                         if (--k <= 0) {
737                                 continue;
738                         }
739                         fp += frowstep;
740                         dp += drowstep;
741
742                         /* Process fourth sample in vertical scan. */
743                         jpc_rawrefpass_step(fp, dp, poshalf, neghalf, in, 0);
744                 }
745         }
746         return 0;
747 }
748
749 /******************************************************************************\
750 * Code for cleanup pass.
751 \******************************************************************************/
752
753 #define jpc_clnpass_step(f, fp, frowstep, dp, oneplushalf, orient, mqdec, flabel, plabel, vcausalflag) \
754 { \
755         int v; \
756 flabel \
757         if (!((f) & (JPC_SIG | JPC_VISIT))) { \
758                 jpc_mqdec_setcurctx((mqdec), JPC_GETZCCTXNO((f), (orient))); \
759                 JPC_T1D_GETBIT((mqdec), v, "CLN", "ZC"); \
760                 if (v) { \
761 plabel \
762                         /* Coefficient is significant. */ \
763                         jpc_mqdec_setcurctx((mqdec), JPC_GETSCCTXNO(f)); \
764                         JPC_T1D_GETBIT((mqdec), v, "CLN", "SC"); \
765                         v ^= JPC_GETSPB(f); \
766                         *(dp) = (v) ? (-(oneplushalf)) : (oneplushalf); \
767                         JPC_UPDATEFLAGS4((fp), (frowstep), v, (vcausalflag)); \
768                         *(fp) |= JPC_SIG; \
769                 } \
770         } \
771         /* XXX - Is this correct?  Can aggregation cause some VISIT bits not to be reset?  Check. */ \
772         *(fp) &= ~JPC_VISIT; \
773 }
774
775 static int dec_clnpass(jpc_dec_t *dec, register jpc_mqdec_t *mqdec, int bitpos, int orient,
776   int vcausalflag, int segsymflag, jas_matrix_t *flags, jas_matrix_t *data)
777 {
778         int i;
779         int j;
780         int k;
781         int vscanlen;
782         int v;
783         int half;
784         int runlen;
785         int f;
786         int width;
787         int height;
788         int one;
789         int oneplushalf;
790
791         jpc_fix_t *fp;
792         int frowstep;
793         int fstripestep;
794         jpc_fix_t *fstripestart;
795         jpc_fix_t *fvscanstart;
796
797         jpc_fix_t *dp;
798         int drowstep;
799         int dstripestep;
800         jpc_fix_t *dstripestart;
801         jpc_fix_t *dvscanstart;
802
803         /* Avoid compiler warning about unused parameters. */
804         dec = 0;
805
806         one = 1 << bitpos;
807         half = one >> 1;
808         oneplushalf = one | half;
809
810         width = jas_matrix_numcols(data);
811         height = jas_matrix_numrows(data);
812
813         frowstep = jas_matrix_rowstep(flags);
814         drowstep = jas_matrix_rowstep(data);
815         fstripestep = frowstep << 2;
816         dstripestep = drowstep << 2;
817
818         fstripestart = jas_matrix_getref(flags, 1, 1);
819         dstripestart = jas_matrix_getref(data, 0, 0);
820         for (i = 0; i < height; i += 4, fstripestart += fstripestep,
821           dstripestart += dstripestep) {
822                 fvscanstart = fstripestart;
823                 dvscanstart = dstripestart;
824                 vscanlen = JAS_MIN(4, height - i);
825                 for (j = width; j > 0; --j, ++fvscanstart, ++dvscanstart) {
826                         fp = fvscanstart;
827                         if (vscanlen >= 4 && (!((*fp) & (JPC_SIG | JPC_VISIT |
828                           JPC_OTHSIGMSK))) && (fp += frowstep, !((*fp) & (JPC_SIG |
829                           JPC_VISIT | JPC_OTHSIGMSK))) && (fp += frowstep, !((*fp) &
830                           (JPC_SIG | JPC_VISIT | JPC_OTHSIGMSK))) && (fp += frowstep,
831                           !((*fp) & (JPC_SIG | JPC_VISIT | JPC_OTHSIGMSK)))) {
832
833                                 jpc_mqdec_setcurctx(mqdec, JPC_AGGCTXNO);
834                                 JPC_T1D_GETBIT(mqdec, v, "CLN", "AGG");
835                                 if (!v) {
836                                         continue;
837                                 }
838                                 jpc_mqdec_setcurctx(mqdec, JPC_UCTXNO);
839                                 JPC_T1D_GETBITNOSKEW(mqdec, v, "CLN", "RL");
840                                 runlen = v;
841                                 JPC_T1D_GETBITNOSKEW(mqdec, v, "CLN", "RL");
842                                 runlen = (runlen << 1) | v;
843                                 f = *(fp = fvscanstart + frowstep * runlen);
844                                 dp = dvscanstart + drowstep * runlen;
845                                 k = vscanlen - runlen;
846                                 switch (runlen) {
847                                 case 0:
848                                         goto clnpass_partial0;
849                                         break;
850                                 case 1:
851                                         goto clnpass_partial1;
852                                         break;
853                                 case 2:
854                                         goto clnpass_partial2;
855                                         break;
856                                 case 3:
857                                         goto clnpass_partial3;
858                                         break;
859                                 }
860                         } else {
861                                 f = *(fp = fvscanstart);
862                                 dp = dvscanstart;
863                                 k = vscanlen;
864                                 goto clnpass_full0;
865                         }
866
867                         /* Process first sample in vertical scan. */
868                         jpc_clnpass_step(f, fp, frowstep, dp, oneplushalf, orient,
869                           mqdec, clnpass_full0:, clnpass_partial0:,
870                           vcausalflag);
871                         if (--k <= 0) {
872                                 continue;
873                         }
874                         fp += frowstep;
875                         dp += drowstep;
876
877                         /* Process second sample in vertical scan. */
878                         f = *fp;
879                         jpc_clnpass_step(f, fp, frowstep, dp, oneplushalf, orient,
880                                 mqdec, ;, clnpass_partial1:, 0);
881                         if (--k <= 0) {
882                                 continue;
883                         }
884                         fp += frowstep;
885                         dp += drowstep;
886
887                         /* Process third sample in vertical scan. */
888                         f = *fp;
889                         jpc_clnpass_step(f, fp, frowstep, dp, oneplushalf, orient,
890                                 mqdec, ;, clnpass_partial2:, 0);
891                         if (--k <= 0) {
892                                 continue;
893                         }
894                         fp += frowstep;
895                         dp += drowstep;
896
897                         /* Process fourth sample in vertical scan. */
898                         f = *fp;
899                         jpc_clnpass_step(f, fp, frowstep, dp, oneplushalf, orient,
900                                 mqdec, ;, clnpass_partial3:, 0);
901                 }
902         }
903
904         if (segsymflag) {
905                 int segsymval;
906                 segsymval = 0;
907                 jpc_mqdec_setcurctx(mqdec, JPC_UCTXNO);
908                 JPC_T1D_GETBITNOSKEW(mqdec, v, "CLN", "SEGSYM");
909                 segsymval = (segsymval << 1) | (v & 1);
910                 JPC_T1D_GETBITNOSKEW(mqdec, v, "CLN", "SEGSYM");
911                 segsymval = (segsymval << 1) | (v & 1);
912                 JPC_T1D_GETBITNOSKEW(mqdec, v, "CLN", "SEGSYM");
913                 segsymval = (segsymval << 1) | (v & 1);
914                 JPC_T1D_GETBITNOSKEW(mqdec, v, "CLN", "SEGSYM");
915                 segsymval = (segsymval << 1) | (v & 1);
916                 if (segsymval != 0xa) {
917                         jas_eprintf("warning: bad segmentation symbol\n");
918                 }
919         }
920
921         return 0;
922 }