All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.sun.pdfview.decode.CCITTFaxDecoder Maven / Gradle / Ivy

The newest version!
/*
 * Based on the SUN code (see license beyond) changes are made to handle CCITTFax encoded
 * data in a PDF image. This may or may not apply to real world CCITT documents.
 *
 * Copyright (c) 2007, intarsys consulting GmbH
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 *
 * - Redistributions of source code must retain the above copyright notice,
 *   this list of conditions and the following disclaimer.
 *
 * - Redistributions in binary form must reproduce the above copyright notice,
 *   this list of conditions and the following disclaimer in the documentation
 *   and/or other materials provided with the distribution.
 *
 * - Neither the name of intarsys nor the names of its contributors may be used
 *   to endorse or promote products derived from this software without specific
 *   prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 * POSSIBILITY OF SUCH DAMAGE.
 */
/*
 * Copyright (c) 2001 Sun Microsystems, Inc. All Rights Reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 *
 * -Redistributions of source code must retain the above copyright notice, this
 * list of conditions and the following disclaimer.
 *
 * -Redistribution in binary form must reproduct the above copyright notice,
 * this list of conditions and the following disclaimer in the documentation
 * and/or other materials provided with the distribution.
 *
 * Neither the name of Sun Microsystems, Inc. or the names of contributors may
 * be used to endorse or promote products derived from this software without
 * specific prior written permission.
 *
 * This software is provided "AS IS," without a warranty of any kind. ALL
 * EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES, INCLUDING ANY
 * IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE OR
 * NON-INFRINGEMENT, ARE HEREBY EXCLUDED. SUN AND ITS LICENSORS SHALL NOT BE
 * LIABLE FOR ANY DAMAGES SUFFERED BY LICENSEE AS A RESULT OF USING, MODIFYING
 * OR DISTRIBUTING THE SOFTWARE OR ITS DERIVATIVES. IN NO EVENT WILL SUN OR ITS
 * LICENSORS BE LIABLE FOR ANY LOST REVENUE, PROFIT OR DATA, OR FOR DIRECT,
 * INDIRECT, SPECIAL, CONSEQUENTIAL, INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER
 * CAUSED AND REGARDLESS OF THE THEORY OF LIABILITY, ARISING OUT OF THE USE OF
 * OR INABILITY TO USE SOFTWARE, EVEN IF SUN HAS BEEN ADVISED OF THE
 * POSSIBILITY OF SUCH DAMAGES.
 *
 * You acknowledge that Software is not designed,licensed or intended for use in
 * the design, construction, operation or maintenance of any nuclear facility.
 */
package com.sun.pdfview.decode;

public class CCITTFaxDecoder {
	static int[] table1 = { 0x00, // 0 bits are left in first byte - SHOULD
			// NOT HAPPEN
			0x01, // 1 bits are left in first byte
			0x03, // 2 bits are left in first byte
			0x07, // 3 bits are left in first byte
			0x0f, // 4 bits are left in first byte
			0x1f, // 5 bits are left in first byte
			0x3f, // 6 bits are left in first byte
			0x7f, // 7 bits are left in first byte
			0xff // 8 bits are left in first byte
	};

	static int[] table2 = { 0x00, // 0
			0x80, // 1
			0xc0, // 2
			0xe0, // 3
			0xf0, // 4
			0xf8, // 5
			0xfc, // 6
			0xfe, // 7
			0xff // 8
	};

	// Table to be used when fillOrder = 2, for flipping bytes.
	static byte[] flipTable = { 0, -128, 64, -64, 32, -96, 96, -32, 16, -112,
			80, -48, 48, -80, 112, -16, 8, -120, 72, -56, 40, -88, 104, -24,
			24, -104, 88, -40, 56, -72, 120, -8, 4, -124, 68, -60, 36, -92,
			100, -28, 20, -108, 84, -44, 52, -76, 116, -12, 12, -116, 76, -52,
			44, -84, 108, -20, 28, -100, 92, -36, 60, -68, 124, -4, 2, -126,
			66, -62, 34, -94, 98, -30, 18, -110, 82, -46, 50, -78, 114, -14,
			10, -118, 74, -54, 42, -86, 106, -22, 26, -102, 90, -38, 58, -70,
			122, -6, 6, -122, 70, -58, 38, -90, 102, -26, 22, -106, 86, -42,
			54, -74, 118, -10, 14, -114, 78, -50, 46, -82, 110, -18, 30, -98,
			94, -34, 62, -66, 126, -2, 1, -127, 65, -63, 33, -95, 97, -31, 17,
			-111, 81, -47, 49, -79, 113, -15, 9, -119, 73, -55, 41, -87, 105,
			-23, 25, -103, 89, -39, 57, -71, 121, -7, 5, -123, 69, -59, 37,
			-91, 101, -27, 21, -107, 85, -43, 53, -75, 117, -11, 13, -115, 77,
			-51, 45, -83, 109, -19, 29, -99, 93, -35, 61, -67, 125, -3, 3,
			-125, 67, -61, 35, -93, 99, -29, 19, -109, 83, -45, 51, -77, 115,
			-13, 11, -117, 75, -53, 43, -85, 107, -21, 27, -101, 91, -37, 59,
			-69, 123, -5, 7, -121, 71, -57, 39, -89, 103, -25, 23, -105, 87,
			-41, 55, -73, 119, -9, 15, -113, 79, -49, 47, -81, 111, -17, 31,
			-97, 95, -33, 63, -65, 127, -1, };

	// The main 10 bit white runs lookup table
	static short white[] = {
	// 0 - 7
			6430, 6400, 6400, 6400, 3225, 3225, 3225, 3225,
			// 8 - 15
			944, 944, 944, 944, 976, 976, 976, 976,
			// 16 - 23
			1456, 1456, 1456, 1456, 1488, 1488, 1488, 1488,
			// 24 - 31
			718, 718, 718, 718, 718, 718, 718, 718,
			// 32 - 39
			750, 750, 750, 750, 750, 750, 750, 750,
			// 40 - 47
			1520, 1520, 1520, 1520, 1552, 1552, 1552, 1552,
			// 48 - 55
			428, 428, 428, 428, 428, 428, 428, 428,
			// 56 - 63
			428, 428, 428, 428, 428, 428, 428, 428,
			// 64 - 71
			654, 654, 654, 654, 654, 654, 654, 654,
			// 72 - 79
			1072, 1072, 1072, 1072, 1104, 1104, 1104, 1104,
			// 80 - 87
			1136, 1136, 1136, 1136, 1168, 1168, 1168, 1168,
			// 88 - 95
			1200, 1200, 1200, 1200, 1232, 1232, 1232, 1232,
			// 96 - 103
			622, 622, 622, 622, 622, 622, 622, 622,
			// 104 - 111
			1008, 1008, 1008, 1008, 1040, 1040, 1040, 1040,
			// 112 - 119
			44, 44, 44, 44, 44, 44, 44, 44,
			// 120 - 127
			44, 44, 44, 44, 44, 44, 44, 44,
			// 128 - 135
			396, 396, 396, 396, 396, 396, 396, 396,
			// 136 - 143
			396, 396, 396, 396, 396, 396, 396, 396,
			// 144 - 151
			1712, 1712, 1712, 1712, 1744, 1744, 1744, 1744,
			// 152 - 159
			846, 846, 846, 846, 846, 846, 846, 846,
			// 160 - 167
			1264, 1264, 1264, 1264, 1296, 1296, 1296, 1296,
			// 168 - 175
			1328, 1328, 1328, 1328, 1360, 1360, 1360, 1360,
			// 176 - 183
			1392, 1392, 1392, 1392, 1424, 1424, 1424, 1424,
			// 184 - 191
			686, 686, 686, 686, 686, 686, 686, 686,
			// 192 - 199
			910, 910, 910, 910, 910, 910, 910, 910,
			// 200 - 207
			1968, 1968, 1968, 1968, 2000, 2000, 2000, 2000,
			// 208 - 215
			2032, 2032, 2032, 2032, 16, 16, 16, 16,
			// 216 - 223
			10257, 10257, 10257, 10257, 12305, 12305, 12305, 12305,
			// 224 - 231
			330, 330, 330, 330, 330, 330, 330, 330,
			// 232 - 239
			330, 330, 330, 330, 330, 330, 330, 330,
			// 240 - 247
			330, 330, 330, 330, 330, 330, 330, 330,
			// 248 - 255
			330, 330, 330, 330, 330, 330, 330, 330,
			// 256 - 263
			362, 362, 362, 362, 362, 362, 362, 362,
			// 264 - 271
			362, 362, 362, 362, 362, 362, 362, 362,
			// 272 - 279
			362, 362, 362, 362, 362, 362, 362, 362,
			// 280 - 287
			362, 362, 362, 362, 362, 362, 362, 362,
			// 288 - 295
			878, 878, 878, 878, 878, 878, 878, 878,
			// 296 - 303
			1904, 1904, 1904, 1904, 1936, 1936, 1936, 1936,
			// 304 - 311
			-18413, -18413, -16365, -16365, -14317, -14317, -10221, -10221,
			// 312 - 319
			590, 590, 590, 590, 590, 590, 590, 590,
			// 320 - 327
			782, 782, 782, 782, 782, 782, 782, 782,
			// 328 - 335
			1584, 1584, 1584, 1584, 1616, 1616, 1616, 1616,
			// 336 - 343
			1648, 1648, 1648, 1648, 1680, 1680, 1680, 1680,
			// 344 - 351
			814, 814, 814, 814, 814, 814, 814, 814,
			// 352 - 359
			1776, 1776, 1776, 1776, 1808, 1808, 1808, 1808,
			// 360 - 367
			1840, 1840, 1840, 1840, 1872, 1872, 1872, 1872,
			// 368 - 375
			6157, 6157, 6157, 6157, 6157, 6157, 6157, 6157,
			// 376 - 383
			6157, 6157, 6157, 6157, 6157, 6157, 6157, 6157,
			// 384 - 391
			-12275, -12275, -12275, -12275, -12275, -12275, -12275, -12275,
			// 392 - 399
			-12275, -12275, -12275, -12275, -12275, -12275, -12275, -12275,
			// 400 - 407
			14353, 14353, 14353, 14353, 16401, 16401, 16401, 16401,
			// 408 - 415
			22547, 22547, 24595, 24595, 20497, 20497, 20497, 20497,
			// 416 - 423
			18449, 18449, 18449, 18449, 26643, 26643, 28691, 28691,
			// 424 - 431
			30739, 30739, -32749, -32749, -30701, -30701, -28653, -28653,
			// 432 - 439
			-26605, -26605, -24557, -24557, -22509, -22509, -20461, -20461,
			// 440 - 447
			8207, 8207, 8207, 8207, 8207, 8207, 8207, 8207,
			// 448 - 455
			72, 72, 72, 72, 72, 72, 72, 72,
			// 456 - 463
			72, 72, 72, 72, 72, 72, 72, 72,
			// 464 - 471
			72, 72, 72, 72, 72, 72, 72, 72,
			// 472 - 479
			72, 72, 72, 72, 72, 72, 72, 72,
			// 480 - 487
			72, 72, 72, 72, 72, 72, 72, 72,
			// 488 - 495
			72, 72, 72, 72, 72, 72, 72, 72,
			// 496 - 503
			72, 72, 72, 72, 72, 72, 72, 72,
			// 504 - 511
			72, 72, 72, 72, 72, 72, 72, 72,
			// 512 - 519
			104, 104, 104, 104, 104, 104, 104, 104,
			// 520 - 527
			104, 104, 104, 104, 104, 104, 104, 104,
			// 528 - 535
			104, 104, 104, 104, 104, 104, 104, 104,
			// 536 - 543
			104, 104, 104, 104, 104, 104, 104, 104,
			// 544 - 551
			104, 104, 104, 104, 104, 104, 104, 104,
			// 552 - 559
			104, 104, 104, 104, 104, 104, 104, 104,
			// 560 - 567
			104, 104, 104, 104, 104, 104, 104, 104,
			// 568 - 575
			104, 104, 104, 104, 104, 104, 104, 104,
			// 576 - 583
			4107, 4107, 4107, 4107, 4107, 4107, 4107, 4107,
			// 584 - 591
			4107, 4107, 4107, 4107, 4107, 4107, 4107, 4107,
			// 592 - 599
			4107, 4107, 4107, 4107, 4107, 4107, 4107, 4107,
			// 600 - 607
			4107, 4107, 4107, 4107, 4107, 4107, 4107, 4107,
			// 608 - 615
			266, 266, 266, 266, 266, 266, 266, 266,
			// 616 - 623
			266, 266, 266, 266, 266, 266, 266, 266,
			// 624 - 631
			266, 266, 266, 266, 266, 266, 266, 266,
			// 632 - 639
			266, 266, 266, 266, 266, 266, 266, 266,
			// 640 - 647
			298, 298, 298, 298, 298, 298, 298, 298,
			// 648 - 655
			298, 298, 298, 298, 298, 298, 298, 298,
			// 656 - 663
			298, 298, 298, 298, 298, 298, 298, 298,
			// 664 - 671
			298, 298, 298, 298, 298, 298, 298, 298,
			// 672 - 679
			524, 524, 524, 524, 524, 524, 524, 524,
			// 680 - 687
			524, 524, 524, 524, 524, 524, 524, 524,
			// 688 - 695
			556, 556, 556, 556, 556, 556, 556, 556,
			// 696 - 703
			556, 556, 556, 556, 556, 556, 556, 556,
			// 704 - 711
			136, 136, 136, 136, 136, 136, 136, 136,
			// 712 - 719
			136, 136, 136, 136, 136, 136, 136, 136,
			// 720 - 727
			136, 136, 136, 136, 136, 136, 136, 136,
			// 728 - 735
			136, 136, 136, 136, 136, 136, 136, 136,
			// 736 - 743
			136, 136, 136, 136, 136, 136, 136, 136,
			// 744 - 751
			136, 136, 136, 136, 136, 136, 136, 136,
			// 752 - 759
			136, 136, 136, 136, 136, 136, 136, 136,
			// 760 - 767
			136, 136, 136, 136, 136, 136, 136, 136,
			// 768 - 775
			168, 168, 168, 168, 168, 168, 168, 168,
			// 776 - 783
			168, 168, 168, 168, 168, 168, 168, 168,
			// 784 - 791
			168, 168, 168, 168, 168, 168, 168, 168,
			// 792 - 799
			168, 168, 168, 168, 168, 168, 168, 168,
			// 800 - 807
			168, 168, 168, 168, 168, 168, 168, 168,
			// 808 - 815
			168, 168, 168, 168, 168, 168, 168, 168,
			// 816 - 823
			168, 168, 168, 168, 168, 168, 168, 168,
			// 824 - 831
			168, 168, 168, 168, 168, 168, 168, 168,
			// 832 - 839
			460, 460, 460, 460, 460, 460, 460, 460,
			// 840 - 847
			460, 460, 460, 460, 460, 460, 460, 460,
			// 848 - 855
			492, 492, 492, 492, 492, 492, 492, 492,
			// 856 - 863
			492, 492, 492, 492, 492, 492, 492, 492,
			// 864 - 871
			2059, 2059, 2059, 2059, 2059, 2059, 2059, 2059,
			// 872 - 879
			2059, 2059, 2059, 2059, 2059, 2059, 2059, 2059,
			// 880 - 887
			2059, 2059, 2059, 2059, 2059, 2059, 2059, 2059,
			// 888 - 895
			2059, 2059, 2059, 2059, 2059, 2059, 2059, 2059,
			// 896 - 903
			200, 200, 200, 200, 200, 200, 200, 200,
			// 904 - 911
			200, 200, 200, 200, 200, 200, 200, 200,
			// 912 - 919
			200, 200, 200, 200, 200, 200, 200, 200,
			// 920 - 927
			200, 200, 200, 200, 200, 200, 200, 200,
			// 928 - 935
			200, 200, 200, 200, 200, 200, 200, 200,
			// 936 - 943
			200, 200, 200, 200, 200, 200, 200, 200,
			// 944 - 951
			200, 200, 200, 200, 200, 200, 200, 200,
			// 952 - 959
			200, 200, 200, 200, 200, 200, 200, 200,
			// 960 - 967
			232, 232, 232, 232, 232, 232, 232, 232,
			// 968 - 975
			232, 232, 232, 232, 232, 232, 232, 232,
			// 976 - 983
			232, 232, 232, 232, 232, 232, 232, 232,
			// 984 - 991
			232, 232, 232, 232, 232, 232, 232, 232,
			// 992 - 999
			232, 232, 232, 232, 232, 232, 232, 232,
			// 1000 - 1007
			232, 232, 232, 232, 232, 232, 232, 232,
			// 1008 - 1015
			232, 232, 232, 232, 232, 232, 232, 232,
			// 1016 - 1023
			232, 232, 232, 232, 232, 232, 232, 232, };

	// Additional make up codes for both White and Black runs
	static short[] additionalMakeup = { 28679, 28679, 31752, (short) 32777,
			(short) 33801, (short) 34825, (short) 35849, (short) 36873,
			(short) 29703, (short) 29703, (short) 30727, (short) 30727,
			(short) 37897, (short) 38921, (short) 39945, (short) 40969 };

	// Initial black run look up table, uses the first 4 bits of a code
	static short[] initBlack = {
	// 0 - 7
			3226, 6412, 200, 168, 38, 38, 134, 134, // 8 - 15
			100, 100, 100, 100, 68, 68, 68, 68 };

	//
	static short[] twoBitBlack = { 292, 260, 226, 226 }; // 0 - 3

	// Main black run table, using the last 9 bits of possible 13 bit code
	static short black[] = {
	// 0 - 7
			62, 62, 30, 30, 0, 0, 0, 0,
			// 8 - 15
			0, 0, 0, 0, 0, 0, 0, 0,
			// 16 - 23
			0, 0, 0, 0, 0, 0, 0, 0,
			// 24 - 31
			0, 0, 0, 0, 0, 0, 0, 0,
			// 32 - 39
			3225, 3225, 3225, 3225, 3225, 3225, 3225, 3225,
			// 40 - 47
			3225, 3225, 3225, 3225, 3225, 3225, 3225, 3225,
			// 48 - 55
			3225, 3225, 3225, 3225, 3225, 3225, 3225, 3225,
			// 56 - 63
			3225, 3225, 3225, 3225, 3225, 3225, 3225, 3225,
			// 64 - 71
			588, 588, 588, 588, 588, 588, 588, 588,
			// 72 - 79
			1680, 1680, 20499, 22547, 24595, 26643, 1776, 1776,
			// 80 - 87
			1808, 1808, -24557, -22509, -20461, -18413, 1904, 1904,
			// 88 - 95
			1936, 1936, -16365, -14317, 782, 782, 782, 782,
			// 96 - 103
			814, 814, 814, 814, -12269, -10221, 10257, 10257,
			// 104 - 111
			12305, 12305, 14353, 14353, 16403, 18451, 1712, 1712,
			// 112 - 119
			1744, 1744, 28691, 30739, -32749, -30701, -28653, -26605,
			// 120 - 127
			2061, 2061, 2061, 2061, 2061, 2061, 2061, 2061,
			// 128 - 135
			424, 424, 424, 424, 424, 424, 424, 424,
			// 136 - 143
			424, 424, 424, 424, 424, 424, 424, 424,
			// 144 - 151
			424, 424, 424, 424, 424, 424, 424, 424,
			// 152 - 159
			424, 424, 424, 424, 424, 424, 424, 424,
			// 160 - 167
			750, 750, 750, 750, 1616, 1616, 1648, 1648,
			// 168 - 175
			1424, 1424, 1456, 1456, 1488, 1488, 1520, 1520,
			// 176 - 183
			1840, 1840, 1872, 1872, 1968, 1968, 8209, 8209,
			// 184 - 191
			524, 524, 524, 524, 524, 524, 524, 524,
			// 192 - 199
			556, 556, 556, 556, 556, 556, 556, 556,
			// 200 - 207
			1552, 1552, 1584, 1584, 2000, 2000, 2032, 2032,
			// 208 - 215
			976, 976, 1008, 1008, 1040, 1040, 1072, 1072,
			// 216 - 223
			1296, 1296, 1328, 1328, 718, 718, 718, 718,
			// 224 - 231
			456, 456, 456, 456, 456, 456, 456, 456,
			// 232 - 239
			456, 456, 456, 456, 456, 456, 456, 456,
			// 240 - 247
			456, 456, 456, 456, 456, 456, 456, 456,
			// 248 - 255
			456, 456, 456, 456, 456, 456, 456, 456,
			// 256 - 263
			326, 326, 326, 326, 326, 326, 326, 326,
			// 264 - 271
			326, 326, 326, 326, 326, 326, 326, 326,
			// 272 - 279
			326, 326, 326, 326, 326, 326, 326, 326,
			// 280 - 287
			326, 326, 326, 326, 326, 326, 326, 326,
			// 288 - 295
			326, 326, 326, 326, 326, 326, 326, 326,
			// 296 - 303
			326, 326, 326, 326, 326, 326, 326, 326,
			// 304 - 311
			326, 326, 326, 326, 326, 326, 326, 326,
			// 312 - 319
			326, 326, 326, 326, 326, 326, 326, 326,
			// 320 - 327
			358, 358, 358, 358, 358, 358, 358, 358,
			// 328 - 335
			358, 358, 358, 358, 358, 358, 358, 358,
			// 336 - 343
			358, 358, 358, 358, 358, 358, 358, 358,
			// 344 - 351
			358, 358, 358, 358, 358, 358, 358, 358,
			// 352 - 359
			358, 358, 358, 358, 358, 358, 358, 358,
			// 360 - 367
			358, 358, 358, 358, 358, 358, 358, 358,
			// 368 - 375
			358, 358, 358, 358, 358, 358, 358, 358,
			// 376 - 383
			358, 358, 358, 358, 358, 358, 358, 358,
			// 384 - 391
			490, 490, 490, 490, 490, 490, 490, 490,
			// 392 - 399
			490, 490, 490, 490, 490, 490, 490, 490,
			// 400 - 407
			4113, 4113, 6161, 6161, 848, 848, 880, 880,
			// 408 - 415
			912, 912, 944, 944, 622, 622, 622, 622,
			// 416 - 423
			654, 654, 654, 654, 1104, 1104, 1136, 1136,
			// 424 - 431
			1168, 1168, 1200, 1200, 1232, 1232, 1264, 1264,
			// 432 - 439
			686, 686, 686, 686, 1360, 1360, 1392, 1392,
			// 440 - 447
			12, 12, 12, 12, 12, 12, 12, 12,
			// 448 - 455
			390, 390, 390, 390, 390, 390, 390, 390,
			// 456 - 463
			390, 390, 390, 390, 390, 390, 390, 390,
			// 464 - 471
			390, 390, 390, 390, 390, 390, 390, 390,
			// 472 - 479
			390, 390, 390, 390, 390, 390, 390, 390,
			// 480 - 487
			390, 390, 390, 390, 390, 390, 390, 390,
			// 488 - 495
			390, 390, 390, 390, 390, 390, 390, 390,
			// 496 - 503
			390, 390, 390, 390, 390, 390, 390, 390,
			// 504 - 511
			390, 390, 390, 390, 390, 390, 390, 390, };

	static byte[] twoDCodes = {
	// 0 - 7
			80, 88, 23, 71, 30, 30, 62, 62, // 8 - 15
			4, 4, 4, 4, 4, 4, 4, 4, // 16 - 23
			11, 11, 11, 11, 11, 11, 11, 11, // 24 - 31
			11, 11, 11, 11, 11, 11, 11, 11, // 32 - 39
			35, 35, 35, 35, 35, 35, 35, 35, // 40 - 47
			35, 35, 35, 35, 35, 35, 35, 35, // 48 - 55
			51, 51, 51, 51, 51, 51, 51, 51, // 56 - 63
			51, 51, 51, 51, 51, 51, 51, 51, // 64 - 71
			41, 41, 41, 41, 41, 41, 41, 41, // 72 - 79
			41, 41, 41, 41, 41, 41, 41, 41, // 80 - 87
			41, 41, 41, 41, 41, 41, 41, 41, // 88 - 95
			41, 41, 41, 41, 41, 41, 41, 41, // 96 - 103
			41, 41, 41, 41, 41, 41, 41, 41, // 104 - 111
			41, 41, 41, 41, 41, 41, 41, 41, // 112 - 119
			41, 41, 41, 41, 41, 41, 41, 41, // 120 - 127
			41, 41, 41, 41, 41, 41, 41, 41, };

	private int bitPointer;

	private int bytePointer;

	private byte[] data;

	private int w;

	private boolean align = false;

	private int fillOrder;

	// Data structures needed to store changing elements for the previous
	// and the current scanline
	private int changingElemSize = 0;

	private int[] prevChangingElems;

	private int[] currChangingElems;

	// Element at which to start search in getNextChangingElement
	private int lastChangingElement = 0;

	private boolean fillBits = false;

	/**
	 * @param fillOrder
	 *            The fill order of the compressed data bytes.
	 * @param w
	 * @param h
	 */
	public CCITTFaxDecoder(int fillOrder, int w, int h) {
		this.fillOrder = fillOrder;
		this.w = w;

		this.bitPointer = 0;
		this.bytePointer = 0;
		this.prevChangingElems = new int[w];
		this.currChangingElems = new int[w];
	}

	private boolean align() {
		if (align && bitPointer != 0) {
			bytePointer++;
			bitPointer = 0;
			return true;
		}
		return false;
	}

	protected boolean consumeEOL() {
		// Get the next 12 bits.
		int next12Bits = nextNBits(12);
		if (next12Bits == 1) {
			// EOL found & consumed
			return true;
		}
		// no EOL - unread and return
		updatePointer(12);
		return false;
	}

	// Returns run length
	private int decodeBlackCodeWord() {
		int current;
		int entry;
		int bits;
		int isT;
		int code = -1;
		int runLength = 0;
		boolean isWhite = false;

		while (!isWhite) {
			current = nextLesserThan8Bits(4);
			entry = initBlack[current];

			// Get the 3 fields from the entry
			isT = entry & 0x0001;
			bits = (entry >>> 1) & 0x000f;
			code = (entry >>> 5) & 0x07ff;

			if (code == 100) {
				current = nextNBits(9);
				entry = black[current];

				// Get the 3 fields from the entry
				isT = entry & 0x0001;
				bits = (entry >>> 1) & 0x000f;
				code = (entry >>> 5) & 0x07ff;

				if (bits == 12) {
					// Additional makeup codes
					updatePointer(5);
					current = nextLesserThan8Bits(4);
					entry = additionalMakeup[current];
					bits = (entry >>> 1) & 0x07; // 3 bits 0000 0111
					code = (entry >>> 4) & 0x0fff; // 12 bits
					runLength += code;

					updatePointer(4 - bits);
				} else if (bits == 15) {
					// EOL code
					throw new RuntimeException(
							"EOL code word encountered in Black run."); //$NON-NLS-1$
				} else {
					runLength += code;
					updatePointer(9 - bits);
					if (isT == 0) {
						isWhite = true;
					}
				}
			} else if (code == 200) {
				// Is a Terminating code
				current = nextLesserThan8Bits(2);
				entry = twoBitBlack[current];
				code = (entry >>> 5) & 0x07ff;
				runLength += code;
				bits = (entry >>> 1) & 0x0f;
				updatePointer(2 - bits);
				isWhite = true;
			} else {
				// Is a Terminating code
				runLength += code;
				updatePointer(4 - bits);
				isWhite = true;
			}
		}

		return runLength;
	}

	protected void decodeNextScanline(byte[] buffer, int lineOffset,
			int bitOffset) {
		int bits = 0;
		int code = 0;
		int isT = 0;
		int current;
		int entry;
		int twoBits;
		boolean isWhite = true;

		// Initialize starting of the changing elements array
		changingElemSize = 0;

		// While scanline not complete
		while (bitOffset < w) {
			while (isWhite) {
				// White run
				current = nextNBits(10);
				entry = white[current];

				// Get the 3 fields from the entry
				isT = entry & 0x0001;
				bits = (entry >>> 1) & 0x0f;

				if (bits == 12) { // Additional Make up code
					// Get the next 2 bits
					twoBits = nextLesserThan8Bits(2);
					// Consolidate the 2 new bits and last 2 bits into 4 bits
					current = ((current << 2) & 0x000c) | twoBits;
					entry = additionalMakeup[current];
					bits = (entry >>> 1) & 0x07; // 3 bits 0000 0111
					code = (entry >>> 4) & 0x0fff; // 12 bits
					bitOffset += code; // Skip white run

					updatePointer(4 - bits);
				} else if (bits == 0) { // ERROR
					throw new RuntimeException("Invalid code encountered.");
				} else if (bits == 15) {
					// EOL recover
					// move bits back...
					updatePointer(10);
					return;
				} else {
					// 11 bits - 0000 0111 1111 1111 = 0x07ff
					code = (entry >>> 5) & 0x07ff;
					bitOffset += code;

					updatePointer(10 - bits);
					if (isT == 0) {
						isWhite = false;
						currChangingElems[changingElemSize++] = bitOffset;
					}
				}
			}

			// Check whether this run completed one width, if so
			// advance to next byte boundary for compression = 2.
			if (bitOffset == w) {
				align();
				break;
			}

			while (isWhite == false) {
				// Black run
				current = nextLesserThan8Bits(4);
				entry = initBlack[current];

				// Get the 3 fields from the entry
				isT = entry & 0x0001;
				bits = (entry >>> 1) & 0x000f;
				code = (entry >>> 5) & 0x07ff;

				if (code == 100) {
					current = nextNBits(9);
					entry = black[current];

					// Get the 3 fields from the entry
					isT = entry & 0x0001;
					bits = (entry >>> 1) & 0x000f;
					code = (entry >>> 5) & 0x07ff;

					if (bits == 12) {
						// Additional makeup codes
						updatePointer(5);
						current = nextLesserThan8Bits(4);
						entry = additionalMakeup[current];
						bits = (entry >>> 1) & 0x07; // 3 bits 0000 0111
						code = (entry >>> 4) & 0x0fff; // 12 bits

						setToBlack(buffer, lineOffset, bitOffset, code);
						bitOffset += code;

						updatePointer(4 - bits);
					} else if (bits == 15) {
						// EOL recover
						// unread bits ???
						updatePointer(9);
						return;
					} else {
						setToBlack(buffer, lineOffset, bitOffset, code);
						bitOffset += code;

						updatePointer(9 - bits);
						if (isT == 0) {
							isWhite = true;
							currChangingElems[changingElemSize++] = bitOffset;
						}
					}
				} else if (code == 200) {
					// Is a Terminating code
					current = nextLesserThan8Bits(2);
					entry = twoBitBlack[current];
					code = (entry >>> 5) & 0x07ff;
					bits = (entry >>> 1) & 0x0f;

					setToBlack(buffer, lineOffset, bitOffset, code);
					bitOffset += code;

					updatePointer(2 - bits);
					isWhite = true;
					currChangingElems[changingElemSize++] = bitOffset;
				} else {
					// Is a Terminating code
					setToBlack(buffer, lineOffset, bitOffset, code);
					bitOffset += code;

					updatePointer(4 - bits);
					isWhite = true;
					currChangingElems[changingElemSize++] = bitOffset;
				}
			}

			// Check whether this run completed one width
			if (bitOffset == w) {
				align();
				break;
			}
		}

		currChangingElems[changingElemSize++] = bitOffset;
	}

	// One-dimensional decoding methods
	public void decodeT41D(byte[] buffer, byte[] compData, int startX,
			int height) {
		this.data = compData;
		int scanlineStride = (w + 7) / 8;
		bitPointer = 0;
		bytePointer = 0;

		int lineOffset = 0;
		for (int i = 0; i < height; i++) {
			consumeEOL();
			decodeNextScanline(buffer, lineOffset, startX);
			lineOffset += scanlineStride;
		}
	}

	// Two-dimensional decoding methods
	public void decodeT42D(byte[] buffer, byte[] compData, int startX,
			int height) {
		this.data = compData;
		int scanlineStride = (w + 7) / 8;
		bitPointer = 0;
		bytePointer = 0;

		int a0;
		int a1;
		int b1;
		int b2;
		int[] b = new int[2];
		int entry;
		int code;
		int bits;
		boolean isWhite;
		int currIndex = 0;
		int[] temp;

		// The data must start with an EOL code
		if (readEOL(true) != 1) {
			throw new RuntimeException("First scanline must be 1D encoded."); //$NON-NLS-1$
		}

		int lineOffset = 0;
		int bitOffset;

		// Then the 1D encoded scanline data will occur, changing elements
		// array gets set.
		decodeNextScanline(buffer, lineOffset, startX);
		lineOffset += scanlineStride;

		for (int lines = 1; lines < height; lines++) {
			// Every line must begin with an EOL followed by a bit which
			// indicates whether the following scanline is 1D or 2D encoded.
			if (readEOL(false) == 0) {
				// 2D encoded scanline follows

				// Initialize previous scanlines changing elements, and
				// initialize current scanline's changing elements array
				temp = prevChangingElems;
				prevChangingElems = currChangingElems;
				currChangingElems = temp;
				currIndex = 0;

				// a0 has to be set just before the start of this scanline.
				a0 = -1;
				isWhite = true;
				bitOffset = startX;

				lastChangingElement = 0;

				while (bitOffset < w) {
					// Get the next changing element
					getNextChangingElement(a0, isWhite, b);

					b1 = b[0];
					b2 = b[1];

					// Get the next seven bits
					entry = nextLesserThan8Bits(7);

					// Run these through the 2DCodes table
					entry = (twoDCodes[entry] & 0xff);

					// Get the code and the number of bits used up
					code = (entry & 0x78) >>> 3;
					bits = entry & 0x07;

					if (code == 0) {
						if (!isWhite) {
							setToBlack(buffer, lineOffset, bitOffset, b2
									- bitOffset);
						}
						bitOffset = a0 = b2;

						// Set pointer to consume the correct number of bits.
						updatePointer(7 - bits);
					} else if (code == 1) {
						// Horizontal
						updatePointer(7 - bits);

						// identify the next 2 codes.
						int number;
						if (isWhite) {
							number = decodeWhiteCodeWord();
							bitOffset += number;
							currChangingElems[currIndex++] = bitOffset;

							number = decodeBlackCodeWord();
							setToBlack(buffer, lineOffset, bitOffset, number);
							bitOffset += number;
							currChangingElems[currIndex++] = bitOffset;
						} else {
							number = decodeBlackCodeWord();
							setToBlack(buffer, lineOffset, bitOffset, number);
							bitOffset += number;
							currChangingElems[currIndex++] = bitOffset;

							number = decodeWhiteCodeWord();
							bitOffset += number;
							currChangingElems[currIndex++] = bitOffset;
						}

						a0 = bitOffset;
					} else if (code <= 8) {
						// Vertical
						a1 = b1 + (code - 5);

						currChangingElems[currIndex++] = a1;

						// We write the current color till a1 - 1 pos,
						// since a1 is where the next color starts
						if (!isWhite) {
							setToBlack(buffer, lineOffset, bitOffset, a1
									- bitOffset);
						}
						bitOffset = a0 = a1;
						isWhite = !isWhite;

						updatePointer(7 - bits);
					} else {
						throw new RuntimeException(
								"Invalid code encountered while decoding 2D group 3 compressed data."); //$NON-NLS-1$
					}
				}

				// Add the changing element beyond the current scanline for the
				// other color too
				currChangingElems[currIndex++] = bitOffset;
				changingElemSize = currIndex;
			} else {
				// 1D encoded scanline follows
				decodeNextScanline(buffer, lineOffset, startX);
			}

			lineOffset += scanlineStride;
		}
	}

	public synchronized void decodeT6(byte[] buffer, byte[] compData,
			int startX, int height) {
		this.data = compData;
		int scanlineStride = (w + 7) / 8;
		bitPointer = 0;
		bytePointer = 0;

		int a0;
		int a1;
		int b1;
		int b2;
		int entry;
		int code;
		int bits;
		boolean isWhite;
		int currIndex;
		int[] temp;

		// Return values from getNextChangingElement
		int[] b = new int[2];

		// uncompressedMode - have written some code for this, but this
		// has not been tested due to lack of test images using this optional

		// Local cached reference
		int[] cce = currChangingElems;

		// Assume invisible preceding row of all white pixels and insert
		// both black and white changing elements beyond the end of this
		// imaginary scanline.
		changingElemSize = 0;
		cce[changingElemSize++] = w;
		cce[changingElemSize++] = w;

		int lineOffset = 0;
		int bitOffset;

		for (int lines = 0; lines < height; lines++) {
			// a0 has to be set just before the start of the scanline.
			a0 = -1;
			isWhite = true;

			// Assign the changing elements of the previous scanline to
			// prevChangingElems and start putting this new scanline's
			// changing elements into the currChangingElems.
			temp = prevChangingElems;
			prevChangingElems = currChangingElems;
			cce = currChangingElems = temp;
			currIndex = 0;

			// Start decoding the scanline at startX in the raster
			bitOffset = startX;

			// Reset search start position for getNextChangingElement
			lastChangingElement = 0;

			// Till one whole scanline is decoded
			while (bitOffset < w) {
				// Get the next changing element
				getNextChangingElement(a0, isWhite, b);
				b1 = b[0];
				b2 = b[1];

				// Get the next seven bits
				entry = nextLesserThan8Bits(7);
				// Run these through the 2DCodes table
				entry = (twoDCodes[entry] & 0xff);

				// Get the code and the number of bits used up
				code = (entry & 0x78) >>> 3;
				bits = entry & 0x07;

				if (code == 0) { // Pass
					// We always assume WhiteIsZero format for fax.
					if (!isWhite) {
						if (b2 > w) {
							b2 = w;
						}
						setToBlack(buffer, lineOffset, bitOffset, b2
								- bitOffset);
					}
					bitOffset = a0 = b2;

					// Set pointer to only consume the correct number of bits.
					updatePointer(7 - bits);
				} else if (code == 1) { // Horizontal
					// Set pointer to only consume the correct number of bits.
					updatePointer(7 - bits);

					// identify the next 2 alternating color codes.
					int number;
					if (isWhite) {
						// Following are white and black runs
						number = decodeWhiteCodeWord();
						bitOffset += number;
						cce[currIndex++] = bitOffset;

						number = decodeBlackCodeWord();
						if (number > w - bitOffset) {
							number = w - bitOffset;
						}
						setToBlack(buffer, lineOffset, bitOffset, number);
						bitOffset += number;
						cce[currIndex++] = bitOffset;
					} else {
						// First a black run and then a white run follows
						number = decodeBlackCodeWord();
						if (number > w - bitOffset) {
							number = w - bitOffset;
						}
						setToBlack(buffer, lineOffset, bitOffset, number);
						bitOffset += number;
						cce[currIndex++] = bitOffset;

						number = decodeWhiteCodeWord();
						bitOffset += number;
						cce[currIndex++] = bitOffset;
					}

					a0 = bitOffset;
				} else if (code <= 8) { // Vertical
					a1 = b1 + (code - 5);
					cce[currIndex++] = a1;

					// We write the current color till a1 - 1 pos,
					// since a1 is where the next color starts
					if (!isWhite) {
						if (a1 > w) {
							a1 = w;
						}
						setToBlack(buffer, lineOffset, bitOffset, a1
								- bitOffset);
					}
					bitOffset = a0 = a1;
					isWhite = !isWhite;

					updatePointer(7 - bits);
				} else if (code == 11) {
					if (nextLesserThan8Bits(3) != 7) {
						throw new RuntimeException(
								"Invalid code encountered while decoding 2D group 4 compressed data."); //$NON-NLS-1$
					}

					int zeros = 0;
					boolean exit = false;

					while (!exit) {
						while (nextLesserThan8Bits(1) != 1) {
							zeros++;
						}

						if (zeros > 5) {
							// Exit code

							// Zeros before exit code
							zeros = zeros - 6;

							if (!isWhite && (zeros > 0)) {
								cce[currIndex++] = bitOffset;
							}

							// Zeros before the exit code
							bitOffset += zeros;
							if (zeros > 0) {
								// Some zeros have been written
								isWhite = true;
							}

							// Read in the bit which specifies the color of
							// the following run
							if (nextLesserThan8Bits(1) == 0) {
								if (!isWhite) {
									cce[currIndex++] = bitOffset;
								}
								isWhite = true;
							} else {
								if (isWhite) {
									cce[currIndex++] = bitOffset;
								}
								isWhite = false;
							}

							exit = true;
						}

						if (zeros == 5) {
							if (!isWhite) {
								cce[currIndex++] = bitOffset;
							}
							bitOffset += zeros;

							// Last thing written was white
							isWhite = true;
						} else {
							bitOffset += zeros;

							cce[currIndex++] = bitOffset;
							setToBlack(buffer, lineOffset, bitOffset, 1);
							++bitOffset;

							// Last thing written was black
							isWhite = false;
						}
					}
				} else {
					// break line - seems to be a common failure
					// unread
					updatePointer(7 - bits);
					// and mark lines as complete
					bitOffset = w;
					// throw new RuntimeException(
					// "Invalid code encountered while decoding 2D group 4
					// compressed data."); //$NON-NLS-1$
				}
			}

			align();

			// Add the changing element beyond the current scanline for the
			// other color too
			// make sure that the index does not exceed the bounds of the array
			if (currIndex <= w) {
				cce[currIndex++] = bitOffset;
			}

			// Number of changing elements in this scanline.
			changingElemSize = currIndex;

			lineOffset += scanlineStride;
		}
	}

	// Returns run length
	private int decodeWhiteCodeWord() {
		int current;
		int entry;
		int bits;
		int isT;
		int twoBits;
		int code = -1;
		int runLength = 0;
		boolean isWhite = true;

		while (isWhite) {
			current = nextNBits(10);
			entry = white[current];

			// Get the 3 fields from the entry
			isT = entry & 0x0001;
			bits = (entry >>> 1) & 0x0f;

			if (bits == 12) { // Additional Make up code
				// Get the next 2 bits
				twoBits = nextLesserThan8Bits(2);
				// Consolidate the 2 new bits and last 2 bits into 4 bits
				current = ((current << 2) & 0x000c) | twoBits;
				entry = additionalMakeup[current];
				bits = (entry >>> 1) & 0x07; // 3 bits 0000 0111
				code = (entry >>> 4) & 0x0fff; // 12 bits
				runLength += code;
				updatePointer(4 - bits);
			} else if (bits == 0) { // ERROR
				throw new RuntimeException("Invalid code encountered."); //$NON-NLS-1$
			} else if (bits == 15) { // EOL
				throw new RuntimeException(
						"EOL code word encountered in White run."); //$NON-NLS-1$
			} else {
				// 11 bits - 0000 0111 1111 1111 = 0x07ff
				code = (entry >>> 5) & 0x07ff;
				runLength += code;
				updatePointer(10 - bits);
				if (isT == 0) {
					isWhite = false;
				}
			}
		}

		return runLength;
	}

	private void getNextChangingElement(int a0, boolean isWhite, int[] ret) {
		// Local copies of instance variables
		int[] pce = this.prevChangingElems;
		int ces = this.changingElemSize;

		// If the previous match was at an odd element, we still
		// have to search the preceeding element.
		// int start = lastChangingElement & ~0x1;
		int start = (lastChangingElement > 0) ? (lastChangingElement - 1) : 0;
		if (isWhite) {
			start &= ~0x1; // Search even numbered elements
		} else {
			start |= 0x1; // Search odd numbered elements
		}

		int i = start;
		for (; i < ces; i += 2) {
			int temp = pce[i];
			if (temp > a0) {
				lastChangingElement = i;
				ret[0] = temp;
				break;
			}
		}

		if ((i + 1) < ces) {
			ret[1] = pce[i + 1];
		}
	}

	public boolean isAlign() {
		return align;
	}

	public boolean isFillBits() {
		return fillBits;
	}

	private int nextLesserThan8Bits(int bitsToGet) {
		byte b;
		byte next;
		int l = data.length - 1;
		int bp = this.bytePointer;

		if (fillOrder == 1) {
			b = data[bp];
			if (bp == l) {
				next = 0x00;
			} else {
				next = data[bp + 1];
			}
		} else if (fillOrder == 2) {
			b = flipTable[data[bp] & 0xff];
			if (bp == l) {
				next = 0x00;
			} else {
				next = flipTable[data[bp + 1] & 0xff];
			}
		} else {
			throw new RuntimeException("tag must be either 1 or 2."); //$NON-NLS-1$
		}

		int bitsLeft = 8 - bitPointer;
		int bitsFromNextByte = bitsToGet - bitsLeft;

		int shift = bitsLeft - bitsToGet;
		int i1;
		int i2;
		if (shift >= 0) {
			i1 = (b & table1[bitsLeft]) >>> shift;
			bitPointer += bitsToGet;
			if (bitPointer == 8) {
				bitPointer = 0;
				bytePointer++;
			}
		} else {
			i1 = (b & table1[bitsLeft]) << (-shift);
			i2 = (next & table2[bitsFromNextByte]) >>> (8 - bitsFromNextByte);

			i1 |= i2;
			bytePointer++;
			bitPointer = bitsFromNextByte;
		}

		return i1;
	}

	private int nextNBits(int bitsToGet) {
		byte b;
		byte next;
		byte next2next;
		int l = data.length - 1;
		int bp = this.bytePointer;

		if (fillOrder == 1) {
			b = data[bp];

			if (bp == l) {
				next = 0x00;
				next2next = 0x00;
			} else if ((bp + 1) == l) {
				next = data[bp + 1];
				next2next = 0x00;
			} else {
				next = data[bp + 1];
				next2next = data[bp + 2];
			}
		} else if (fillOrder == 2) {
			b = flipTable[data[bp] & 0xff];

			if (bp == l) {
				next = 0x00;
				next2next = 0x00;
			} else if ((bp + 1) == l) {
				next = flipTable[data[bp + 1] & 0xff];
				next2next = 0x00;
			} else {
				next = flipTable[data[bp + 1] & 0xff];
				next2next = flipTable[data[bp + 2] & 0xff];
			}
		} else {
			throw new RuntimeException("tag must be either 1 or 2."); //$NON-NLS-1$
		}

		int bitsLeft = 8 - bitPointer;
		int bitsFromNextByte = bitsToGet - bitsLeft;
		int bitsFromNext2NextByte = 0;
		if (bitsFromNextByte > 8) {
			bitsFromNext2NextByte = bitsFromNextByte - 8;
			bitsFromNextByte = 8;
		}

		bytePointer++;

		int i1 = (b & table1[bitsLeft]) << (bitsToGet - bitsLeft);
		int i2 = (next & table2[bitsFromNextByte]) >>> (8 - bitsFromNextByte);

		int i3 = 0;
		if (bitsFromNext2NextByte != 0) {
			i2 <<= bitsFromNext2NextByte;
			i3 = (next2next & table2[bitsFromNext2NextByte]) >>> (8 - bitsFromNext2NextByte);
			i2 |= i3;
			bytePointer++;
			bitPointer = bitsFromNext2NextByte;
		} else {
			if (bitsFromNextByte == 8) {
				bitPointer = 0;
				bytePointer++;
			} else {
				bitPointer = bitsFromNextByte;
			}
		}

		int i = i1 | i2;
		return i;
	}

	private int readEOL(boolean isFirstEOL) {
		// Seek to the next EOL.
		if (!seekEOL()) {
			throw new RuntimeException("EOL not found");
		}

		if (!fillBits) {
			int next12Bits = nextNBits(12);
			if (isFirstEOL && (next12Bits == 0)) {
				// Might have the case of EOL padding being used even
				// though it was not flagged.
				// This was observed to be the case in TIFFs produced
				// by a well known vendor who shall remain nameless.
				if (nextNBits(4) == 1) {
					// EOL must be padded: reset the fillBits flag.
					fillBits = true;
					return 1;
				}
			}
			if (next12Bits != 1) {
				throw new RuntimeException(
						"Scanline must begin with EOL code word."); //$NON-NLS-1$
			}
		} else {
			// First EOL code word xxxx 0000 0000 0001 will occur
			// As many fill bits will be present as required to make
			// the EOL code of 12 bits end on a byte boundary.
			int bitsLeft = 8 - bitPointer;

			if (nextNBits(bitsLeft) != 0) {
				throw new RuntimeException(
						"All fill bits preceding EOL code must be 0."); //$NON-NLS-1$
			}

			// If the number of bitsLeft is less than 8, then to have a 12
			// bit EOL sequence, two more bytes are certainly going to be
			// required. The first of them has to be all zeros, so ensure
			// that.
			if (bitsLeft < 4) {
				if (nextNBits(8) != 0) {
					throw new RuntimeException(
							"All fill bits preceding EOL code must be 0."); //$NON-NLS-1$
				}
			}

			//
			// Some encoders under Group 3 Fax compression 1D writes TIFF
			// files without the fill bits, but say otherwise.
			// Need to check for this here.
			//
			int next8 = nextNBits(8);

			if (isFirstEOL && (next8 & 0xf0) == 0x10) {
				//
				// Fill bits are not actually used despite what the flag
				// says. So switch fillBits off and then rewind so that
				// only 12 bits have effectively been read.
				//
				fillBits = false;
				updatePointer(4);
			} else {
				//
				// This is the normal case.
				// There might be a random number of fill bytes with 0s, so
				// loop till the EOL of 0000 0001 is found, as long as all
				// the bytes preceding it are 0's.
				//
				while (next8 != 1) {
					// If not all zeros
					if (next8 != 0) {
						throw new RuntimeException("0 bits expected before EOL");
					}
					next8 = nextNBits(8);
				}
			}
		}
		// The next one bit signifies 1D/2D encoding of next line.
		return nextLesserThan8Bits(1);
	}

	// Seeks to the next EOL in the compressed bitstream.
	// Returns 'true' if and only if an EOL is found; if 'false'
	// is returned it may be inferred that the EOF was reached first.
	private boolean seekEOL() {
		// Set maximum and current bit index into the compressed data.
		int bitIndexMax = data.length * 8 - 1;
		int bitIndex = bytePointer * 8 + bitPointer;

		// Loop while at least 12 bits are available.
		while (bitIndex <= bitIndexMax - 12) {
			// Get the next 12 bits.
			int next12Bits = nextNBits(12);
			bitIndex += 12;

			// Loop while the 12 bits are not unity, i.e., while the EOL
			// has not been reached, and there is at least one bit left.
			while (next12Bits != 1 && bitIndex < bitIndexMax) {
				next12Bits = ((next12Bits & 0x000007ff) << 1)
						| (nextLesserThan8Bits(1) & 0x00000001);
				bitIndex++;
			}

			// If EOL reached, rewind the pointers and return 'true'.
			if (next12Bits == 1) {
				updatePointer(12);
				return true;
			}
		}

		// EOL not found: return 'false'.
		return false;
	}

	public void setAlign(boolean align) {
		this.align = align;
	}

	public void setFillBits(boolean fillBits) {
		this.fillBits = fillBits;
	}

	private void setToBlack(byte[] buffer, int lineOffset, int bitOffset,
			int numBits) {
		int bitNum = (8 * lineOffset) + bitOffset;
		int lastBit = bitNum + numBits;

		int byteNum = bitNum >> 3;

		// Handle bits in first byte
		int shift = bitNum & 0x7;
		if (shift > 0) {
			int maskVal = 1 << (7 - shift);
			byte val = buffer[byteNum];
			while ((maskVal > 0) && (bitNum < lastBit)) {
				val |= maskVal;
				maskVal >>= 1;
				++bitNum;
			}
			buffer[byteNum] = val;
		}

		// Fill in 8 bits at a time
		byteNum = bitNum >> 3;
		while (bitNum < (lastBit - 7)) {
			buffer[byteNum++] = (byte) 255;
			bitNum += 8;
		}

		// Fill in remaining bits
		while (bitNum < lastBit) {
			byteNum = bitNum >> 3;
			buffer[byteNum] |= (1 << (7 - (bitNum & 0x7)));
			++bitNum;
		}
	}

	// Move pointer backwards by given amount of bits
	private void updatePointer(int bitsToMoveBack) {
		if (bitsToMoveBack > 8) {
			bytePointer -= bitsToMoveBack / 8;
			bitsToMoveBack %= 8;
		}

		int i = bitPointer - bitsToMoveBack;

		if (i < 0) {
			bytePointer--;
			bitPointer = 8 + i;
		} else {
			bitPointer = i;
		}
	}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy