| /* | 
 |  * Copyright (c) 1988-1997 Sam Leffler | 
 |  * Copyright (c) 1991-1997 Silicon Graphics, Inc. | 
 |  * | 
 |  * Permission to use, copy, modify, distribute, and sell this software and  | 
 |  * its documentation for any purpose is hereby granted without fee, provided | 
 |  * that (i) the above copyright notices and this permission notice appear in | 
 |  * all copies of the software and related documentation, and (ii) the names of | 
 |  * Sam Leffler and Silicon Graphics may not be used in any advertising or | 
 |  * publicity relating to the software without the specific, prior written | 
 |  * permission of Sam Leffler and Silicon Graphics. | 
 |  *  | 
 |  * THE SOFTWARE IS PROVIDED "AS-IS" AND WITHOUT WARRANTY OF ANY KIND,  | 
 |  * EXPRESS, IMPLIED OR OTHERWISE, INCLUDING WITHOUT LIMITATION, ANY  | 
 |  * WARRANTY OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE.   | 
 |  *  | 
 |  * IN NO EVENT SHALL SAM LEFFLER OR SILICON GRAPHICS BE LIABLE FOR | 
 |  * ANY SPECIAL, INCIDENTAL, INDIRECT OR CONSEQUENTIAL DAMAGES OF ANY KIND, | 
 |  * OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, | 
 |  * WHETHER OR NOT ADVISED OF THE POSSIBILITY OF DAMAGE, AND ON ANY THEORY OF  | 
 |  * LIABILITY, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE  | 
 |  * OF THIS SOFTWARE. | 
 |  */ | 
 |  | 
 | #include "tiffiop.h" | 
 | #ifdef NEXT_SUPPORT | 
 | /* | 
 |  * TIFF Library. | 
 |  * | 
 |  * NeXT 2-bit Grey Scale Compression Algorithm Support | 
 |  */ | 
 |  | 
 | #define SETPIXEL(op, v) {			\ | 
 | 	switch (npixels++ & 3) {		\ | 
 | 	case 0:	op[0]  = (unsigned char) ((v) << 6); break;	\ | 
 | 	case 1:	op[0] |= (v) << 4; break;	\ | 
 | 	case 2:	op[0] |= (v) << 2; break;	\ | 
 | 	case 3:	*op++ |= (v);	   op_offset++; break;	\ | 
 | 	}					\ | 
 | } | 
 |  | 
 | #define LITERALROW	0x00 | 
 | #define LITERALSPAN	0x40 | 
 | #define WHITE   	((1<<2)-1) | 
 |  | 
 | static int | 
 | NeXTDecode(TIFF* tif, uint8* buf, tmsize_t occ, uint16 s) | 
 | { | 
 | 	static const char module[] = "NeXTDecode"; | 
 | 	unsigned char *bp, *op; | 
 | 	tmsize_t cc; | 
 | 	uint8* row; | 
 | 	tmsize_t scanline, n; | 
 |  | 
 | 	(void) s; | 
 | 	/* | 
 | 	 * Each scanline is assumed to start off as all | 
 | 	 * white (we assume a PhotometricInterpretation | 
 | 	 * of ``min-is-black''). | 
 | 	 */ | 
 | 	for (op = (unsigned char*) buf, cc = occ; cc-- > 0;) | 
 | 		*op++ = 0xff; | 
 |  | 
 | 	bp = (unsigned char *)tif->tif_rawcp; | 
 | 	cc = tif->tif_rawcc; | 
 | 	scanline = tif->tif_scanlinesize; | 
 | 	if (occ % scanline) | 
 | 	{ | 
 | 		TIFFErrorExt(tif->tif_clientdata, module, "Fractional scanlines cannot be read"); | 
 | 		return (0); | 
 | 	} | 
 | 	for (row = buf; cc > 0 && occ > 0; occ -= scanline, row += scanline) { | 
 | 		n = *bp++; | 
 | 		cc--; | 
 | 		switch (n) { | 
 | 		case LITERALROW: | 
 | 			/* | 
 | 			 * The entire scanline is given as literal values. | 
 | 			 */ | 
 | 			if (cc < scanline) | 
 | 				goto bad; | 
 | 			_TIFFmemcpy(row, bp, scanline); | 
 | 			bp += scanline; | 
 | 			cc -= scanline; | 
 | 			break; | 
 | 		case LITERALSPAN: { | 
 | 			tmsize_t off; | 
 | 			/* | 
 | 			 * The scanline has a literal span that begins at some | 
 | 			 * offset. | 
 | 			 */ | 
 | 			if( cc < 4 ) | 
 | 				goto bad; | 
 | 			off = (bp[0] * 256) + bp[1]; | 
 | 			n = (bp[2] * 256) + bp[3]; | 
 | 			if (cc < 4+n || off+n > scanline) | 
 | 				goto bad; | 
 | 			_TIFFmemcpy(row+off, bp+4, n); | 
 | 			bp += 4+n; | 
 | 			cc -= 4+n; | 
 | 			break; | 
 | 		} | 
 | 		default: { | 
 | 			uint32 npixels = 0, grey; | 
 | 			tmsize_t op_offset = 0; | 
 | 			uint32 imagewidth = tif->tif_dir.td_imagewidth; | 
 |             if( isTiled(tif) ) | 
 |                 imagewidth = tif->tif_dir.td_tilewidth; | 
 |  | 
 | 			/* | 
 | 			 * The scanline is composed of a sequence of constant | 
 | 			 * color ``runs''.  We shift into ``run mode'' and | 
 | 			 * interpret bytes as codes of the form | 
 | 			 * <color><npixels> until we've filled the scanline. | 
 | 			 */ | 
 | 			op = row; | 
 | 			for (;;) { | 
 | 				grey = (uint32)((n>>6) & 0x3); | 
 | 				n &= 0x3f; | 
 | 				/* | 
 | 				 * Ensure the run does not exceed the scanline | 
 | 				 * bounds, potentially resulting in a security | 
 | 				 * issue. | 
 | 				 */ | 
 | 				while (n-- > 0 && npixels < imagewidth && op_offset < scanline) | 
 | 					SETPIXEL(op, grey); | 
 | 				if (npixels >= imagewidth) | 
 | 					break; | 
 |                 if (op_offset >= scanline ) { | 
 |                     TIFFErrorExt(tif->tif_clientdata, module, "Invalid data for scanline %ld", | 
 |                         (long) tif->tif_row); | 
 |                     return (0); | 
 |                 } | 
 | 				if (cc == 0) | 
 | 					goto bad; | 
 | 				n = *bp++; | 
 | 				cc--; | 
 | 			} | 
 | 			break; | 
 | 		} | 
 | 		} | 
 | 	} | 
 | 	tif->tif_rawcp = (uint8*) bp; | 
 | 	tif->tif_rawcc = cc; | 
 | 	return (1); | 
 | bad: | 
 | 	TIFFErrorExt(tif->tif_clientdata, module, "Not enough data for scanline %ld", | 
 | 	    (long) tif->tif_row); | 
 | 	return (0); | 
 | } | 
 |  | 
 | static int | 
 | NeXTPreDecode(TIFF* tif, uint16 s) | 
 | { | 
 | 	static const char module[] = "NeXTPreDecode"; | 
 | 	TIFFDirectory *td = &tif->tif_dir; | 
 | 	(void)s; | 
 |  | 
 | 	if( td->td_bitspersample != 2 ) | 
 | 	{ | 
 | 		TIFFErrorExt(tif->tif_clientdata, module, "Unsupported BitsPerSample = %d", | 
 | 					 td->td_bitspersample); | 
 | 		return (0); | 
 | 	} | 
 | 	return (1); | 
 | } | 
 | 	 | 
 | int | 
 | TIFFInitNeXT(TIFF* tif, int scheme) | 
 | { | 
 | 	(void) scheme; | 
 | 	tif->tif_predecode = NeXTPreDecode;   | 
 | 	tif->tif_decoderow = NeXTDecode;   | 
 | 	tif->tif_decodestrip = NeXTDecode;   | 
 | 	tif->tif_decodetile = NeXTDecode; | 
 | 	return (1); | 
 | } | 
 | #endif /* NEXT_SUPPORT */ | 
 |  | 
 | /* vim: set ts=8 sts=8 sw=8 noet: */ | 
 | /* | 
 |  * Local Variables: | 
 |  * mode: c | 
 |  * c-basic-offset: 8 | 
 |  * fill-column: 78 | 
 |  * End: | 
 |  */ |