| /* | 
 |  * 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_t *buf, tmsize_t occ, uint16_t s) | 
 | { | 
 |     static const char module[] = "NeXTDecode"; | 
 |     unsigned char *bp, *op; | 
 |     tmsize_t cc; | 
 |     uint8_t *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) | 
 |     { | 
 |         TIFFErrorExtR(tif, 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_t npixels = 0, grey; | 
 |                 tmsize_t op_offset = 0; | 
 |                 uint32_t 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_t)((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) | 
 |                     { | 
 |                         TIFFErrorExtR(tif, module, | 
 |                                       "Invalid data for scanline %" PRIu32, | 
 |                                       tif->tif_row); | 
 |                         return (0); | 
 |                     } | 
 |                     if (cc == 0) | 
 |                         goto bad; | 
 |                     n = *bp++; | 
 |                     cc--; | 
 |                 } | 
 |                 break; | 
 |             } | 
 |         } | 
 |     } | 
 |     tif->tif_rawcp = (uint8_t *)bp; | 
 |     tif->tif_rawcc = cc; | 
 |     return (1); | 
 | bad: | 
 |     TIFFErrorExtR(tif, module, "Not enough data for scanline %" PRIu32, | 
 |                   tif->tif_row); | 
 |     return (0); | 
 | } | 
 |  | 
 | static int NeXTPreDecode(TIFF *tif, uint16_t s) | 
 | { | 
 |     static const char module[] = "NeXTPreDecode"; | 
 |     TIFFDirectory *td = &tif->tif_dir; | 
 |     (void)s; | 
 |  | 
 |     if (td->td_bitspersample != 2) | 
 |     { | 
 |         TIFFErrorExtR(tif, module, "Unsupported BitsPerSample = %" PRIu16, | 
 |                       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 */ |