pngrutil.c
上传用户:sesekoo
上传日期:2020-07-18
资源大小:21543k
文件大小:96k
源码类别:

界面编程

开发平台:

Visual C++

  1. /* pngrutil.c - utilities to read a PNG file
  2.  *
  3.  * Last changed in libpng 1.2.34 [December 18, 2008]
  4.  * For conditions of distribution and use, see copyright notice in png.h
  5.  * Copyright (c) 1998-2008 Glenn Randers-Pehrson
  6.  * (Version 0.96 Copyright (c) 1996, 1997 Andreas Dilger)
  7.  * (Version 0.88 Copyright (c) 1995, 1996 Guy Eric Schalnat, Group 42, Inc.)
  8.  *
  9.  * This file contains routines that are only called from within
  10.  * libpng itself during the course of reading an image.
  11.  */
  12. #if ( ! defined _CRT_SECURE_NO_WARNINGS )
  13. #define _CRT_SECURE_NO_WARNINGS
  14. #endif
  15. #define PNG_INTERNAL
  16. #include "png.h"
  17. #if defined(PNG_READ_SUPPORTED)
  18. #if defined(_WIN32_WCE) && (_WIN32_WCE<0x500)
  19. #  define WIN32_WCE_OLD
  20. #endif
  21. #ifdef PNG_FLOATING_POINT_SUPPORTED
  22. #  if defined(WIN32_WCE_OLD)
  23. /* strtod() function is not supported on WindowsCE */
  24. __inline double png_strtod(png_structp png_ptr, PNG_CONST char *nptr, char **endptr)
  25. {
  26.    double result = 0;
  27.    int len;
  28.    wchar_t *str, *end;
  29.    len = MultiByteToWideChar(CP_ACP, 0, nptr, -1, NULL, 0);
  30.    str = (wchar_t *)png_malloc(png_ptr, len * png_sizeof(wchar_t));
  31.    if ( NULL != str )
  32.    {
  33.       MultiByteToWideChar(CP_ACP, 0, nptr, -1, str, len);
  34.       result = wcstod(str, &end);
  35.       len = WideCharToMultiByte(CP_ACP, 0, end, -1, NULL, 0, NULL, NULL);
  36.       *endptr = (char *)nptr + (png_strlen(nptr) - len + 1);
  37.       png_free(png_ptr, str);
  38.    }
  39.    return result;
  40. }
  41. #  else
  42. #    define png_strtod(p,a,b) strtod(a,b)
  43. #  endif
  44. #endif
  45. png_uint_32 PNGAPI
  46. png_get_uint_31(png_structp png_ptr, png_bytep buf)
  47. {
  48. #ifdef PNG_READ_BIG_ENDIAN_SUPPORTED
  49.    png_uint_32 i = png_get_uint_32(buf);
  50. #else
  51.    /* Avoid an extra function call by inlining the result. */
  52.    png_uint_32 i = ((png_uint_32)(*buf) << 24) +
  53.       ((png_uint_32)(*(buf + 1)) << 16) +
  54.       ((png_uint_32)(*(buf + 2)) << 8) +
  55.       (png_uint_32)(*(buf + 3));
  56. #endif
  57.    if (i > PNG_UINT_31_MAX)
  58.      png_error(png_ptr, "PNG unsigned integer out of range.");
  59.    return (i);
  60. }
  61. #ifndef PNG_READ_BIG_ENDIAN_SUPPORTED
  62. /* Grab an unsigned 32-bit integer from a buffer in big-endian format. */
  63. png_uint_32 PNGAPI
  64. png_get_uint_32(png_bytep buf)
  65. {
  66.    png_uint_32 i = ((png_uint_32)(*buf) << 24) +
  67.       ((png_uint_32)(*(buf + 1)) << 16) +
  68.       ((png_uint_32)(*(buf + 2)) << 8) +
  69.       (png_uint_32)(*(buf + 3));
  70.    return (i);
  71. }
  72. /* Grab a signed 32-bit integer from a buffer in big-endian format.  The
  73.  * data is stored in the PNG file in two's complement format, and it is
  74.  * assumed that the machine format for signed integers is the same. */
  75. png_int_32 PNGAPI
  76. png_get_int_32(png_bytep buf)
  77. {
  78.    png_int_32 i = ((png_int_32)(*buf) << 24) +
  79.       ((png_int_32)(*(buf + 1)) << 16) +
  80.       ((png_int_32)(*(buf + 2)) << 8) +
  81.       (png_int_32)(*(buf + 3));
  82.    return (i);
  83. }
  84. /* Grab an unsigned 16-bit integer from a buffer in big-endian format. */
  85. png_uint_16 PNGAPI
  86. png_get_uint_16(png_bytep buf)
  87. {
  88.    png_uint_16 i = (png_uint_16)(((png_uint_16)(*buf) << 8) +
  89.       (png_uint_16)(*(buf + 1)));
  90.    return (i);
  91. }
  92. #endif /* PNG_READ_BIG_ENDIAN_SUPPORTED */
  93. /* Read the chunk header (length + type name).
  94.  * Put the type name into png_ptr->chunk_name, and return the length.
  95.  */
  96. png_uint_32 /* PRIVATE */
  97. png_read_chunk_header(png_structp png_ptr)
  98. {
  99.    png_byte buf[8];
  100.    png_uint_32 length;
  101.    /* read the length and the chunk name */
  102.    png_read_data(png_ptr, buf, 8);
  103.    length = png_get_uint_31(png_ptr, buf);
  104.    /* put the chunk name into png_ptr->chunk_name */
  105.    png_memcpy(png_ptr->chunk_name, buf + 4, 4);
  106.    png_debug2(0, "Reading %s chunk, length = %lu",
  107.       png_ptr->chunk_name, length);
  108.    /* reset the crc and run it over the chunk name */
  109.    png_reset_crc(png_ptr);
  110.    png_calculate_crc(png_ptr, png_ptr->chunk_name, 4);
  111.    /* check to see if chunk name is valid */
  112.    png_check_chunk_name(png_ptr, png_ptr->chunk_name);
  113.    return length;
  114. }
  115. /* Read data, and (optionally) run it through the CRC. */
  116. void /* PRIVATE */
  117. png_crc_read(png_structp png_ptr, png_bytep buf, png_size_t length)
  118. {
  119.    if (png_ptr == NULL) return;
  120.    png_read_data(png_ptr, buf, length);
  121.    png_calculate_crc(png_ptr, buf, length);
  122. }
  123. /* Optionally skip data and then check the CRC.  Depending on whether we
  124.    are reading a ancillary or critical chunk, and how the program has set
  125.    things up, we may calculate the CRC on the data and print a message.
  126.    Returns '1' if there was a CRC error, '0' otherwise. */
  127. int /* PRIVATE */
  128. png_crc_finish(png_structp png_ptr, png_uint_32 skip)
  129. {
  130.    png_size_t i;
  131.    png_size_t istop = png_ptr->zbuf_size;
  132.    for (i = (png_size_t)skip; i > istop; i -= istop)
  133.    {
  134.       png_crc_read(png_ptr, png_ptr->zbuf, png_ptr->zbuf_size);
  135.    }
  136.    if (i)
  137.    {
  138.       png_crc_read(png_ptr, png_ptr->zbuf, i);
  139.    }
  140.    if (png_crc_error(png_ptr))
  141.    {
  142.       if (((png_ptr->chunk_name[0] & 0x20) &&                /* Ancillary */
  143.            !(png_ptr->flags & PNG_FLAG_CRC_ANCILLARY_NOWARN)) ||
  144.           (!(png_ptr->chunk_name[0] & 0x20) &&             /* Critical  */
  145.           (png_ptr->flags & PNG_FLAG_CRC_CRITICAL_USE)))
  146.       {
  147.          png_chunk_warning(png_ptr, "CRC error");
  148.       }
  149.       else
  150.       {
  151.          png_chunk_error(png_ptr, "CRC error");
  152.       }
  153.       return (1);
  154.    }
  155.    return (0);
  156. }
  157. /* Compare the CRC stored in the PNG file with that calculated by libpng from
  158.    the data it has read thus far. */
  159. int /* PRIVATE */
  160. png_crc_error(png_structp png_ptr)
  161. {
  162.    png_byte crc_bytes[4];
  163.    png_uint_32 crc;
  164.    int need_crc = 1;
  165.    if (png_ptr->chunk_name[0] & 0x20)                     /* ancillary */
  166.    {
  167.       if ((png_ptr->flags & PNG_FLAG_CRC_ANCILLARY_MASK) ==
  168.           (PNG_FLAG_CRC_ANCILLARY_USE | PNG_FLAG_CRC_ANCILLARY_NOWARN))
  169.          need_crc = 0;
  170.    }
  171.    else                                                    /* critical */
  172.    {
  173.       if (png_ptr->flags & PNG_FLAG_CRC_CRITICAL_IGNORE)
  174.          need_crc = 0;
  175.    }
  176.    png_read_data(png_ptr, crc_bytes, 4);
  177.    if (need_crc)
  178.    {
  179.       crc = png_get_uint_32(crc_bytes);
  180.       return ((int)(crc != png_ptr->crc));
  181.    }
  182.    else
  183.       return (0);
  184. }
  185. #if defined(PNG_READ_zTXt_SUPPORTED) || defined(PNG_READ_iTXt_SUPPORTED) || 
  186.     defined(PNG_READ_iCCP_SUPPORTED)
  187. /*
  188.  * Decompress trailing data in a chunk.  The assumption is that chunkdata
  189.  * points at an allocated area holding the contents of a chunk with a
  190.  * trailing compressed part.  What we get back is an allocated area
  191.  * holding the original prefix part and an uncompressed version of the
  192.  * trailing part (the malloc area passed in is freed).
  193.  */
  194. void /* PRIVATE */
  195. png_decompress_chunk(png_structp png_ptr, int comp_type,
  196.                               png_size_t chunklength,
  197.                               png_size_t prefix_size, png_size_t *newlength)
  198. {
  199.    static PNG_CONST char msg[] = "Error decoding compressed text";
  200.    png_charp text;
  201.    png_size_t text_size;
  202.    if (comp_type == PNG_COMPRESSION_TYPE_BASE)
  203.    {
  204.       int ret = Z_OK;
  205.       png_ptr->zstream.next_in = (png_bytep)(png_ptr->chunkdata + prefix_size);
  206.       png_ptr->zstream.avail_in = (uInt)(chunklength - prefix_size);
  207.       png_ptr->zstream.next_out = png_ptr->zbuf;
  208.       png_ptr->zstream.avail_out = (uInt)png_ptr->zbuf_size;
  209.       text_size = 0;
  210.       text = NULL;
  211.       while (png_ptr->zstream.avail_in)
  212.       {
  213.          ret = inflate(&png_ptr->zstream, Z_PARTIAL_FLUSH);
  214.          if (ret != Z_OK && ret != Z_STREAM_END)
  215.          {
  216.             if (png_ptr->zstream.msg != NULL)
  217.                png_warning(png_ptr, png_ptr->zstream.msg);
  218.             else
  219.                png_warning(png_ptr, msg);
  220.             inflateReset(&png_ptr->zstream);
  221.             png_ptr->zstream.avail_in = 0;
  222.             if (text ==  NULL)
  223.             {
  224.                text_size = prefix_size + png_sizeof(msg) + 1;
  225.                text = (png_charp)png_malloc_warn(png_ptr, (png_uint_32)text_size);
  226.                if (text ==  NULL)
  227.                  {
  228.                     png_free(png_ptr, png_ptr->chunkdata);
  229.                     png_ptr->chunkdata = NULL;
  230.                     png_error(png_ptr, "Not enough memory to decompress chunk");
  231.                  }
  232.                png_memcpy(text, png_ptr->chunkdata, prefix_size);
  233.             }
  234.             text[text_size - 1] = 0x00;
  235.             /* Copy what we can of the error message into the text chunk */
  236.             text_size = (png_size_t)(chunklength -
  237.               (text - png_ptr->chunkdata) - 1);
  238.             if (text_size > png_sizeof(msg))
  239.                text_size = png_sizeof(msg);
  240.             png_memcpy(text + prefix_size, msg, text_size);
  241.             break;
  242.          }
  243.          if (!png_ptr->zstream.avail_out || ret == Z_STREAM_END)
  244.          {
  245.             if (text == NULL)
  246.             {
  247.                text_size = prefix_size +
  248.                    png_ptr->zbuf_size - png_ptr->zstream.avail_out;
  249.                text = (png_charp)png_malloc_warn(png_ptr, (png_uint_32)text_size + 1);
  250.                if (text ==  NULL)
  251.                {
  252.                   png_free(png_ptr, png_ptr->chunkdata);
  253.                   png_ptr->chunkdata = NULL;
  254.                   png_error(png_ptr,
  255.                     "Not enough memory to decompress chunk.");
  256.                }
  257.                png_memcpy(text + prefix_size, png_ptr->zbuf,
  258.                     text_size - prefix_size);
  259.                png_memcpy(text, png_ptr->chunkdata, prefix_size);
  260.                *(text + text_size) = 0x00;
  261.             }
  262.             else
  263.             {
  264.                png_charp tmp;
  265.                tmp = text;
  266.                text = (png_charp)png_malloc_warn(png_ptr,
  267.                   (png_uint_32)(text_size +
  268.                   png_ptr->zbuf_size - png_ptr->zstream.avail_out + 1));
  269.                if (text == NULL)
  270.                {
  271.                   png_free(png_ptr, tmp);
  272.                   png_free(png_ptr, png_ptr->chunkdata);
  273.                   png_ptr->chunkdata = NULL;
  274.                   png_error(png_ptr,
  275.                     "Not enough memory to decompress chunk..");
  276.                }
  277.                png_memcpy(text, tmp, text_size);
  278.                png_free(png_ptr, tmp);
  279.                png_memcpy(text + text_size, png_ptr->zbuf,
  280.                   (png_ptr->zbuf_size - png_ptr->zstream.avail_out));
  281.                text_size += png_ptr->zbuf_size - png_ptr->zstream.avail_out;
  282.                *(text + text_size) = 0x00;
  283.             }
  284.             if (ret == Z_STREAM_END)
  285.                break;
  286.             else
  287.             {
  288.                png_ptr->zstream.next_out = png_ptr->zbuf;
  289.                png_ptr->zstream.avail_out = (uInt)png_ptr->zbuf_size;
  290.             }
  291.          }
  292.       }
  293.       if (ret != Z_STREAM_END)
  294.       {
  295. #if !defined(PNG_NO_STDIO) && !defined(_WIN32_WCE)
  296.          char umsg[52];
  297.          if (ret == Z_BUF_ERROR)
  298.             png_snprintf(umsg, 52,
  299.                 "Buffer error in compressed datastream in %s chunk",
  300.                 png_ptr->chunk_name);
  301.          else if (ret == Z_DATA_ERROR)
  302.             png_snprintf(umsg, 52,
  303.                 "Data error in compressed datastream in %s chunk",
  304.                 png_ptr->chunk_name);
  305.          else
  306.             png_snprintf(umsg, 52,
  307.                 "Incomplete compressed datastream in %s chunk",
  308.                 png_ptr->chunk_name);
  309.          png_warning(png_ptr, umsg);
  310. #else
  311.          png_warning(png_ptr,
  312.             "Incomplete compressed datastream in chunk other than IDAT");
  313. #endif
  314.          text_size = prefix_size;
  315.          if (text ==  NULL)
  316.          {
  317.             text = (png_charp)png_malloc_warn(png_ptr, (png_uint_32)text_size+1);
  318.             if (text == NULL)
  319.               {
  320.                 png_free(png_ptr, png_ptr->chunkdata);
  321.                 png_ptr->chunkdata = NULL;
  322.                 png_error(png_ptr, "Not enough memory for text.");
  323.               }
  324.             png_memcpy(text, png_ptr->chunkdata, prefix_size);
  325.          }
  326.          *(text + text_size) = 0x00;
  327.       }
  328.       inflateReset(&png_ptr->zstream);
  329.       png_ptr->zstream.avail_in = 0;
  330.       png_free(png_ptr, png_ptr->chunkdata);
  331.       png_ptr->chunkdata = text;
  332.       *newlength=text_size;
  333.    }
  334.    else /* if (comp_type != PNG_COMPRESSION_TYPE_BASE) */
  335.    {
  336. #if !defined(PNG_NO_STDIO) && !defined(_WIN32_WCE)
  337.       char umsg[50];
  338.       png_snprintf(umsg, 50, "Unknown zTXt compression type %d", comp_type);
  339.       png_warning(png_ptr, umsg);
  340. #else
  341.       png_warning(png_ptr, "Unknown zTXt compression type");
  342. #endif
  343.       *(png_ptr->chunkdata + prefix_size) = 0x00;
  344.       *newlength = prefix_size;
  345.    }
  346. }
  347. #endif
  348. /* read and check the IDHR chunk */
  349. void /* PRIVATE */
  350. png_handle_IHDR(png_structp png_ptr, png_infop info_ptr, png_uint_32 length)
  351. {
  352.    png_byte buf[13];
  353.    png_uint_32 width, height;
  354.    int bit_depth, color_type, compression_type, filter_type;
  355.    int interlace_type;
  356.    png_debug(1, "in png_handle_IHDR");
  357.    if (png_ptr->mode & PNG_HAVE_IHDR)
  358.       png_error(png_ptr, "Out of place IHDR");
  359.    /* check the length */
  360.    if (length != 13)
  361.       png_error(png_ptr, "Invalid IHDR chunk");
  362.    png_ptr->mode |= PNG_HAVE_IHDR;
  363.    png_crc_read(png_ptr, buf, 13);
  364.    png_crc_finish(png_ptr, 0);
  365.    width = png_get_uint_31(png_ptr, buf);
  366.    height = png_get_uint_31(png_ptr, buf + 4);
  367.    bit_depth = buf[8];
  368.    color_type = buf[9];
  369.    compression_type = buf[10];
  370.    filter_type = buf[11];
  371.    interlace_type = buf[12];
  372.    /* set internal variables */
  373.    png_ptr->width = width;
  374.    png_ptr->height = height;
  375.    png_ptr->bit_depth = (png_byte)bit_depth;
  376.    png_ptr->interlaced = (png_byte)interlace_type;
  377.    png_ptr->color_type = (png_byte)color_type;
  378. #if defined(PNG_MNG_FEATURES_SUPPORTED)
  379.    png_ptr->filter_type = (png_byte)filter_type;
  380. #endif
  381.    png_ptr->compression_type = (png_byte)compression_type;
  382.    /* find number of channels */
  383.    switch (png_ptr->color_type)
  384.    {
  385.       case PNG_COLOR_TYPE_GRAY:
  386.       case PNG_COLOR_TYPE_PALETTE:
  387.          png_ptr->channels = 1;
  388.          break;
  389.       case PNG_COLOR_TYPE_RGB:
  390.          png_ptr->channels = 3;
  391.          break;
  392.       case PNG_COLOR_TYPE_GRAY_ALPHA:
  393.          png_ptr->channels = 2;
  394.          break;
  395.       case PNG_COLOR_TYPE_RGB_ALPHA:
  396.          png_ptr->channels = 4;
  397.          break;
  398.    }
  399.    /* set up other useful info */
  400.    png_ptr->pixel_depth = (png_byte)(png_ptr->bit_depth *
  401.    png_ptr->channels);
  402.    png_ptr->rowbytes = PNG_ROWBYTES(png_ptr->pixel_depth, png_ptr->width);
  403.    png_debug1(3, "bit_depth = %d", png_ptr->bit_depth);
  404.    png_debug1(3, "channels = %d", png_ptr->channels);
  405.    png_debug1(3, "rowbytes = %lu", png_ptr->rowbytes);
  406.    png_set_IHDR(png_ptr, info_ptr, width, height, bit_depth,
  407.       color_type, interlace_type, compression_type, filter_type);
  408. }
  409. /* read and check the palette */
  410. void /* PRIVATE */
  411. png_handle_PLTE(png_structp png_ptr, png_infop info_ptr, png_uint_32 length)
  412. {
  413.    png_color palette[PNG_MAX_PALETTE_LENGTH];
  414.    int num, i;
  415. #ifndef PNG_NO_POINTER_INDEXING
  416.    png_colorp pal_ptr;
  417. #endif
  418.    png_debug(1, "in png_handle_PLTE");
  419.    if (!(png_ptr->mode & PNG_HAVE_IHDR))
  420.       png_error(png_ptr, "Missing IHDR before PLTE");
  421.    else if (png_ptr->mode & PNG_HAVE_IDAT)
  422.    {
  423.       png_warning(png_ptr, "Invalid PLTE after IDAT");
  424.       png_crc_finish(png_ptr, length);
  425.       return;
  426.    }
  427.    else if (png_ptr->mode & PNG_HAVE_PLTE)
  428.       png_error(png_ptr, "Duplicate PLTE chunk");
  429.    png_ptr->mode |= PNG_HAVE_PLTE;
  430.    if (!(png_ptr->color_type&PNG_COLOR_MASK_COLOR))
  431.    {
  432.       png_warning(png_ptr,
  433.         "Ignoring PLTE chunk in grayscale PNG");
  434.       png_crc_finish(png_ptr, length);
  435.       return;
  436.    }
  437. #if !defined(PNG_READ_OPT_PLTE_SUPPORTED)
  438.    if (png_ptr->color_type != PNG_COLOR_TYPE_PALETTE)
  439.    {
  440.       png_crc_finish(png_ptr, length);
  441.       return;
  442.    }
  443. #endif
  444.    if (length > 3*PNG_MAX_PALETTE_LENGTH || length % 3)
  445.    {
  446.       if (png_ptr->color_type != PNG_COLOR_TYPE_PALETTE)
  447.       {
  448.          png_warning(png_ptr, "Invalid palette chunk");
  449.          png_crc_finish(png_ptr, length);
  450.          return;
  451.       }
  452.       else
  453.       {
  454.          png_error(png_ptr, "Invalid palette chunk");
  455.       }
  456.    }
  457.    num = (int)length / 3;
  458. #ifndef PNG_NO_POINTER_INDEXING
  459.    for (i = 0, pal_ptr = palette; i < num; i++, pal_ptr++)
  460.    {
  461.       png_byte buf[3];
  462.       png_crc_read(png_ptr, buf, 3);
  463.       pal_ptr->red = buf[0];
  464.       pal_ptr->green = buf[1];
  465.       pal_ptr->blue = buf[2];
  466.    }
  467. #else
  468.    for (i = 0; i < num; i++)
  469.    {
  470.       png_byte buf[3];
  471.       png_crc_read(png_ptr, buf, 3);
  472.       /* don't depend upon png_color being any order */
  473.       palette[i].red = buf[0];
  474.       palette[i].green = buf[1];
  475.       palette[i].blue = buf[2];
  476.    }
  477. #endif
  478.    /* If we actually NEED the PLTE chunk (ie for a paletted image), we do
  479.       whatever the normal CRC configuration tells us.  However, if we
  480.       have an RGB image, the PLTE can be considered ancillary, so
  481.       we will act as though it is. */
  482. #if !defined(PNG_READ_OPT_PLTE_SUPPORTED)
  483.    if (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE)
  484. #endif
  485.    {
  486.       png_crc_finish(png_ptr, 0);
  487.    }
  488. #if !defined(PNG_READ_OPT_PLTE_SUPPORTED)
  489.    else if (png_crc_error(png_ptr))  /* Only if we have a CRC error */
  490.    {
  491.       /* If we don't want to use the data from an ancillary chunk,
  492.          we have two options: an error abort, or a warning and we
  493.          ignore the data in this chunk (which should be OK, since
  494.          it's considered ancillary for a RGB or RGBA image). */
  495.       if (!(png_ptr->flags & PNG_FLAG_CRC_ANCILLARY_USE))
  496.       {
  497.          if (png_ptr->flags & PNG_FLAG_CRC_ANCILLARY_NOWARN)
  498.          {
  499.             png_chunk_error(png_ptr, "CRC error");
  500.          }
  501.          else
  502.          {
  503.             png_chunk_warning(png_ptr, "CRC error");
  504.             return;
  505.          }
  506.       }
  507.       /* Otherwise, we (optionally) emit a warning and use the chunk. */
  508.       else if (!(png_ptr->flags & PNG_FLAG_CRC_ANCILLARY_NOWARN))
  509.       {
  510.          png_chunk_warning(png_ptr, "CRC error");
  511.       }
  512.    }
  513. #endif
  514.    png_set_PLTE(png_ptr, info_ptr, palette, num);
  515. #if defined(PNG_READ_tRNS_SUPPORTED)
  516.    if (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE)
  517.    {
  518.       if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_tRNS))
  519.       {
  520.          if (png_ptr->num_trans > (png_uint_16)num)
  521.          {
  522.             png_warning(png_ptr, "Truncating incorrect tRNS chunk length");
  523.             png_ptr->num_trans = (png_uint_16)num;
  524.          }
  525.          if (info_ptr->num_trans > (png_uint_16)num)
  526.          {
  527.             png_warning(png_ptr, "Truncating incorrect info tRNS chunk length");
  528.             info_ptr->num_trans = (png_uint_16)num;
  529.          }
  530.       }
  531.    }
  532. #endif
  533. }
  534. void /* PRIVATE */
  535. png_handle_IEND(png_structp png_ptr, png_infop info_ptr, png_uint_32 length)
  536. {
  537.    png_debug(1, "in png_handle_IEND");
  538.    if (!(png_ptr->mode & PNG_HAVE_IHDR) || !(png_ptr->mode & PNG_HAVE_IDAT))
  539.    {
  540.       png_error(png_ptr, "No image in file");
  541.    }
  542.    png_ptr->mode |= (PNG_AFTER_IDAT | PNG_HAVE_IEND);
  543.    if (length != 0)
  544.    {
  545.       png_warning(png_ptr, "Incorrect IEND chunk length");
  546.    }
  547.    png_crc_finish(png_ptr, length);
  548.    info_ptr = info_ptr; /* quiet compiler warnings about unused info_ptr */
  549. }
  550. #if defined(PNG_READ_gAMA_SUPPORTED)
  551. void /* PRIVATE */
  552. png_handle_gAMA(png_structp png_ptr, png_infop info_ptr, png_uint_32 length)
  553. {
  554.    png_fixed_point igamma;
  555. #ifdef PNG_FLOATING_POINT_SUPPORTED
  556.    float file_gamma;
  557. #endif
  558.    png_byte buf[4];
  559.    png_debug(1, "in png_handle_gAMA");
  560.    if (!(png_ptr->mode & PNG_HAVE_IHDR))
  561.       png_error(png_ptr, "Missing IHDR before gAMA");
  562.    else if (png_ptr->mode & PNG_HAVE_IDAT)
  563.    {
  564.       png_warning(png_ptr, "Invalid gAMA after IDAT");
  565.       png_crc_finish(png_ptr, length);
  566.       return;
  567.    }
  568.    else if (png_ptr->mode & PNG_HAVE_PLTE)
  569.       /* Should be an error, but we can cope with it */
  570.       png_warning(png_ptr, "Out of place gAMA chunk");
  571.    if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_gAMA)
  572. #if defined(PNG_READ_sRGB_SUPPORTED)
  573.       && !(info_ptr->valid & PNG_INFO_sRGB)
  574. #endif
  575.       )
  576.    {
  577.       png_warning(png_ptr, "Duplicate gAMA chunk");
  578.       png_crc_finish(png_ptr, length);
  579.       return;
  580.    }
  581.    if (length != 4)
  582.    {
  583.       png_warning(png_ptr, "Incorrect gAMA chunk length");
  584.       png_crc_finish(png_ptr, length);
  585.       return;
  586.    }
  587.    png_crc_read(png_ptr, buf, 4);
  588.    if (png_crc_finish(png_ptr, 0))
  589.       return;
  590.    igamma = (png_fixed_point)png_get_uint_32(buf);
  591.    /* check for zero gamma */
  592.    if (igamma == 0)
  593.       {
  594.          png_warning(png_ptr,
  595.            "Ignoring gAMA chunk with gamma=0");
  596.          return;
  597.       }
  598. #if defined(PNG_READ_sRGB_SUPPORTED)
  599.    if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_sRGB))
  600.       if (PNG_OUT_OF_RANGE(igamma, 45500L, 500))
  601.       {
  602.          png_warning(png_ptr,
  603.            "Ignoring incorrect gAMA value when sRGB is also present");
  604. #ifndef PNG_NO_CONSOLE_IO
  605.          fprintf(stderr, "gamma = (%d/100000)", (int)igamma);
  606. #endif
  607.          return;
  608.       }
  609. #endif /* PNG_READ_sRGB_SUPPORTED */
  610. #ifdef PNG_FLOATING_POINT_SUPPORTED
  611.    file_gamma = (float)igamma / (float)100000.0;
  612. #  ifdef PNG_READ_GAMMA_SUPPORTED
  613.      png_ptr->gamma = file_gamma;
  614. #  endif
  615.      png_set_gAMA(png_ptr, info_ptr, file_gamma);
  616. #endif
  617. #ifdef PNG_FIXED_POINT_SUPPORTED
  618.    png_set_gAMA_fixed(png_ptr, info_ptr, igamma);
  619. #endif
  620. }
  621. #endif
  622. #if defined(PNG_READ_sBIT_SUPPORTED)
  623. void /* PRIVATE */
  624. png_handle_sBIT(png_structp png_ptr, png_infop info_ptr, png_uint_32 length)
  625. {
  626.    png_size_t truelen;
  627.    png_byte buf[4];
  628.    png_debug(1, "in png_handle_sBIT");
  629.    buf[0] = buf[1] = buf[2] = buf[3] = 0;
  630.    if (!(png_ptr->mode & PNG_HAVE_IHDR))
  631.       png_error(png_ptr, "Missing IHDR before sBIT");
  632.    else if (png_ptr->mode & PNG_HAVE_IDAT)
  633.    {
  634.       png_warning(png_ptr, "Invalid sBIT after IDAT");
  635.       png_crc_finish(png_ptr, length);
  636.       return;
  637.    }
  638.    else if (png_ptr->mode & PNG_HAVE_PLTE)
  639.    {
  640.       /* Should be an error, but we can cope with it */
  641.       png_warning(png_ptr, "Out of place sBIT chunk");
  642.    }
  643.    if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_sBIT))
  644.    {
  645.       png_warning(png_ptr, "Duplicate sBIT chunk");
  646.       png_crc_finish(png_ptr, length);
  647.       return;
  648.    }
  649.    if (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE)
  650.       truelen = 3;
  651.    else
  652.       truelen = (png_size_t)png_ptr->channels;
  653.    if (length != truelen || length > 4)
  654.    {
  655.       png_warning(png_ptr, "Incorrect sBIT chunk length");
  656.       png_crc_finish(png_ptr, length);
  657.       return;
  658.    }
  659.    png_crc_read(png_ptr, buf, truelen);
  660.    if (png_crc_finish(png_ptr, 0))
  661.       return;
  662.    if (png_ptr->color_type & PNG_COLOR_MASK_COLOR)
  663.    {
  664.       png_ptr->sig_bit.red = buf[0];
  665.       png_ptr->sig_bit.green = buf[1];
  666.       png_ptr->sig_bit.blue = buf[2];
  667.       png_ptr->sig_bit.alpha = buf[3];
  668.    }
  669.    else
  670.    {
  671.       png_ptr->sig_bit.gray = buf[0];
  672.       png_ptr->sig_bit.red = buf[0];
  673.       png_ptr->sig_bit.green = buf[0];
  674.       png_ptr->sig_bit.blue = buf[0];
  675.       png_ptr->sig_bit.alpha = buf[1];
  676.    }
  677.    png_set_sBIT(png_ptr, info_ptr, &(png_ptr->sig_bit));
  678. }
  679. #endif
  680. #if defined(PNG_READ_cHRM_SUPPORTED)
  681. void /* PRIVATE */
  682. png_handle_cHRM(png_structp png_ptr, png_infop info_ptr, png_uint_32 length)
  683. {
  684.    png_byte buf[32];
  685. #ifdef PNG_FLOATING_POINT_SUPPORTED
  686.    float white_x, white_y, red_x, red_y, green_x, green_y, blue_x, blue_y;
  687. #endif
  688.    png_fixed_point int_x_white, int_y_white, int_x_red, int_y_red, int_x_green,
  689.       int_y_green, int_x_blue, int_y_blue;
  690.    png_uint_32 uint_x, uint_y;
  691.    png_debug(1, "in png_handle_cHRM");
  692.    if (!(png_ptr->mode & PNG_HAVE_IHDR))
  693.       png_error(png_ptr, "Missing IHDR before cHRM");
  694.    else if (png_ptr->mode & PNG_HAVE_IDAT)
  695.    {
  696.       png_warning(png_ptr, "Invalid cHRM after IDAT");
  697.       png_crc_finish(png_ptr, length);
  698.       return;
  699.    }
  700.    else if (png_ptr->mode & PNG_HAVE_PLTE)
  701.       /* Should be an error, but we can cope with it */
  702.       png_warning(png_ptr, "Missing PLTE before cHRM");
  703.    if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_cHRM)
  704. #if defined(PNG_READ_sRGB_SUPPORTED)
  705.       && !(info_ptr->valid & PNG_INFO_sRGB)
  706. #endif
  707.       )
  708.    {
  709.       png_warning(png_ptr, "Duplicate cHRM chunk");
  710.       png_crc_finish(png_ptr, length);
  711.       return;
  712.    }
  713.    if (length != 32)
  714.    {
  715.       png_warning(png_ptr, "Incorrect cHRM chunk length");
  716.       png_crc_finish(png_ptr, length);
  717.       return;
  718.    }
  719.    png_crc_read(png_ptr, buf, 32);
  720.    if (png_crc_finish(png_ptr, 0))
  721.       return;
  722.    uint_x = png_get_uint_32(buf);
  723.    uint_y = png_get_uint_32(buf + 4);
  724.    int_x_white = (png_fixed_point)uint_x;
  725.    int_y_white = (png_fixed_point)uint_y;
  726.    uint_x = png_get_uint_32(buf + 8);
  727.    uint_y = png_get_uint_32(buf + 12);
  728.    int_x_red = (png_fixed_point)uint_x;
  729.    int_y_red = (png_fixed_point)uint_y;
  730.    uint_x = png_get_uint_32(buf + 16);
  731.    uint_y = png_get_uint_32(buf + 20);
  732.    int_x_green = (png_fixed_point)uint_x;
  733.    int_y_green = (png_fixed_point)uint_y;
  734.    uint_x = png_get_uint_32(buf + 24);
  735.    uint_y = png_get_uint_32(buf + 28);
  736.    int_x_blue = (png_fixed_point)uint_x;
  737.    int_y_blue = (png_fixed_point)uint_y;
  738. #ifdef PNG_FLOATING_POINT_SUPPORTED
  739.    white_x = (float)int_x_white / (float)100000.0;
  740.    white_y = (float)int_y_white / (float)100000.0;
  741.    red_x   = (float)int_x_red   / (float)100000.0;
  742.    red_y   = (float)int_y_red   / (float)100000.0;
  743.    green_x = (float)int_x_green / (float)100000.0;
  744.    green_y = (float)int_y_green / (float)100000.0;
  745.    blue_x  = (float)int_x_blue  / (float)100000.0;
  746.    blue_y  = (float)int_y_blue  / (float)100000.0;
  747. #endif
  748. #if defined(PNG_READ_sRGB_SUPPORTED)
  749.    if ((info_ptr != NULL) && (info_ptr->valid & PNG_INFO_sRGB))
  750.       {
  751.       if (PNG_OUT_OF_RANGE(int_x_white, 31270,  1000) ||
  752.           PNG_OUT_OF_RANGE(int_y_white, 32900,  1000) ||
  753.           PNG_OUT_OF_RANGE(int_x_red,   64000L, 1000) ||
  754.           PNG_OUT_OF_RANGE(int_y_red,   33000,  1000) ||
  755.           PNG_OUT_OF_RANGE(int_x_green, 30000,  1000) ||
  756.           PNG_OUT_OF_RANGE(int_y_green, 60000L, 1000) ||
  757.           PNG_OUT_OF_RANGE(int_x_blue,  15000,  1000) ||
  758.           PNG_OUT_OF_RANGE(int_y_blue,   6000,  1000))
  759.          {
  760.             png_warning(png_ptr,
  761.               "Ignoring incorrect cHRM value when sRGB is also present");
  762. #ifndef PNG_NO_CONSOLE_IO
  763. #ifdef PNG_FLOATING_POINT_SUPPORTED
  764.             fprintf(stderr, "wx=%f, wy=%f, rx=%f, ry=%fn",
  765.                white_x, white_y, red_x, red_y);
  766.             fprintf(stderr, "gx=%f, gy=%f, bx=%f, by=%fn",
  767.                green_x, green_y, blue_x, blue_y);
  768. #else
  769.             fprintf(stderr, "wx=%ld, wy=%ld, rx=%ld, ry=%ldn",
  770.                int_x_white, int_y_white, int_x_red, int_y_red);
  771.             fprintf(stderr, "gx=%ld, gy=%ld, bx=%ld, by=%ldn",
  772.                int_x_green, int_y_green, int_x_blue, int_y_blue);
  773. #endif
  774. #endif /* PNG_NO_CONSOLE_IO */
  775.          }
  776.          return;
  777.       }
  778. #endif /* PNG_READ_sRGB_SUPPORTED */
  779. #ifdef PNG_FLOATING_POINT_SUPPORTED
  780.    png_set_cHRM(png_ptr, info_ptr,
  781.       white_x, white_y, red_x, red_y, green_x, green_y, blue_x, blue_y);
  782. #endif
  783. #ifdef PNG_FIXED_POINT_SUPPORTED
  784.    png_set_cHRM_fixed(png_ptr, info_ptr,
  785.       int_x_white, int_y_white, int_x_red, int_y_red, int_x_green,
  786.       int_y_green, int_x_blue, int_y_blue);
  787. #endif
  788. }
  789. #endif
  790. #if defined(PNG_READ_sRGB_SUPPORTED)
  791. void /* PRIVATE */
  792. png_handle_sRGB(png_structp png_ptr, png_infop info_ptr, png_uint_32 length)
  793. {
  794.    int intent;
  795.    png_byte buf[1];
  796.    png_debug(1, "in png_handle_sRGB");
  797.    if (!(png_ptr->mode & PNG_HAVE_IHDR))
  798.       png_error(png_ptr, "Missing IHDR before sRGB");
  799.    else if (png_ptr->mode & PNG_HAVE_IDAT)
  800.    {
  801.       png_warning(png_ptr, "Invalid sRGB after IDAT");
  802.       png_crc_finish(png_ptr, length);
  803.       return;
  804.    }
  805.    else if (png_ptr->mode & PNG_HAVE_PLTE)
  806.       /* Should be an error, but we can cope with it */
  807.       png_warning(png_ptr, "Out of place sRGB chunk");
  808.    if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_sRGB))
  809.    {
  810.       png_warning(png_ptr, "Duplicate sRGB chunk");
  811.       png_crc_finish(png_ptr, length);
  812.       return;
  813.    }
  814.    if (length != 1)
  815.    {
  816.       png_warning(png_ptr, "Incorrect sRGB chunk length");
  817.       png_crc_finish(png_ptr, length);
  818.       return;
  819.    }
  820.    png_crc_read(png_ptr, buf, 1);
  821.    if (png_crc_finish(png_ptr, 0))
  822.       return;
  823.    intent = buf[0];
  824.    /* check for bad intent */
  825.    if (intent >= PNG_sRGB_INTENT_LAST)
  826.    {
  827.       png_warning(png_ptr, "Unknown sRGB intent");
  828.       return;
  829.    }
  830. #if defined(PNG_READ_gAMA_SUPPORTED) && defined(PNG_READ_GAMMA_SUPPORTED)
  831.    if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_gAMA))
  832.    {
  833.    png_fixed_point igamma;
  834. #ifdef PNG_FIXED_POINT_SUPPORTED
  835.       igamma=info_ptr->int_gamma;
  836. #else
  837. #  ifdef PNG_FLOATING_POINT_SUPPORTED
  838.       igamma=(png_fixed_point)(info_ptr->gamma * 100000.);
  839. #  endif
  840. #endif
  841.       if (PNG_OUT_OF_RANGE(igamma, 45500L, 500))
  842.       {
  843.          png_warning(png_ptr,
  844.            "Ignoring incorrect gAMA value when sRGB is also present");
  845. #ifndef PNG_NO_CONSOLE_IO
  846. #  ifdef PNG_FIXED_POINT_SUPPORTED
  847.          fprintf(stderr, "incorrect gamma=(%d/100000)n",
  848.             (int)png_ptr->int_gamma);
  849. #  else
  850. #    ifdef PNG_FLOATING_POINT_SUPPORTED
  851.          fprintf(stderr, "incorrect gamma=%fn", png_ptr->gamma);
  852. #    endif
  853. #  endif
  854. #endif
  855.       }
  856.    }
  857. #endif /* PNG_READ_gAMA_SUPPORTED */
  858. #ifdef PNG_READ_cHRM_SUPPORTED
  859. #ifdef PNG_FIXED_POINT_SUPPORTED
  860.    if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_cHRM))
  861.       if (PNG_OUT_OF_RANGE(info_ptr->int_x_white, 31270,  1000) ||
  862.           PNG_OUT_OF_RANGE(info_ptr->int_y_white, 32900,  1000) ||
  863.           PNG_OUT_OF_RANGE(info_ptr->int_x_red,   64000L, 1000) ||
  864.           PNG_OUT_OF_RANGE(info_ptr->int_y_red,   33000,  1000) ||
  865.           PNG_OUT_OF_RANGE(info_ptr->int_x_green, 30000,  1000) ||
  866.           PNG_OUT_OF_RANGE(info_ptr->int_y_green, 60000L, 1000) ||
  867.           PNG_OUT_OF_RANGE(info_ptr->int_x_blue,  15000,  1000) ||
  868.           PNG_OUT_OF_RANGE(info_ptr->int_y_blue,   6000,  1000))
  869.          {
  870.             png_warning(png_ptr,
  871.               "Ignoring incorrect cHRM value when sRGB is also present");
  872.          }
  873. #endif /* PNG_FIXED_POINT_SUPPORTED */
  874. #endif /* PNG_READ_cHRM_SUPPORTED */
  875.    png_set_sRGB_gAMA_and_cHRM(png_ptr, info_ptr, intent);
  876. }
  877. #endif /* PNG_READ_sRGB_SUPPORTED */
  878. #if defined(PNG_READ_iCCP_SUPPORTED)
  879. void /* PRIVATE */
  880. png_handle_iCCP(png_structp png_ptr, png_infop info_ptr, png_uint_32 length)
  881. /* Note: this does not properly handle chunks that are > 64K under DOS */
  882. {
  883.    png_byte compression_type;
  884.    png_bytep pC;
  885.    png_charp profile;
  886.    png_uint_32 skip = 0;
  887.    png_uint_32 profile_size, profile_length;
  888.    png_size_t slength, prefix_length, data_length;
  889.    png_debug(1, "in png_handle_iCCP");
  890.    if (!(png_ptr->mode & PNG_HAVE_IHDR))
  891.       png_error(png_ptr, "Missing IHDR before iCCP");
  892.    else if (png_ptr->mode & PNG_HAVE_IDAT)
  893.    {
  894.       png_warning(png_ptr, "Invalid iCCP after IDAT");
  895.       png_crc_finish(png_ptr, length);
  896.       return;
  897.    }
  898.    else if (png_ptr->mode & PNG_HAVE_PLTE)
  899.       /* Should be an error, but we can cope with it */
  900.       png_warning(png_ptr, "Out of place iCCP chunk");
  901.    if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_iCCP))
  902.    {
  903.       png_warning(png_ptr, "Duplicate iCCP chunk");
  904.       png_crc_finish(png_ptr, length);
  905.       return;
  906.    }
  907. #ifdef PNG_MAX_MALLOC_64K
  908.    if (length > (png_uint_32)65535L)
  909.    {
  910.       png_warning(png_ptr, "iCCP chunk too large to fit in memory");
  911.       skip = length - (png_uint_32)65535L;
  912.       length = (png_uint_32)65535L;
  913.    }
  914. #endif
  915.    png_free(png_ptr, png_ptr->chunkdata);
  916.    png_ptr->chunkdata = (png_charp)png_malloc(png_ptr, length + 1);
  917.    slength = (png_size_t)length;
  918.    png_crc_read(png_ptr, (png_bytep)png_ptr->chunkdata, slength);
  919.    if (png_crc_finish(png_ptr, skip))
  920.    {
  921.       png_free(png_ptr, png_ptr->chunkdata);
  922.       png_ptr->chunkdata = NULL;
  923.       return;
  924.    }
  925.    png_ptr->chunkdata[slength] = 0x00;
  926.    for (profile = png_ptr->chunkdata; *profile; profile++)
  927.       /* empty loop to find end of name */ ;
  928.    ++profile;
  929.    /* there should be at least one zero (the compression type byte)
  930.       following the separator, and we should be on it  */
  931.    if ( profile >= png_ptr->chunkdata + slength - 1)
  932.    {
  933.       png_free(png_ptr, png_ptr->chunkdata);
  934.       png_ptr->chunkdata = NULL;
  935.       png_warning(png_ptr, "Malformed iCCP chunk");
  936.       return;
  937.    }
  938.    /* compression_type should always be zero */
  939.    compression_type = *profile++;
  940.    if (compression_type)
  941.    {
  942.       png_warning(png_ptr, "Ignoring nonzero compression type in iCCP chunk");
  943.       compression_type = 0x00;  /* Reset it to zero (libpng-1.0.6 through 1.0.8
  944.                                  wrote nonzero) */
  945.    }
  946.    prefix_length = profile - png_ptr->chunkdata;
  947.    png_decompress_chunk(png_ptr, compression_type,
  948.      slength, prefix_length, &data_length);
  949.    profile_length = (png_uint_32)(data_length - prefix_length);
  950.    if ( prefix_length > data_length || profile_length < 4)
  951.    {
  952.       png_free(png_ptr, png_ptr->chunkdata);
  953.       png_ptr->chunkdata = NULL;
  954.       png_warning(png_ptr, "Profile size field missing from iCCP chunk");
  955.       return;
  956.    }
  957.    /* Check the profile_size recorded in the first 32 bits of the ICC profile */
  958.    pC = (png_bytep)(png_ptr->chunkdata + prefix_length);
  959.    profile_size = ((*(pC    ))<<24) |
  960.                   ((*(pC + 1))<<16) |
  961.                   ((*(pC + 2))<< 8) |
  962.                   ((*(pC + 3))    );
  963.    if (profile_size < profile_length)
  964.       profile_length = profile_size;
  965.    if (profile_size > profile_length)
  966.    {
  967.       png_free(png_ptr, png_ptr->chunkdata);
  968.       png_ptr->chunkdata = NULL;
  969.       png_warning(png_ptr, "Ignoring truncated iCCP profile.");
  970.       return;
  971.    }
  972.    png_set_iCCP(png_ptr, info_ptr, png_ptr->chunkdata,
  973.      compression_type, png_ptr->chunkdata + prefix_length, profile_length);
  974.    png_free(png_ptr, png_ptr->chunkdata);
  975.    png_ptr->chunkdata = NULL;
  976. }
  977. #endif /* PNG_READ_iCCP_SUPPORTED */
  978. #if defined(PNG_READ_sPLT_SUPPORTED)
  979. void /* PRIVATE */
  980. png_handle_sPLT(png_structp png_ptr, png_infop info_ptr, png_uint_32 length)
  981. /* Note: this does not properly handle chunks that are > 64K under DOS */
  982. {
  983.    png_bytep entry_start;
  984.    png_sPLT_t new_palette;
  985. #ifdef PNG_NO_POINTER_INDEXING
  986.    png_sPLT_entryp pp;
  987. #endif
  988.    int data_length, entry_size, i;
  989.    png_uint_32 skip = 0;
  990.    png_size_t slength;
  991.    png_debug(1, "in png_handle_sPLT");
  992.    if (!(png_ptr->mode & PNG_HAVE_IHDR))
  993.       png_error(png_ptr, "Missing IHDR before sPLT");
  994.    else if (png_ptr->mode & PNG_HAVE_IDAT)
  995.    {
  996.       png_warning(png_ptr, "Invalid sPLT after IDAT");
  997.       png_crc_finish(png_ptr, length);
  998.       return;
  999.    }
  1000. #ifdef PNG_MAX_MALLOC_64K
  1001.    if (length > (png_uint_32)65535L)
  1002.    {
  1003.       png_warning(png_ptr, "sPLT chunk too large to fit in memory");
  1004.       skip = length - (png_uint_32)65535L;
  1005.       length = (png_uint_32)65535L;
  1006.    }
  1007. #endif
  1008.    png_free(png_ptr, png_ptr->chunkdata);
  1009.    png_ptr->chunkdata = (png_charp)png_malloc(png_ptr, length + 1);
  1010.    slength = (png_size_t)length;
  1011.    png_crc_read(png_ptr, (png_bytep)png_ptr->chunkdata, slength);
  1012.    if (png_crc_finish(png_ptr, skip))
  1013.    {
  1014.       png_free(png_ptr, png_ptr->chunkdata);
  1015.       png_ptr->chunkdata = NULL;
  1016.       return;
  1017.    }
  1018.    png_ptr->chunkdata[slength] = 0x00;
  1019.    for (entry_start = (png_bytep)png_ptr->chunkdata; *entry_start; entry_start++)
  1020.       /* empty loop to find end of name */ ;
  1021.    ++entry_start;
  1022.    /* a sample depth should follow the separator, and we should be on it  */
  1023.    if (entry_start > (png_bytep)png_ptr->chunkdata + slength - 2)
  1024.    {
  1025.       png_free(png_ptr, png_ptr->chunkdata);
  1026.       png_ptr->chunkdata = NULL;
  1027.       png_warning(png_ptr, "malformed sPLT chunk");
  1028.       return;
  1029.    }
  1030.    new_palette.depth = *entry_start++;
  1031.    entry_size = (new_palette.depth == 8 ? 6 : 10);
  1032.    data_length = (png_uint_32)(slength - (entry_start - (png_bytep)png_ptr->chunkdata));
  1033.    /* integrity-check the data length */
  1034.    if (data_length % entry_size)
  1035.    {
  1036.       png_free(png_ptr, png_ptr->chunkdata);
  1037.       png_ptr->chunkdata = NULL;
  1038.       png_warning(png_ptr, "sPLT chunk has bad length");
  1039.       return;
  1040.    }
  1041.    new_palette.nentries = (png_int_32) ( data_length / entry_size);
  1042.    if ((png_uint_32) new_palette.nentries >
  1043.        (png_uint_32) (PNG_SIZE_MAX / png_sizeof(png_sPLT_entry)))
  1044.    {
  1045.        png_warning(png_ptr, "sPLT chunk too long");
  1046.        return;
  1047.    }
  1048.    new_palette.entries = (png_sPLT_entryp)png_malloc_warn(
  1049.        png_ptr, new_palette.nentries * png_sizeof(png_sPLT_entry));
  1050.    if (new_palette.entries == NULL)
  1051.    {
  1052.        png_warning(png_ptr, "sPLT chunk requires too much memory");
  1053.        return;
  1054.    }
  1055. #ifndef PNG_NO_POINTER_INDEXING
  1056.    for (i = 0; i < new_palette.nentries; i++)
  1057.    {
  1058.       png_sPLT_entryp pp = new_palette.entries + i;
  1059.       if (new_palette.depth == 8)
  1060.       {
  1061.           pp->red = *entry_start++;
  1062.           pp->green = *entry_start++;
  1063.           pp->blue = *entry_start++;
  1064.           pp->alpha = *entry_start++;
  1065.       }
  1066.       else
  1067.       {
  1068.           pp->red   = png_get_uint_16(entry_start); entry_start += 2;
  1069.           pp->green = png_get_uint_16(entry_start); entry_start += 2;
  1070.           pp->blue  = png_get_uint_16(entry_start); entry_start += 2;
  1071.           pp->alpha = png_get_uint_16(entry_start); entry_start += 2;
  1072.       }
  1073.       pp->frequency = png_get_uint_16(entry_start); entry_start += 2;
  1074.    }
  1075. #else
  1076.    pp = new_palette.entries;
  1077.    for (i = 0; i < new_palette.nentries; i++)
  1078.    {
  1079.       if (new_palette.depth == 8)
  1080.       {
  1081.           pp[i].red   = *entry_start++;
  1082.           pp[i].green = *entry_start++;
  1083.           pp[i].blue  = *entry_start++;
  1084.           pp[i].alpha = *entry_start++;
  1085.       }
  1086.       else
  1087.       {
  1088.           pp[i].red   = png_get_uint_16(entry_start); entry_start += 2;
  1089.           pp[i].green = png_get_uint_16(entry_start); entry_start += 2;
  1090.           pp[i].blue  = png_get_uint_16(entry_start); entry_start += 2;
  1091.           pp[i].alpha = png_get_uint_16(entry_start); entry_start += 2;
  1092.       }
  1093.       pp->frequency = png_get_uint_16(entry_start); entry_start += 2;
  1094.    }
  1095. #endif
  1096.    /* discard all chunk data except the name and stash that */
  1097.    new_palette.name = png_ptr->chunkdata;
  1098.    png_set_sPLT(png_ptr, info_ptr, &new_palette, 1);
  1099.    png_free(png_ptr, png_ptr->chunkdata);
  1100.    png_ptr->chunkdata = NULL;
  1101.    png_free(png_ptr, new_palette.entries);
  1102. }
  1103. #endif /* PNG_READ_sPLT_SUPPORTED */
  1104. #if defined(PNG_READ_tRNS_SUPPORTED)
  1105. void /* PRIVATE */
  1106. png_handle_tRNS(png_structp png_ptr, png_infop info_ptr, png_uint_32 length)
  1107. {
  1108.    png_byte readbuf[PNG_MAX_PALETTE_LENGTH];
  1109.    png_debug(1, "in png_handle_tRNS");
  1110.    if (!(png_ptr->mode & PNG_HAVE_IHDR))
  1111.       png_error(png_ptr, "Missing IHDR before tRNS");
  1112.    else if (png_ptr->mode & PNG_HAVE_IDAT)
  1113.    {
  1114.       png_warning(png_ptr, "Invalid tRNS after IDAT");
  1115.       png_crc_finish(png_ptr, length);
  1116.       return;
  1117.    }
  1118.    else if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_tRNS))
  1119.    {
  1120.       png_warning(png_ptr, "Duplicate tRNS chunk");
  1121.       png_crc_finish(png_ptr, length);
  1122.       return;
  1123.    }
  1124.    if (png_ptr->color_type == PNG_COLOR_TYPE_GRAY)
  1125.    {
  1126.       png_byte buf[2];
  1127.       if (length != 2)
  1128.       {
  1129.          png_warning(png_ptr, "Incorrect tRNS chunk length");
  1130.          png_crc_finish(png_ptr, length);
  1131.          return;
  1132.       }
  1133.       png_crc_read(png_ptr, buf, 2);
  1134.       png_ptr->num_trans = 1;
  1135.       png_ptr->trans_values.gray = png_get_uint_16(buf);
  1136.    }
  1137.    else if (png_ptr->color_type == PNG_COLOR_TYPE_RGB)
  1138.    {
  1139.       png_byte buf[6];
  1140.       if (length != 6)
  1141.       {
  1142.          png_warning(png_ptr, "Incorrect tRNS chunk length");
  1143.          png_crc_finish(png_ptr, length);
  1144.          return;
  1145.       }
  1146.       png_crc_read(png_ptr, buf, (png_size_t)length);
  1147.       png_ptr->num_trans = 1;
  1148.       png_ptr->trans_values.red = png_get_uint_16(buf);
  1149.       png_ptr->trans_values.green = png_get_uint_16(buf + 2);
  1150.       png_ptr->trans_values.blue = png_get_uint_16(buf + 4);
  1151.    }
  1152.    else if (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE)
  1153.    {
  1154.       if (!(png_ptr->mode & PNG_HAVE_PLTE))
  1155.       {
  1156.          /* Should be an error, but we can cope with it. */
  1157.          png_warning(png_ptr, "Missing PLTE before tRNS");
  1158.       }
  1159.       if (length > (png_uint_32)png_ptr->num_palette ||
  1160.           length > PNG_MAX_PALETTE_LENGTH)
  1161.       {
  1162.          png_warning(png_ptr, "Incorrect tRNS chunk length");
  1163.          png_crc_finish(png_ptr, length);
  1164.          return;
  1165.       }
  1166.       if (length == 0)
  1167.       {
  1168.          png_warning(png_ptr, "Zero length tRNS chunk");
  1169.          png_crc_finish(png_ptr, length);
  1170.          return;
  1171.       }
  1172.       png_crc_read(png_ptr, readbuf, (png_size_t)length);
  1173.       png_ptr->num_trans = (png_uint_16)length;
  1174.    }
  1175.    else
  1176.    {
  1177.       png_warning(png_ptr, "tRNS chunk not allowed with alpha channel");
  1178.       png_crc_finish(png_ptr, length);
  1179.       return;
  1180.    }
  1181.    if (png_crc_finish(png_ptr, 0))
  1182.    {
  1183.       png_ptr->num_trans = 0;
  1184.       return;
  1185.    }
  1186.    png_set_tRNS(png_ptr, info_ptr, readbuf, png_ptr->num_trans,
  1187.       &(png_ptr->trans_values));
  1188. }
  1189. #endif
  1190. #if defined(PNG_READ_bKGD_SUPPORTED)
  1191. void /* PRIVATE */
  1192. png_handle_bKGD(png_structp png_ptr, png_infop info_ptr, png_uint_32 length)
  1193. {
  1194.    png_size_t truelen;
  1195.    png_byte buf[6];
  1196.    png_debug(1, "in png_handle_bKGD");
  1197.    if (!(png_ptr->mode & PNG_HAVE_IHDR))
  1198.       png_error(png_ptr, "Missing IHDR before bKGD");
  1199.    else if (png_ptr->mode & PNG_HAVE_IDAT)
  1200.    {
  1201.       png_warning(png_ptr, "Invalid bKGD after IDAT");
  1202.       png_crc_finish(png_ptr, length);
  1203.       return;
  1204.    }
  1205.    else if (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE &&
  1206.             !(png_ptr->mode & PNG_HAVE_PLTE))
  1207.    {
  1208.       png_warning(png_ptr, "Missing PLTE before bKGD");
  1209.       png_crc_finish(png_ptr, length);
  1210.       return;
  1211.    }
  1212.    else if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_bKGD))
  1213.    {
  1214.       png_warning(png_ptr, "Duplicate bKGD chunk");
  1215.       png_crc_finish(png_ptr, length);
  1216.       return;
  1217.    }
  1218.    if (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE)
  1219.       truelen = 1;
  1220.    else if (png_ptr->color_type & PNG_COLOR_MASK_COLOR)
  1221.       truelen = 6;
  1222.    else
  1223.       truelen = 2;
  1224.    if (length != truelen)
  1225.    {
  1226.       png_warning(png_ptr, "Incorrect bKGD chunk length");
  1227.       png_crc_finish(png_ptr, length);
  1228.       return;
  1229.    }
  1230.    png_crc_read(png_ptr, buf, truelen);
  1231.    if (png_crc_finish(png_ptr, 0))
  1232.       return;
  1233.    /* We convert the index value into RGB components so that we can allow
  1234.     * arbitrary RGB values for background when we have transparency, and
  1235.     * so it is easy to determine the RGB values of the background color
  1236.     * from the info_ptr struct. */
  1237.    if (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE)
  1238.    {
  1239.       png_ptr->background.index = buf[0];
  1240.       if (info_ptr && info_ptr->num_palette)
  1241.       {
  1242.           if (buf[0] >= info_ptr->num_palette)
  1243.           {
  1244.              png_warning(png_ptr, "Incorrect bKGD chunk index value");
  1245.              return;
  1246.           }
  1247.           png_ptr->background.red =
  1248.              (png_uint_16)png_ptr->palette[buf[0]].red;
  1249.           png_ptr->background.green =
  1250.              (png_uint_16)png_ptr->palette[buf[0]].green;
  1251.           png_ptr->background.blue =
  1252.              (png_uint_16)png_ptr->palette[buf[0]].blue;
  1253.       }
  1254.    }
  1255.    else if (!(png_ptr->color_type & PNG_COLOR_MASK_COLOR)) /* GRAY */
  1256.    {
  1257.       png_ptr->background.red =
  1258.       png_ptr->background.green =
  1259.       png_ptr->background.blue =
  1260.       png_ptr->background.gray = png_get_uint_16(buf);
  1261.    }
  1262.    else
  1263.    {
  1264.       png_ptr->background.red = png_get_uint_16(buf);
  1265.       png_ptr->background.green = png_get_uint_16(buf + 2);
  1266.       png_ptr->background.blue = png_get_uint_16(buf + 4);
  1267.    }
  1268.    png_set_bKGD(png_ptr, info_ptr, &(png_ptr->background));
  1269. }
  1270. #endif
  1271. #if defined(PNG_READ_hIST_SUPPORTED)
  1272. void /* PRIVATE */
  1273. png_handle_hIST(png_structp png_ptr, png_infop info_ptr, png_uint_32 length)
  1274. {
  1275.    unsigned int num, i;
  1276.    png_uint_16 readbuf[PNG_MAX_PALETTE_LENGTH];
  1277.    png_debug(1, "in png_handle_hIST");
  1278.    if (!(png_ptr->mode & PNG_HAVE_IHDR))
  1279.       png_error(png_ptr, "Missing IHDR before hIST");
  1280.    else if (png_ptr->mode & PNG_HAVE_IDAT)
  1281.    {
  1282.       png_warning(png_ptr, "Invalid hIST after IDAT");
  1283.       png_crc_finish(png_ptr, length);
  1284.       return;
  1285.    }
  1286.    else if (!(png_ptr->mode & PNG_HAVE_PLTE))
  1287.    {
  1288.       png_warning(png_ptr, "Missing PLTE before hIST");
  1289.       png_crc_finish(png_ptr, length);
  1290.       return;
  1291.    }
  1292.    else if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_hIST))
  1293.    {
  1294.       png_warning(png_ptr, "Duplicate hIST chunk");
  1295.       png_crc_finish(png_ptr, length);
  1296.       return;
  1297.    }
  1298.    num = length / 2 ;
  1299.    if (num != (unsigned int) png_ptr->num_palette || num >
  1300.       (unsigned int) PNG_MAX_PALETTE_LENGTH)
  1301.    {
  1302.       png_warning(png_ptr, "Incorrect hIST chunk length");
  1303.       png_crc_finish(png_ptr, length);
  1304.       return;
  1305.    }
  1306.    for (i = 0; i < num; i++)
  1307.    {
  1308.       png_byte buf[2];
  1309.       png_crc_read(png_ptr, buf, 2);
  1310.       readbuf[i] = png_get_uint_16(buf);
  1311.    }
  1312.    if (png_crc_finish(png_ptr, 0))
  1313.       return;
  1314.    png_set_hIST(png_ptr, info_ptr, readbuf);
  1315. }
  1316. #endif
  1317. #if defined(PNG_READ_pHYs_SUPPORTED)
  1318. void /* PRIVATE */
  1319. png_handle_pHYs(png_structp png_ptr, png_infop info_ptr, png_uint_32 length)
  1320. {
  1321.    png_byte buf[9];
  1322.    png_uint_32 res_x, res_y;
  1323.    int unit_type;
  1324.    png_debug(1, "in png_handle_pHYs");
  1325.    if (!(png_ptr->mode & PNG_HAVE_IHDR))
  1326.       png_error(png_ptr, "Missing IHDR before pHYs");
  1327.    else if (png_ptr->mode & PNG_HAVE_IDAT)
  1328.    {
  1329.       png_warning(png_ptr, "Invalid pHYs after IDAT");
  1330.       png_crc_finish(png_ptr, length);
  1331.       return;
  1332.    }
  1333.    else if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_pHYs))
  1334.    {
  1335.       png_warning(png_ptr, "Duplicate pHYs chunk");
  1336.       png_crc_finish(png_ptr, length);
  1337.       return;
  1338.    }
  1339.    if (length != 9)
  1340.    {
  1341.       png_warning(png_ptr, "Incorrect pHYs chunk length");
  1342.       png_crc_finish(png_ptr, length);
  1343.       return;
  1344.    }
  1345.    png_crc_read(png_ptr, buf, 9);
  1346.    if (png_crc_finish(png_ptr, 0))
  1347.       return;
  1348.    res_x = png_get_uint_32(buf);
  1349.    res_y = png_get_uint_32(buf + 4);
  1350.    unit_type = buf[8];
  1351.    png_set_pHYs(png_ptr, info_ptr, res_x, res_y, unit_type);
  1352. }
  1353. #endif
  1354. #if defined(PNG_READ_oFFs_SUPPORTED)
  1355. void /* PRIVATE */
  1356. png_handle_oFFs(png_structp png_ptr, png_infop info_ptr, png_uint_32 length)
  1357. {
  1358.    png_byte buf[9];
  1359.    png_int_32 offset_x, offset_y;
  1360.    int unit_type;
  1361.    png_debug(1, "in png_handle_oFFs");
  1362.    if (!(png_ptr->mode & PNG_HAVE_IHDR))
  1363.       png_error(png_ptr, "Missing IHDR before oFFs");
  1364.    else if (png_ptr->mode & PNG_HAVE_IDAT)
  1365.    {
  1366.       png_warning(png_ptr, "Invalid oFFs after IDAT");
  1367.       png_crc_finish(png_ptr, length);
  1368.       return;
  1369.    }
  1370.    else if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_oFFs))
  1371.    {
  1372.       png_warning(png_ptr, "Duplicate oFFs chunk");
  1373.       png_crc_finish(png_ptr, length);
  1374.       return;
  1375.    }
  1376.    if (length != 9)
  1377.    {
  1378.       png_warning(png_ptr, "Incorrect oFFs chunk length");
  1379.       png_crc_finish(png_ptr, length);
  1380.       return;
  1381.    }
  1382.    png_crc_read(png_ptr, buf, 9);
  1383.    if (png_crc_finish(png_ptr, 0))
  1384.       return;
  1385.    offset_x = png_get_int_32(buf);
  1386.    offset_y = png_get_int_32(buf + 4);
  1387.    unit_type = buf[8];
  1388.    png_set_oFFs(png_ptr, info_ptr, offset_x, offset_y, unit_type);
  1389. }
  1390. #endif
  1391. #if defined(PNG_READ_pCAL_SUPPORTED)
  1392. /* read the pCAL chunk (described in the PNG Extensions document) */
  1393. void /* PRIVATE */
  1394. png_handle_pCAL(png_structp png_ptr, png_infop info_ptr, png_uint_32 length)
  1395. {
  1396.    png_int_32 X0, X1;
  1397.    png_byte type, nparams;
  1398.    png_charp buf, units, endptr;
  1399.    png_charpp params;
  1400.    png_size_t slength;
  1401.    int i;
  1402.    png_debug(1, "in png_handle_pCAL");
  1403.    if (!(png_ptr->mode & PNG_HAVE_IHDR))
  1404.       png_error(png_ptr, "Missing IHDR before pCAL");
  1405.    else if (png_ptr->mode & PNG_HAVE_IDAT)
  1406.    {
  1407.       png_warning(png_ptr, "Invalid pCAL after IDAT");
  1408.       png_crc_finish(png_ptr, length);
  1409.       return;
  1410.    }
  1411.    else if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_pCAL))
  1412.    {
  1413.       png_warning(png_ptr, "Duplicate pCAL chunk");
  1414.       png_crc_finish(png_ptr, length);
  1415.       return;
  1416.    }
  1417.    png_debug1(2, "Allocating and reading pCAL chunk data (%lu bytes)",
  1418.       length + 1);
  1419.    png_free(png_ptr, png_ptr->chunkdata);
  1420.    png_ptr->chunkdata = (png_charp)png_malloc_warn(png_ptr, length + 1);
  1421.    if (png_ptr->chunkdata == NULL)
  1422.      {
  1423.        png_warning(png_ptr, "No memory for pCAL purpose.");
  1424.        return;
  1425.      }
  1426.    slength = (png_size_t)length;
  1427.    png_crc_read(png_ptr, (png_bytep)png_ptr->chunkdata, slength);
  1428.    if (png_crc_finish(png_ptr, 0))
  1429.    {
  1430.       png_free(png_ptr, png_ptr->chunkdata);
  1431.       png_ptr->chunkdata = NULL;
  1432.       return;
  1433.    }
  1434.    png_ptr->chunkdata[slength] = 0x00; /* null terminate the last string */
  1435.    png_debug(3, "Finding end of pCAL purpose string");
  1436.    for (buf = png_ptr->chunkdata; *buf; buf++)
  1437.       /* empty loop */ ;
  1438.    endptr = png_ptr->chunkdata + slength;
  1439.    /* We need to have at least 12 bytes after the purpose string
  1440.       in order to get the parameter information. */
  1441.    if (endptr <= buf + 12)
  1442.    {
  1443.       png_warning(png_ptr, "Invalid pCAL data");
  1444.       png_free(png_ptr, png_ptr->chunkdata);
  1445.       png_ptr->chunkdata = NULL;
  1446.       return;
  1447.    }
  1448.    png_debug(3, "Reading pCAL X0, X1, type, nparams, and units");
  1449.    X0 = png_get_int_32((png_bytep)buf+1);
  1450.    X1 = png_get_int_32((png_bytep)buf+5);
  1451.    type = buf[9];
  1452.    nparams = buf[10];
  1453.    units = buf + 11;
  1454.    png_debug(3, "Checking pCAL equation type and number of parameters");
  1455.    /* Check that we have the right number of parameters for known
  1456.       equation types. */
  1457.    if ((type == PNG_EQUATION_LINEAR && nparams != 2) ||
  1458.        (type == PNG_EQUATION_BASE_E && nparams != 3) ||
  1459.        (type == PNG_EQUATION_ARBITRARY && nparams != 3) ||
  1460.        (type == PNG_EQUATION_HYPERBOLIC && nparams != 4))
  1461.    {
  1462.       png_warning(png_ptr, "Invalid pCAL parameters for equation type");
  1463.       png_free(png_ptr, png_ptr->chunkdata);
  1464.       png_ptr->chunkdata = NULL;
  1465.       return;
  1466.    }
  1467.    else if (type >= PNG_EQUATION_LAST)
  1468.    {
  1469.       png_warning(png_ptr, "Unrecognized equation type for pCAL chunk");
  1470.    }
  1471.    for (buf = units; *buf; buf++)
  1472.       /* Empty loop to move past the units string. */ ;
  1473.    png_debug(3, "Allocating pCAL parameters array");
  1474.    params = (png_charpp)png_malloc_warn(png_ptr,
  1475.       (png_uint_32)(nparams * png_sizeof(png_charp))) ;
  1476.    if (params == NULL)
  1477.      {
  1478.        png_free(png_ptr, png_ptr->chunkdata);
  1479.        png_ptr->chunkdata = NULL;
  1480.        png_warning(png_ptr, "No memory for pCAL params.");
  1481.        return;
  1482.      }
  1483.    /* Get pointers to the start of each parameter string. */
  1484.    for (i = 0; i < (int)nparams; i++)
  1485.    {
  1486.       buf++; /* Skip the null string terminator from previous parameter. */
  1487.       png_debug1(3, "Reading pCAL parameter %d", i);
  1488.       for (params[i] = buf; buf <= endptr && *buf != 0x00; buf++)
  1489.          /* Empty loop to move past each parameter string */ ;
  1490.       /* Make sure we haven't run out of data yet */
  1491.       if (buf > endptr)
  1492.       {
  1493.          png_warning(png_ptr, "Invalid pCAL data");
  1494.          png_free(png_ptr, png_ptr->chunkdata);
  1495.          png_ptr->chunkdata = NULL;
  1496.          png_free(png_ptr, params);
  1497.          return;
  1498.       }
  1499.    }
  1500.    png_set_pCAL(png_ptr, info_ptr, png_ptr->chunkdata, X0, X1, type, nparams,
  1501.       units, params);
  1502.    png_free(png_ptr, png_ptr->chunkdata);
  1503.    png_ptr->chunkdata = NULL;
  1504.    png_free(png_ptr, params);
  1505. }
  1506. #endif
  1507. #if defined(PNG_READ_sCAL_SUPPORTED)
  1508. /* read the sCAL chunk */
  1509. void /* PRIVATE */
  1510. png_handle_sCAL(png_structp png_ptr, png_infop info_ptr, png_uint_32 length)
  1511. {
  1512.    png_charp ep;
  1513. #ifdef PNG_FLOATING_POINT_SUPPORTED
  1514.    double width, height;
  1515.    png_charp vp;
  1516. #else
  1517. #ifdef PNG_FIXED_POINT_SUPPORTED
  1518.    png_charp swidth, sheight;
  1519. #endif
  1520. #endif
  1521.    png_size_t slength;
  1522.    png_debug(1, "in png_handle_sCAL");
  1523.    if (!(png_ptr->mode & PNG_HAVE_IHDR))
  1524.       png_error(png_ptr, "Missing IHDR before sCAL");
  1525.    else if (png_ptr->mode & PNG_HAVE_IDAT)
  1526.    {
  1527.       png_warning(png_ptr, "Invalid sCAL after IDAT");
  1528.       png_crc_finish(png_ptr, length);
  1529.       return;
  1530.    }
  1531.    else if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_sCAL))
  1532.    {
  1533.       png_warning(png_ptr, "Duplicate sCAL chunk");
  1534.       png_crc_finish(png_ptr, length);
  1535.       return;
  1536.    }
  1537.    png_debug1(2, "Allocating and reading sCAL chunk data (%lu bytes)",
  1538.       length + 1);
  1539.    png_ptr->chunkdata = (png_charp)png_malloc_warn(png_ptr, length + 1);
  1540.    if (png_ptr->chunkdata == NULL)
  1541.    {
  1542.       png_warning(png_ptr, "Out of memory while processing sCAL chunk");
  1543.       return;
  1544.    }
  1545.    slength = (png_size_t)length;
  1546.    png_crc_read(png_ptr, (png_bytep)png_ptr->chunkdata, slength);
  1547.    if (png_crc_finish(png_ptr, 0))
  1548.    {
  1549.       png_free(png_ptr, png_ptr->chunkdata);
  1550.       png_ptr->chunkdata = NULL;
  1551.       return;
  1552.    }
  1553.    png_ptr->chunkdata[slength] = 0x00; /* null terminate the last string */
  1554.    ep = png_ptr->chunkdata + 1;        /* skip unit byte */
  1555. #ifdef PNG_FLOATING_POINT_SUPPORTED
  1556.    width = png_strtod(png_ptr, ep, &vp);
  1557.    if (*vp)
  1558.    {
  1559.       png_warning(png_ptr, "malformed width string in sCAL chunk");
  1560.       return;
  1561.    }
  1562. #else
  1563. #ifdef PNG_FIXED_POINT_SUPPORTED
  1564.    swidth = (png_charp)png_malloc_warn(png_ptr, png_strlen(ep) + 1);
  1565.    if (swidth == NULL)
  1566.    {
  1567.       png_warning(png_ptr, "Out of memory while processing sCAL chunk width");
  1568.       return;
  1569.    }
  1570.    png_memcpy(swidth, ep, (png_size_t)png_strlen(ep));
  1571. #endif
  1572. #endif
  1573.    for (ep = png_ptr->chunkdata; *ep; ep++)
  1574.       /* empty loop */ ;
  1575.    ep++;
  1576.    if (png_ptr->chunkdata + slength < ep)
  1577.    {
  1578.       png_warning(png_ptr, "Truncated sCAL chunk");
  1579. #if defined(PNG_FIXED_POINT_SUPPORTED) && 
  1580.     !defined(PNG_FLOATING_POINT_SUPPORTED)
  1581.       png_free(png_ptr, swidth);
  1582. #endif
  1583.       png_free(png_ptr, png_ptr->chunkdata);
  1584.       png_ptr->chunkdata = NULL;
  1585.       return;
  1586.    }
  1587. #ifdef PNG_FLOATING_POINT_SUPPORTED
  1588.    height = png_strtod(png_ptr, ep, &vp);
  1589.    if (*vp)
  1590.    {
  1591.       png_warning(png_ptr, "malformed height string in sCAL chunk");
  1592.       return;
  1593.    }
  1594. #else
  1595. #ifdef PNG_FIXED_POINT_SUPPORTED
  1596.    sheight = (png_charp)png_malloc_warn(png_ptr, png_strlen(ep) + 1);
  1597.    if (sheight == NULL)
  1598.    {
  1599.       png_warning(png_ptr, "Out of memory while processing sCAL chunk height");
  1600.       return;
  1601.    }
  1602.    png_memcpy(sheight, ep, (png_size_t)png_strlen(ep));
  1603. #endif
  1604. #endif
  1605.    if (png_ptr->chunkdata + slength < ep
  1606. #ifdef PNG_FLOATING_POINT_SUPPORTED
  1607.       || width <= 0. || height <= 0.
  1608. #endif
  1609.       )
  1610.    {
  1611.       png_warning(png_ptr, "Invalid sCAL data");
  1612.       png_free(png_ptr, png_ptr->chunkdata);
  1613.       png_ptr->chunkdata = NULL;
  1614. #if defined(PNG_FIXED_POINT_SUPPORTED) && !defined(PNG_FLOATING_POINT_SUPPORTED)
  1615.       png_free(png_ptr, swidth);
  1616.       png_free(png_ptr, sheight);
  1617. #endif
  1618.       return;
  1619.    }
  1620. #ifdef PNG_FLOATING_POINT_SUPPORTED
  1621.    png_set_sCAL(png_ptr, info_ptr, png_ptr->chunkdata[0], width, height);
  1622. #else
  1623. #ifdef PNG_FIXED_POINT_SUPPORTED
  1624.    png_set_sCAL_s(png_ptr, info_ptr, png_ptr->chunkdata[0], swidth, sheight);
  1625. #endif
  1626. #endif
  1627.    png_free(png_ptr, png_ptr->chunkdata);
  1628.    png_ptr->chunkdata = NULL;
  1629. #if defined(PNG_FIXED_POINT_SUPPORTED) && !defined(PNG_FLOATING_POINT_SUPPORTED)
  1630.    png_free(png_ptr, swidth);
  1631.    png_free(png_ptr, sheight);
  1632. #endif
  1633. }
  1634. #endif
  1635. #if defined(PNG_READ_tIME_SUPPORTED)
  1636. void /* PRIVATE */
  1637. png_handle_tIME(png_structp png_ptr, png_infop info_ptr, png_uint_32 length)
  1638. {
  1639.    png_byte buf[7];
  1640.    png_time mod_time;
  1641.    png_debug(1, "in png_handle_tIME");
  1642.    if (!(png_ptr->mode & PNG_HAVE_IHDR))
  1643.       png_error(png_ptr, "Out of place tIME chunk");
  1644.    else if (info_ptr != NULL && (info_ptr->valid & PNG_INFO_tIME))
  1645.    {
  1646.       png_warning(png_ptr, "Duplicate tIME chunk");
  1647.       png_crc_finish(png_ptr, length);
  1648.       return;
  1649.    }
  1650.    if (png_ptr->mode & PNG_HAVE_IDAT)
  1651.       png_ptr->mode |= PNG_AFTER_IDAT;
  1652.    if (length != 7)
  1653.    {
  1654.       png_warning(png_ptr, "Incorrect tIME chunk length");
  1655.       png_crc_finish(png_ptr, length);
  1656.       return;
  1657.    }
  1658.    png_crc_read(png_ptr, buf, 7);
  1659.    if (png_crc_finish(png_ptr, 0))
  1660.       return;
  1661.    mod_time.second = buf[6];
  1662.    mod_time.minute = buf[5];
  1663.    mod_time.hour = buf[4];
  1664.    mod_time.day = buf[3];
  1665.    mod_time.month = buf[2];
  1666.    mod_time.year = png_get_uint_16(buf);
  1667.    png_set_tIME(png_ptr, info_ptr, &mod_time);
  1668. }
  1669. #endif
  1670. #if defined(PNG_READ_tEXt_SUPPORTED)
  1671. /* Note: this does not properly handle chunks that are > 64K under DOS */
  1672. void /* PRIVATE */
  1673. png_handle_tEXt(png_structp png_ptr, png_infop info_ptr, png_uint_32 length)
  1674. {
  1675.    png_textp text_ptr;
  1676.    png_charp key;
  1677.    png_charp text;
  1678.    png_uint_32 skip = 0;
  1679.    png_size_t slength;
  1680.    int ret;
  1681.    png_debug(1, "in png_handle_tEXt");
  1682.    if (!(png_ptr->mode & PNG_HAVE_IHDR))
  1683.       png_error(png_ptr, "Missing IHDR before tEXt");
  1684.    if (png_ptr->mode & PNG_HAVE_IDAT)
  1685.       png_ptr->mode |= PNG_AFTER_IDAT;
  1686. #ifdef PNG_MAX_MALLOC_64K
  1687.    if (length > (png_uint_32)65535L)
  1688.    {
  1689.       png_warning(png_ptr, "tEXt chunk too large to fit in memory");
  1690.       skip = length - (png_uint_32)65535L;
  1691.       length = (png_uint_32)65535L;
  1692.    }
  1693. #endif
  1694.    png_free(png_ptr, png_ptr->chunkdata);
  1695.    png_ptr->chunkdata = (png_charp)png_malloc_warn(png_ptr, length + 1);
  1696.    if (png_ptr->chunkdata == NULL)
  1697.    {
  1698.      png_warning(png_ptr, "No memory to process text chunk.");
  1699.      return;
  1700.    }
  1701.    slength = (png_size_t)length;
  1702.    png_crc_read(png_ptr, (png_bytep)png_ptr->chunkdata, slength);
  1703.    if (png_crc_finish(png_ptr, skip))
  1704.    {
  1705.       png_free(png_ptr, png_ptr->chunkdata);
  1706.       png_ptr->chunkdata = NULL;
  1707.       return;
  1708.    }
  1709.    key = png_ptr->chunkdata;
  1710.    key[slength] = 0x00;
  1711.    for (text = key; *text; text++)
  1712.       /* empty loop to find end of key */ ;
  1713.    if (text != key + slength)
  1714.       text++;
  1715.    text_ptr = (png_textp)png_malloc_warn(png_ptr,
  1716.       (png_uint_32)png_sizeof(png_text));
  1717.    if (text_ptr == NULL)
  1718.    {
  1719.      png_warning(png_ptr, "Not enough memory to process text chunk.");
  1720.      png_free(png_ptr, png_ptr->chunkdata);
  1721.      png_ptr->chunkdata = NULL;
  1722.      return;
  1723.    }
  1724.    text_ptr->compression = PNG_TEXT_COMPRESSION_NONE;
  1725.    text_ptr->key = key;
  1726. #ifdef PNG_iTXt_SUPPORTED
  1727.    text_ptr->lang = NULL;
  1728.    text_ptr->lang_key = NULL;
  1729.    text_ptr->itxt_length = 0;
  1730. #endif
  1731.    text_ptr->text = text;
  1732.    text_ptr->text_length = png_strlen(text);
  1733.    ret = png_set_text_2(png_ptr, info_ptr, text_ptr, 1);
  1734.    png_free(png_ptr, png_ptr->chunkdata);
  1735.    png_ptr->chunkdata = NULL;
  1736.    png_free(png_ptr, text_ptr);
  1737.    if (ret)
  1738.      png_warning(png_ptr, "Insufficient memory to process text chunk.");
  1739. }
  1740. #endif
  1741. #if defined(PNG_READ_zTXt_SUPPORTED)
  1742. /* note: this does not correctly handle chunks that are > 64K under DOS */
  1743. void /* PRIVATE */
  1744. png_handle_zTXt(png_structp png_ptr, png_infop info_ptr, png_uint_32 length)
  1745. {
  1746.    png_textp text_ptr;
  1747.    png_charp text;
  1748.    int comp_type;
  1749.    int ret;
  1750.    png_size_t slength, prefix_len, data_len;
  1751.    png_debug(1, "in png_handle_zTXt");
  1752.    if (!(png_ptr->mode & PNG_HAVE_IHDR))
  1753.       png_error(png_ptr, "Missing IHDR before zTXt");
  1754.    if (png_ptr->mode & PNG_HAVE_IDAT)
  1755.       png_ptr->mode |= PNG_AFTER_IDAT;
  1756. #ifdef PNG_MAX_MALLOC_64K
  1757.    /* We will no doubt have problems with chunks even half this size, but
  1758.       there is no hard and fast rule to tell us where to stop. */
  1759.    if (length > (png_uint_32)65535L)
  1760.    {
  1761.      png_warning(png_ptr, "zTXt chunk too large to fit in memory");
  1762.      png_crc_finish(png_ptr, length);
  1763.      return;
  1764.    }
  1765. #endif
  1766.    png_free(png_ptr, png_ptr->chunkdata);
  1767.    png_ptr->chunkdata = (png_charp)png_malloc_warn(png_ptr, length + 1);
  1768.    if (png_ptr->chunkdata == NULL)
  1769.    {
  1770.      png_warning(png_ptr, "Out of memory processing zTXt chunk.");
  1771.      return;
  1772.    }
  1773.    slength = (png_size_t)length;
  1774.    png_crc_read(png_ptr, (png_bytep)png_ptr->chunkdata, slength);
  1775.    if (png_crc_finish(png_ptr, 0))
  1776.    {
  1777.       png_free(png_ptr, png_ptr->chunkdata);
  1778.       png_ptr->chunkdata = NULL;
  1779.       return;
  1780.    }
  1781.    png_ptr->chunkdata[slength] = 0x00;
  1782.    for (text = png_ptr->chunkdata; *text; text++)
  1783.       /* empty loop */ ;
  1784.    /* zTXt must have some text after the chunkdataword */
  1785.    if (text >= png_ptr->chunkdata + slength - 2)
  1786.    {
  1787.       png_warning(png_ptr, "Truncated zTXt chunk");
  1788.       png_free(png_ptr, png_ptr->chunkdata);
  1789.       png_ptr->chunkdata = NULL;
  1790.       return;
  1791.    }
  1792.    else
  1793.    {
  1794.        comp_type = *(++text);
  1795.        if (comp_type != PNG_TEXT_COMPRESSION_zTXt)
  1796.        {
  1797.           png_warning(png_ptr, "Unknown compression type in zTXt chunk");
  1798.           comp_type = PNG_TEXT_COMPRESSION_zTXt;
  1799.        }
  1800.        text++;        /* skip the compression_method byte */
  1801.    }
  1802.    prefix_len = text - png_ptr->chunkdata;
  1803.    png_decompress_chunk(png_ptr, comp_type,
  1804.      (png_size_t)length, prefix_len, &data_len);
  1805.    text_ptr = (png_textp)png_malloc_warn(png_ptr,
  1806.       (png_uint_32)png_sizeof(png_text));
  1807.    if (text_ptr == NULL)
  1808.    {
  1809.      png_warning(png_ptr, "Not enough memory to process zTXt chunk.");
  1810.      png_free(png_ptr, png_ptr->chunkdata);
  1811.      png_ptr->chunkdata = NULL;
  1812.      return;
  1813.    }
  1814.    text_ptr->compression = comp_type;
  1815.    text_ptr->key = png_ptr->chunkdata;
  1816. #ifdef PNG_iTXt_SUPPORTED
  1817.    text_ptr->lang = NULL;
  1818.    text_ptr->lang_key = NULL;
  1819.    text_ptr->itxt_length = 0;
  1820. #endif
  1821.    text_ptr->text = png_ptr->chunkdata + prefix_len;
  1822.    text_ptr->text_length = data_len;
  1823.    ret = png_set_text_2(png_ptr, info_ptr, text_ptr, 1);
  1824.    png_free(png_ptr, text_ptr);
  1825.    png_free(png_ptr, png_ptr->chunkdata);
  1826.    png_ptr->chunkdata = NULL;
  1827.    if (ret)
  1828.      png_error(png_ptr, "Insufficient memory to store zTXt chunk.");
  1829. }
  1830. #endif
  1831. #if defined(PNG_READ_iTXt_SUPPORTED)
  1832. /* note: this does not correctly handle chunks that are > 64K under DOS */
  1833. void /* PRIVATE */
  1834. png_handle_iTXt(png_structp png_ptr, png_infop info_ptr, png_uint_32 length)
  1835. {
  1836.    png_textp text_ptr;
  1837.    png_charp key, lang, text, lang_key;
  1838.    int comp_flag;
  1839.    int comp_type = 0;
  1840.    int ret;
  1841.    png_size_t slength, prefix_len, data_len;
  1842.    png_debug(1, "in png_handle_iTXt");
  1843.    if (!(png_ptr->mode & PNG_HAVE_IHDR))
  1844.       png_error(png_ptr, "Missing IHDR before iTXt");
  1845.    if (png_ptr->mode & PNG_HAVE_IDAT)
  1846.       png_ptr->mode |= PNG_AFTER_IDAT;
  1847. #ifdef PNG_MAX_MALLOC_64K
  1848.    /* We will no doubt have problems with chunks even half this size, but
  1849.       there is no hard and fast rule to tell us where to stop. */
  1850.    if (length > (png_uint_32)65535L)
  1851.    {
  1852.      png_warning(png_ptr, "iTXt chunk too large to fit in memory");
  1853.      png_crc_finish(png_ptr, length);
  1854.      return;
  1855.    }
  1856. #endif
  1857.    png_free(png_ptr, png_ptr->chunkdata);
  1858.    png_ptr->chunkdata = (png_charp)png_malloc_warn(png_ptr, length + 1);
  1859.    if (png_ptr->chunkdata == NULL)
  1860.    {
  1861.      png_warning(png_ptr, "No memory to process iTXt chunk.");
  1862.      return;
  1863.    }
  1864.    slength = (png_size_t)length;
  1865.    png_crc_read(png_ptr, (png_bytep)png_ptr->chunkdata, slength);
  1866.    if (png_crc_finish(png_ptr, 0))
  1867.    {
  1868.       png_free(png_ptr, png_ptr->chunkdata);
  1869.       png_ptr->chunkdata = NULL;
  1870.       return;
  1871.    }
  1872.    png_ptr->chunkdata[slength] = 0x00;
  1873.    for (lang = png_ptr->chunkdata; *lang; lang++)
  1874.       /* empty loop */ ;
  1875.    lang++;        /* skip NUL separator */
  1876.    /* iTXt must have a language tag (possibly empty), two compression bytes,
  1877.       translated keyword (possibly empty), and possibly some text after the
  1878.       keyword */
  1879.    if (lang >= png_ptr->chunkdata + slength - 3)
  1880.    {
  1881.       png_warning(png_ptr, "Truncated iTXt chunk");
  1882.       png_free(png_ptr, png_ptr->chunkdata);
  1883.       png_ptr->chunkdata = NULL;
  1884.       return;
  1885.    }
  1886.    else
  1887.    {
  1888.        comp_flag = *lang++;
  1889.        comp_type = *lang++;
  1890.    }
  1891.    for (lang_key = lang; *lang_key; lang_key++)
  1892.       /* empty loop */ ;
  1893.    lang_key++;        /* skip NUL separator */
  1894.    if (lang_key >= png_ptr->chunkdata + slength)
  1895.    {
  1896.       png_warning(png_ptr, "Truncated iTXt chunk");
  1897.       png_free(png_ptr, png_ptr->chunkdata);
  1898.       png_ptr->chunkdata = NULL;
  1899.       return;
  1900.    }
  1901.    for (text = lang_key; *text; text++)
  1902.       /* empty loop */ ;
  1903.    text++;        /* skip NUL separator */
  1904.    if (text >= png_ptr->chunkdata + slength)
  1905.    {
  1906.       png_warning(png_ptr, "Malformed iTXt chunk");
  1907.       png_free(png_ptr, png_ptr->chunkdata);
  1908.       png_ptr->chunkdata = NULL;
  1909.       return;
  1910.    }
  1911.    prefix_len = text - png_ptr->chunkdata;
  1912.    key=png_ptr->chunkdata;
  1913.    if (comp_flag)
  1914.        png_decompress_chunk(png_ptr, comp_type,
  1915.          (size_t)length, prefix_len, &data_len);
  1916.    else
  1917.        data_len = png_strlen(png_ptr->chunkdata + prefix_len);
  1918.    text_ptr = (png_textp)png_malloc_warn(png_ptr,
  1919.       (png_uint_32)png_sizeof(png_text));
  1920.    if (text_ptr == NULL)
  1921.    {
  1922.      png_warning(png_ptr, "Not enough memory to process iTXt chunk.");
  1923.      png_free(png_ptr, png_ptr->chunkdata);
  1924.      png_ptr->chunkdata = NULL;
  1925.      return;
  1926.    }
  1927.    text_ptr->compression = (int)comp_flag + 1;
  1928.    text_ptr->lang_key = png_ptr->chunkdata + (lang_key - key);
  1929.    text_ptr->lang = png_ptr->chunkdata + (lang - key);
  1930.    text_ptr->itxt_length = data_len;
  1931.    text_ptr->text_length = 0;
  1932.    text_ptr->key = png_ptr->chunkdata;
  1933.    text_ptr->text = png_ptr->chunkdata + prefix_len;
  1934.    ret = png_set_text_2(png_ptr, info_ptr, text_ptr, 1);
  1935.    png_free(png_ptr, text_ptr);
  1936.    png_free(png_ptr, png_ptr->chunkdata);
  1937.    png_ptr->chunkdata = NULL;
  1938.    if (ret)
  1939.      png_error(png_ptr, "Insufficient memory to store iTXt chunk.");
  1940. }
  1941. #endif
  1942. /* This function is called when we haven't found a handler for a
  1943.    chunk.  If there isn't a problem with the chunk itself (ie bad
  1944.    chunk name, CRC, or a critical chunk), the chunk is silently ignored
  1945.    -- unless the PNG_FLAG_UNKNOWN_CHUNKS_SUPPORTED flag is on in which
  1946.    case it will be saved away to be written out later. */
  1947. void /* PRIVATE */
  1948. png_handle_unknown(png_structp png_ptr, png_infop info_ptr, png_uint_32 length)
  1949. {
  1950.    png_uint_32 skip = 0;
  1951.    png_debug(1, "in png_handle_unknown");
  1952.    if (png_ptr->mode & PNG_HAVE_IDAT)
  1953.    {
  1954. #ifdef PNG_USE_LOCAL_ARRAYS
  1955.       PNG_CONST PNG_IDAT;
  1956. #endif
  1957.       if (png_memcmp(png_ptr->chunk_name, png_IDAT, 4))  /* not an IDAT */
  1958.          png_ptr->mode |= PNG_AFTER_IDAT;
  1959.    }
  1960.    if (!(png_ptr->chunk_name[0] & 0x20))
  1961.    {
  1962. #if defined(PNG_READ_UNKNOWN_CHUNKS_SUPPORTED)
  1963.       if (png_handle_as_unknown(png_ptr, png_ptr->chunk_name) !=
  1964.            PNG_HANDLE_CHUNK_ALWAYS
  1965. #if defined(PNG_READ_USER_CHUNKS_SUPPORTED)
  1966.            && png_ptr->read_user_chunk_fn == NULL
  1967. #endif
  1968.         )
  1969. #endif
  1970.           png_chunk_error(png_ptr, "unknown critical chunk");
  1971.    }
  1972. #if defined(PNG_READ_UNKNOWN_CHUNKS_SUPPORTED)
  1973.    if ((png_ptr->flags & PNG_FLAG_KEEP_UNKNOWN_CHUNKS) ||
  1974.        (png_ptr->read_user_chunk_fn != NULL))
  1975.    {
  1976. #ifdef PNG_MAX_MALLOC_64K
  1977.        if (length > (png_uint_32)65535L)
  1978.        {
  1979.            png_warning(png_ptr, "unknown chunk too large to fit in memory");
  1980.            skip = length - (png_uint_32)65535L;
  1981.            length = (png_uint_32)65535L;
  1982.        }
  1983. #endif
  1984.        png_memcpy((png_charp)png_ptr->unknown_chunk.name,
  1985.                   (png_charp)png_ptr->chunk_name, 
  1986.                   png_sizeof(png_ptr->unknown_chunk.name));
  1987.        png_ptr->unknown_chunk.name[png_sizeof(png_ptr->unknown_chunk.name)-1] = '';
  1988.        png_ptr->unknown_chunk.size = (png_size_t)length;
  1989.        if (length == 0)
  1990.          png_ptr->unknown_chunk.data = NULL;
  1991.        else
  1992.        {
  1993.          png_ptr->unknown_chunk.data = (png_bytep)png_malloc(png_ptr, length);
  1994.          png_crc_read(png_ptr, (png_bytep)png_ptr->unknown_chunk.data, length);
  1995.        }
  1996. #if defined(PNG_READ_USER_CHUNKS_SUPPORTED)
  1997.        if (png_ptr->read_user_chunk_fn != NULL)
  1998.        {
  1999.           /* callback to user unknown chunk handler */
  2000.           int ret;
  2001.           ret = (*(png_ptr->read_user_chunk_fn))
  2002.             (png_ptr, &png_ptr->unknown_chunk);
  2003.           if (ret < 0)
  2004.              png_chunk_error(png_ptr, "error in user chunk");
  2005.           if (ret == 0)
  2006.           {
  2007.              if (!(png_ptr->chunk_name[0] & 0x20))
  2008.                 if (png_handle_as_unknown(png_ptr, png_ptr->chunk_name) !=
  2009.                      PNG_HANDLE_CHUNK_ALWAYS)
  2010.                    png_chunk_error(png_ptr, "unknown critical chunk");
  2011.              png_set_unknown_chunks(png_ptr, info_ptr,
  2012.                &png_ptr->unknown_chunk, 1);
  2013.           }
  2014.        }
  2015.        else
  2016. #endif
  2017.        png_set_unknown_chunks(png_ptr, info_ptr, &png_ptr->unknown_chunk, 1);
  2018.        png_free(png_ptr, png_ptr->unknown_chunk.data);
  2019.        png_ptr->unknown_chunk.data = NULL;
  2020.    }
  2021.    else
  2022. #endif
  2023.       skip = length;
  2024.    png_crc_finish(png_ptr, skip);
  2025. #if !defined(PNG_READ_USER_CHUNKS_SUPPORTED)
  2026.    info_ptr = info_ptr; /* quiet compiler warnings about unused info_ptr */
  2027. #endif
  2028. }
  2029. /* This function is called to verify that a chunk name is valid.
  2030.    This function can't have the "critical chunk check" incorporated
  2031.    into it, since in the future we will need to be able to call user
  2032.    functions to handle unknown critical chunks after we check that
  2033.    the chunk name itself is valid. */
  2034. #define isnonalpha(c) ((c) < 65 || (c) > 122 || ((c) > 90 && (c) < 97))
  2035. void /* PRIVATE */
  2036. png_check_chunk_name(png_structp png_ptr, png_bytep chunk_name)
  2037. {
  2038.    png_debug(1, "in png_check_chunk_name");
  2039.    if (isnonalpha(chunk_name[0]) || isnonalpha(chunk_name[1]) ||
  2040.        isnonalpha(chunk_name[2]) || isnonalpha(chunk_name[3]))
  2041.    {
  2042.       png_chunk_error(png_ptr, "invalid chunk type");
  2043.    }
  2044. }
  2045. /* Combines the row recently read in with the existing pixels in the
  2046.    row.  This routine takes care of alpha and transparency if requested.
  2047.    This routine also handles the two methods of progressive display
  2048.    of interlaced images, depending on the mask value.
  2049.    The mask value describes which pixels are to be combined with
  2050.    the row.  The pattern always repeats every 8 pixels, so just 8
  2051.    bits are needed.  A one indicates the pixel is to be combined,
  2052.    a zero indicates the pixel is to be skipped.  This is in addition
  2053.    to any alpha or transparency value associated with the pixel.  If
  2054.    you want all pixels to be combined, pass 0xff (255) in mask.  */
  2055. void /* PRIVATE */
  2056. png_combine_row(png_structp png_ptr, png_bytep row, int mask)
  2057. {
  2058.    png_debug(1, "in png_combine_row");
  2059.    if (mask == 0xff)
  2060.    {
  2061.       png_memcpy(row, png_ptr->row_buf + 1,
  2062.          PNG_ROWBYTES(png_ptr->row_info.pixel_depth, png_ptr->width));
  2063.    }
  2064.    else
  2065.    {
  2066.       switch (png_ptr->row_info.pixel_depth)
  2067.       {
  2068.          case 1:
  2069.          {
  2070.             png_bytep sp = png_ptr->row_buf + 1;
  2071.             png_bytep dp = row;
  2072.             int s_inc, s_start, s_end;
  2073.             int m = 0x80;
  2074.             int shift;
  2075.             png_uint_32 i;
  2076.             png_uint_32 row_width = png_ptr->width;
  2077. #if defined(PNG_READ_PACKSWAP_SUPPORTED)
  2078.             if (png_ptr->transformations & PNG_PACKSWAP)
  2079.             {
  2080.                 s_start = 0;
  2081.                 s_end = 7;
  2082.                 s_inc = 1;
  2083.             }
  2084.             else
  2085. #endif
  2086.             {
  2087.                 s_start = 7;
  2088.                 s_end = 0;
  2089.                 s_inc = -1;
  2090.             }
  2091.             shift = s_start;
  2092.             for (i = 0; i < row_width; i++)
  2093.             {
  2094.                if (m & mask)
  2095.                {
  2096.                   int value;
  2097.                   value = (*sp >> shift) & 0x01;
  2098.                   *dp &= (png_byte)((0x7f7f >> (7 - shift)) & 0xff);
  2099.                   *dp |= (png_byte)(value << shift);
  2100.                }
  2101.                if (shift == s_end)
  2102.                {
  2103.                   shift = s_start;
  2104.                   sp++;
  2105.                   dp++;
  2106.                }
  2107.                else
  2108.                   shift += s_inc;
  2109.                if (m == 1)
  2110.                   m = 0x80;
  2111.                else
  2112.                   m >>= 1;
  2113.             }
  2114.             break;
  2115.          }
  2116.          case 2:
  2117.          {
  2118.             png_bytep sp = png_ptr->row_buf + 1;
  2119.             png_bytep dp = row;
  2120.             int s_start, s_end, s_inc;
  2121.             int m = 0x80;
  2122.             int shift;
  2123.             png_uint_32 i;
  2124.             png_uint_32 row_width = png_ptr->width;
  2125.             int value;
  2126. #if defined(PNG_READ_PACKSWAP_SUPPORTED)
  2127.             if (png_ptr->transformations & PNG_PACKSWAP)
  2128.             {
  2129.                s_start = 0;
  2130.                s_end = 6;
  2131.                s_inc = 2;
  2132.             }
  2133.             else
  2134. #endif
  2135.             {
  2136.                s_start = 6;
  2137.                s_end = 0;
  2138.                s_inc = -2;
  2139.             }
  2140.             shift = s_start;
  2141.             for (i = 0; i < row_width; i++)
  2142.             {
  2143.                if (m & mask)
  2144.                {
  2145.                   value = (*sp >> shift) & 0x03;
  2146.                   *dp &= (png_byte)((0x3f3f >> (6 - shift)) & 0xff);
  2147.                   *dp |= (png_byte)(value << shift);
  2148.                }
  2149.                if (shift == s_end)
  2150.                {
  2151.                   shift = s_start;
  2152.                   sp++;
  2153.                   dp++;
  2154.                }
  2155.                else
  2156.                   shift += s_inc;
  2157.                if (m == 1)
  2158.                   m = 0x80;
  2159.                else
  2160.                   m >>= 1;
  2161.             }
  2162.             break;
  2163.          }
  2164.          case 4:
  2165.          {
  2166.             png_bytep sp = png_ptr->row_buf + 1;
  2167.             png_bytep dp = row;
  2168.             int s_start, s_end, s_inc;
  2169.             int m = 0x80;
  2170.             int shift;
  2171.             png_uint_32 i;
  2172.             png_uint_32 row_width = png_ptr->width;
  2173.             int value;
  2174. #if defined(PNG_READ_PACKSWAP_SUPPORTED)
  2175.             if (png_ptr->transformations & PNG_PACKSWAP)
  2176.             {
  2177.                s_start = 0;
  2178.                s_end = 4;
  2179.                s_inc = 4;
  2180.             }
  2181.             else
  2182. #endif
  2183.             {
  2184.                s_start = 4;
  2185.                s_end = 0;
  2186.                s_inc = -4;
  2187.             }
  2188.             shift = s_start;
  2189.             for (i = 0; i < row_width; i++)
  2190.             {
  2191.                if (m & mask)
  2192.                {
  2193.                   value = (*sp >> shift) & 0xf;
  2194.                   *dp &= (png_byte)((0xf0f >> (4 - shift)) & 0xff);
  2195.                   *dp |= (png_byte)(value << shift);
  2196.                }
  2197.                if (shift == s_end)
  2198.                {
  2199.                   shift = s_start;
  2200.                   sp++;
  2201.                   dp++;
  2202.                }
  2203.                else
  2204.                   shift += s_inc;
  2205.                if (m == 1)
  2206.                   m = 0x80;
  2207.                else
  2208.                   m >>= 1;
  2209.             }
  2210.             break;
  2211.          }
  2212.          default:
  2213.          {
  2214.             png_bytep sp = png_ptr->row_buf + 1;
  2215.             png_bytep dp = row;
  2216.             png_size_t pixel_bytes = (png_ptr->row_info.pixel_depth >> 3);
  2217.             png_uint_32 i;
  2218.             png_uint_32 row_width = png_ptr->width;
  2219.             png_byte m = 0x80;
  2220.             for (i = 0; i < row_width; i++)
  2221.             {
  2222.                if (m & mask)
  2223.                {
  2224.                   png_memcpy(dp, sp, pixel_bytes);
  2225.                }
  2226.                sp += pixel_bytes;
  2227.                dp += pixel_bytes;
  2228.                if (m == 1)
  2229.                   m = 0x80;
  2230.                else
  2231.                   m >>= 1;
  2232.             }
  2233.             break;
  2234.          }
  2235.       }
  2236.    }
  2237. }
  2238. #ifdef PNG_READ_INTERLACING_SUPPORTED
  2239. /* OLD pre-1.0.9 interface:
  2240. void png_do_read_interlace(png_row_infop row_info, png_bytep row, int pass,
  2241.    png_uint_32 transformations)
  2242.  */
  2243. void /* PRIVATE */
  2244. png_do_read_interlace(png_structp png_ptr)
  2245. {
  2246.    png_row_infop row_info = &(png_ptr->row_info);
  2247.    png_bytep row = png_ptr->row_buf + 1;
  2248.    int pass = png_ptr->pass;
  2249.    png_uint_32 transformations = png_ptr->transformations;
  2250. #ifdef PNG_USE_LOCAL_ARRAYS
  2251.    /* arrays to facilitate easy interlacing - use pass (0 - 6) as index */
  2252.    /* offset to next interlace block */
  2253.    PNG_CONST int png_pass_inc[7] = {8, 8, 4, 4, 2, 2, 1};
  2254. #endif
  2255.    png_debug(1, "in png_do_read_interlace");
  2256.    if (row != NULL && row_info != NULL)
  2257.    {
  2258.       png_uint_32 final_width;
  2259.       final_width = row_info->width * png_pass_inc[pass];
  2260.       switch (row_info->pixel_depth)
  2261.       {
  2262.          case 1:
  2263.          {
  2264.             png_bytep sp = row + (png_size_t)((row_info->width - 1) >> 3);
  2265.             png_bytep dp = row + (png_size_t)((final_width - 1) >> 3);
  2266.             int sshift, dshift;
  2267.             int s_start, s_end, s_inc;
  2268.             int jstop = png_pass_inc[pass];
  2269.             png_byte v;
  2270.             png_uint_32 i;
  2271.             int j;
  2272. #if defined(PNG_READ_PACKSWAP_SUPPORTED)
  2273.             if (transformations & PNG_PACKSWAP)
  2274.             {
  2275.                 sshift = (int)((row_info->width + 7) & 0x07);
  2276.                 dshift = (int)((final_width + 7) & 0x07);
  2277.                 s_start = 7;
  2278.                 s_end = 0;
  2279.                 s_inc = -1;
  2280.             }
  2281.             else
  2282. #endif
  2283.             {
  2284.                 sshift = 7 - (int)((row_info->width + 7) & 0x07);
  2285.                 dshift = 7 - (int)((final_width + 7) & 0x07);
  2286.                 s_start = 0;
  2287.                 s_end = 7;
  2288.                 s_inc = 1;
  2289.             }
  2290.             for (i = 0; i < row_info->width; i++)
  2291.             {
  2292.                v = (png_byte)((*sp >> sshift) & 0x01);
  2293.                for (j = 0; j < jstop; j++)
  2294.                {
  2295.                   *dp &= (png_byte)((0x7f7f >> (7 - dshift)) & 0xff);
  2296.                   *dp |= (png_byte)(v << dshift);
  2297.                   if (dshift == s_end)
  2298.                   {
  2299.                      dshift = s_start;
  2300.                      dp--;
  2301.                   }
  2302.                   else
  2303.                      dshift += s_inc;
  2304.                }
  2305.                if (sshift == s_end)
  2306.                {
  2307.                   sshift = s_start;
  2308.                   sp--;
  2309.                }
  2310.                else
  2311.                   sshift += s_inc;
  2312.             }
  2313.             break;
  2314.          }
  2315.          case 2:
  2316.          {
  2317.             png_bytep sp = row + (png_uint_32)((row_info->width - 1) >> 2);
  2318.             png_bytep dp = row + (png_uint_32)((final_width - 1) >> 2);
  2319.             int sshift, dshift;
  2320.             int s_start, s_end, s_inc;
  2321.             int jstop = png_pass_inc[pass];
  2322.             png_uint_32 i;
  2323. #if defined(PNG_READ_PACKSWAP_SUPPORTED)
  2324.             if (transformations & PNG_PACKSWAP)
  2325.             {
  2326.                sshift = (int)(((row_info->width + 3) & 0x03) << 1);
  2327.                dshift = (int)(((final_width + 3) & 0x03) << 1);
  2328.                s_start = 6;
  2329.                s_end = 0;
  2330.                s_inc = -2;
  2331.             }
  2332.             else
  2333. #endif
  2334.             {
  2335.                sshift = (int)((3 - ((row_info->width + 3) & 0x03)) << 1);
  2336.                dshift = (int)((3 - ((final_width + 3) & 0x03)) << 1);
  2337.                s_start = 0;
  2338.                s_end = 6;
  2339.                s_inc = 2;
  2340.             }
  2341.             for (i = 0; i < row_info->width; i++)
  2342.             {
  2343.                png_byte v;
  2344.                int j;
  2345.                v = (png_byte)((*sp >> sshift) & 0x03);
  2346.                for (j = 0; j < jstop; j++)
  2347.                {
  2348.                   *dp &= (png_byte)((0x3f3f >> (6 - dshift)) & 0xff);
  2349.                   *dp |= (png_byte)(v << dshift);
  2350.                   if (dshift == s_end)
  2351.                   {
  2352.                      dshift = s_start;
  2353.                      dp--;
  2354.                   }
  2355.                   else
  2356.                      dshift += s_inc;
  2357.                }
  2358.                if (sshift == s_end)
  2359.                {
  2360.                   sshift = s_start;
  2361.                   sp--;
  2362.                }
  2363.                else
  2364.                   sshift += s_inc;
  2365.             }
  2366.             break;
  2367.          }
  2368.          case 4:
  2369.          {
  2370.             png_bytep sp = row + (png_size_t)((row_info->width - 1) >> 1);
  2371.             png_bytep dp = row + (png_size_t)((final_width - 1) >> 1);
  2372.             int sshift, dshift;
  2373.             int s_start, s_end, s_inc;
  2374.             png_uint_32 i;
  2375.             int jstop = png_pass_inc[pass];
  2376. #if defined(PNG_READ_PACKSWAP_SUPPORTED)
  2377.             if (transformations & PNG_PACKSWAP)
  2378.             {
  2379.                sshift = (int)(((row_info->width + 1) & 0x01) << 2);
  2380.                dshift = (int)(((final_width + 1) & 0x01) << 2);
  2381.                s_start = 4;
  2382.                s_end = 0;
  2383.                s_inc = -4;
  2384.             }
  2385.             else
  2386. #endif
  2387.             {
  2388.                sshift = (int)((1 - ((row_info->width + 1) & 0x01)) << 2);
  2389.                dshift = (int)((1 - ((final_width + 1) & 0x01)) << 2);
  2390.                s_start = 0;
  2391.                s_end = 4;
  2392.                s_inc = 4;
  2393.             }
  2394.             for (i = 0; i < row_info->width; i++)
  2395.             {
  2396.                png_byte v = (png_byte)((*sp >> sshift) & 0xf);
  2397.                int j;
  2398.                for (j = 0; j < jstop; j++)
  2399.                {
  2400.                   *dp &= (png_byte)((0xf0f >> (4 - dshift)) & 0xff);
  2401.                   *dp |= (png_byte)(v << dshift);
  2402.                   if (dshift == s_end)
  2403.                   {
  2404.                      dshift = s_start;
  2405.                      dp--;
  2406.                   }
  2407.                   else
  2408.                      dshift += s_inc;
  2409.                }
  2410.                if (sshift == s_end)
  2411.                {
  2412.                   sshift = s_start;
  2413.                   sp--;
  2414.                }
  2415.                else
  2416.                   sshift += s_inc;
  2417.             }
  2418.             break;
  2419.          }
  2420.          default:
  2421.          {
  2422.             png_size_t pixel_bytes = (row_info->pixel_depth >> 3);
  2423.             png_bytep sp = row + (png_size_t)(row_info->width - 1) * pixel_bytes;
  2424.             png_bytep dp = row + (png_size_t)(final_width - 1) * pixel_bytes;
  2425.             int jstop = png_pass_inc[pass];
  2426.             png_uint_32 i;
  2427.             for (i = 0; i < row_info->width; i++)
  2428.             {
  2429.                png_byte v[8];
  2430.                int j;
  2431.                png_memcpy(v, sp, pixel_bytes);
  2432.                for (j = 0; j < jstop; j++)
  2433.                {
  2434.                   png_memcpy(dp, v, pixel_bytes);
  2435.                   dp -= pixel_bytes;
  2436.                }
  2437.                sp -= pixel_bytes;
  2438.             }
  2439.             break;
  2440.          }
  2441.       }
  2442.       row_info->width = final_width;
  2443.       row_info->rowbytes = PNG_ROWBYTES(row_info->pixel_depth, final_width);
  2444.    }
  2445. #if !defined(PNG_READ_PACKSWAP_SUPPORTED)
  2446.    transformations = transformations; /* silence compiler warning */
  2447. #endif
  2448. }
  2449. #endif /* PNG_READ_INTERLACING_SUPPORTED */
  2450. void /* PRIVATE */
  2451. png_read_filter_row(png_structp png_ptr, png_row_infop row_info, png_bytep row,
  2452.    png_bytep prev_row, int filter)
  2453. {
  2454.    png_debug(1, "in png_read_filter_row");
  2455.    png_debug2(2, "row = %lu, filter = %d", png_ptr->row_number, filter);
  2456.    switch (filter)
  2457.    {
  2458.       case PNG_FILTER_VALUE_NONE:
  2459.          break;
  2460.       case PNG_FILTER_VALUE_SUB:
  2461.       {
  2462.          png_uint_32 i;
  2463.          png_uint_32 istop = row_info->rowbytes;
  2464.          png_uint_32 bpp = (row_info->pixel_depth + 7) >> 3;
  2465.          png_bytep rp = row + bpp;
  2466.          png_bytep lp = row;
  2467.          for (i = bpp; i < istop; i++)
  2468.          {
  2469.             *rp = (png_byte)(((int)(*rp) + (int)(*lp++)) & 0xff);
  2470.             rp++;
  2471.          }
  2472.          break;
  2473.       }
  2474.       case PNG_FILTER_VALUE_UP:
  2475.       {
  2476.          png_uint_32 i;
  2477.          png_uint_32 istop = row_info->rowbytes;
  2478.          png_bytep rp = row;
  2479.          png_bytep pp = prev_row;
  2480.          for (i = 0; i < istop; i++)
  2481.          {
  2482.             *rp = (png_byte)(((int)(*rp) + (int)(*pp++)) & 0xff);
  2483.             rp++;
  2484.          }
  2485.          break;
  2486.       }
  2487.       case PNG_FILTER_VALUE_AVG:
  2488.       {
  2489.          png_uint_32 i;
  2490.          png_bytep rp = row;
  2491.          png_bytep pp = prev_row;
  2492.          png_bytep lp = row;
  2493.          png_uint_32 bpp = (row_info->pixel_depth + 7) >> 3;
  2494.          png_uint_32 istop = row_info->rowbytes - bpp;
  2495.          for (i = 0; i < bpp; i++)
  2496.          {
  2497.             *rp = (png_byte)(((int)(*rp) +
  2498.                ((int)(*pp++) / 2 )) & 0xff);
  2499.             rp++;
  2500.          }
  2501.          for (i = 0; i < istop; i++)
  2502.          {
  2503.             *rp = (png_byte)(((int)(*rp) +
  2504.                (int)(*pp++ + *lp++) / 2 ) & 0xff);
  2505.             rp++;
  2506.          }
  2507.          break;
  2508.       }
  2509.       case PNG_FILTER_VALUE_PAETH:
  2510.       {
  2511.          png_uint_32 i;
  2512.          png_bytep rp = row;
  2513.          png_bytep pp = prev_row;
  2514.          png_bytep lp = row;
  2515.          png_bytep cp = prev_row;
  2516.          png_uint_32 bpp = (row_info->pixel_depth + 7) >> 3;
  2517.          png_uint_32 istop=row_info->rowbytes - bpp;
  2518.          for (i = 0; i < bpp; i++)
  2519.          {
  2520.             *rp = (png_byte)(((int)(*rp) + (int)(*pp++)) & 0xff);
  2521.             rp++;
  2522.          }
  2523.          for (i = 0; i < istop; i++)   /* use leftover rp,pp */
  2524.          {
  2525.             int a, b, c, pa, pb, pc, p;
  2526.             a = *lp++;
  2527.             b = *pp++;
  2528.             c = *cp++;
  2529.             p = b - c;
  2530.             pc = a - c;
  2531. #ifdef PNG_USE_ABS
  2532.             pa = abs(p);
  2533.             pb = abs(pc);
  2534.             pc = abs(p + pc);
  2535. #else
  2536.             pa = p < 0 ? -p : p;
  2537.             pb = pc < 0 ? -pc : pc;
  2538.             pc = (p + pc) < 0 ? -(p + pc) : p + pc;
  2539. #endif
  2540.             /*
  2541.                if (pa <= pb && pa <= pc)
  2542.                   p = a;
  2543.                else if (pb <= pc)
  2544.                   p = b;
  2545.                else
  2546.                   p = c;
  2547.              */
  2548.             p = (pa <= pb && pa <= pc) ? a : (pb <= pc) ? b : c;
  2549.             *rp = (png_byte)(((int)(*rp) + p) & 0xff);
  2550.             rp++;
  2551.          }
  2552.          break;
  2553.       }
  2554.       default:
  2555.          png_warning(png_ptr, "Ignoring bad adaptive filter type");
  2556.          *row = 0;
  2557.          break;
  2558.    }
  2559. }
  2560. void /* PRIVATE */
  2561. png_read_finish_row(png_structp png_ptr)
  2562. {
  2563. #ifdef PNG_USE_LOCAL_ARRAYS
  2564. #ifdef PNG_READ_INTERLACING_SUPPORTED
  2565.    /* arrays to facilitate easy interlacing - use pass (0 - 6) as index */
  2566.    /* start of interlace block */
  2567.    PNG_CONST int png_pass_start[7] = {0, 4, 0, 2, 0, 1, 0};
  2568.    /* offset to next interlace block */
  2569.    PNG_CONST int png_pass_inc[7] = {8, 8, 4, 4, 2, 2, 1};
  2570.    /* start of interlace block in the y direction */
  2571.    PNG_CONST int png_pass_ystart[7] = {0, 0, 4, 0, 2, 0, 1};
  2572.    /* offset to next interlace block in the y direction */
  2573.    PNG_CONST int png_pass_yinc[7] = {8, 8, 8, 4, 4, 2, 2};
  2574. #endif /* PNG_READ_INTERLACING_SUPPORTED */
  2575. #endif
  2576.    png_debug(1, "in png_read_finish_row");
  2577.    png_ptr->row_number++;
  2578.    if (png_ptr->row_number < png_ptr->num_rows)
  2579.       return;
  2580. #ifdef PNG_READ_INTERLACING_SUPPORTED
  2581.    if (png_ptr->interlaced)
  2582.    {
  2583.       png_ptr->row_number = 0;
  2584.       png_memset_check(png_ptr, png_ptr->prev_row, 0,
  2585.          png_ptr->rowbytes + 1);
  2586.       do
  2587.       {
  2588.          png_ptr->pass++;
  2589.          if (png_ptr->pass >= 7)
  2590.             break;
  2591.          png_ptr->iwidth = (png_ptr->width +
  2592.             png_pass_inc[png_ptr->pass] - 1 -
  2593.             png_pass_start[png_ptr->pass]) /
  2594.             png_pass_inc[png_ptr->pass];
  2595.          png_ptr->irowbytes = PNG_ROWBYTES(png_ptr->pixel_depth,
  2596.             png_ptr->iwidth) + 1;
  2597.          if (!(png_ptr->transformations & PNG_INTERLACE))
  2598.          {
  2599.             png_ptr->num_rows = (png_ptr->height +
  2600.                png_pass_yinc[png_ptr->pass] - 1 -
  2601.                png_pass_ystart[png_ptr->pass]) /
  2602.                png_pass_yinc[png_ptr->pass];
  2603.             if (!(png_ptr->num_rows))
  2604.                continue;
  2605.          }
  2606.          else  /* if (png_ptr->transformations & PNG_INTERLACE) */
  2607.             break;
  2608.       } while (png_ptr->iwidth == 0);
  2609.       if (png_ptr->pass < 7)
  2610.          return;
  2611.    }
  2612. #endif /* PNG_READ_INTERLACING_SUPPORTED */
  2613.    if (!(png_ptr->flags & PNG_FLAG_ZLIB_FINISHED))
  2614.    {
  2615. #ifdef PNG_USE_LOCAL_ARRAYS
  2616.       PNG_CONST PNG_IDAT;
  2617. #endif
  2618.       char extra;
  2619.       int ret;
  2620.       png_ptr->zstream.next_out = (Byte *)&extra;
  2621.       png_ptr->zstream.avail_out = (uInt)1;
  2622.       for (;;)
  2623.       {
  2624.          if (!(png_ptr->zstream.avail_in))
  2625.          {
  2626.             while (!png_ptr->idat_size)
  2627.             {
  2628.                png_byte chunk_length[4];
  2629.                png_crc_finish(png_ptr, 0);
  2630.                png_read_data(png_ptr, chunk_length, 4);
  2631.                png_ptr->idat_size = png_get_uint_31(png_ptr, chunk_length);
  2632.                png_reset_crc(png_ptr);
  2633.                png_crc_read(png_ptr, png_ptr->chunk_name, 4);
  2634.                if (png_memcmp(png_ptr->chunk_name, png_IDAT, 4))
  2635.                   png_error(png_ptr, "Not enough image data");
  2636.             }
  2637.             png_ptr->zstream.avail_in = (uInt)png_ptr->zbuf_size;
  2638.             png_ptr->zstream.next_in = png_ptr->zbuf;
  2639.             if (png_ptr->zbuf_size > png_ptr->idat_size)
  2640.                png_ptr->zstream.avail_in = (uInt)png_ptr->idat_size;
  2641.             png_crc_read(png_ptr, png_ptr->zbuf, png_ptr->zstream.avail_in);
  2642.             png_ptr->idat_size -= png_ptr->zstream.avail_in;
  2643.          }
  2644.          ret = inflate(&png_ptr->zstream, Z_PARTIAL_FLUSH);
  2645.          if (ret == Z_STREAM_END)
  2646.          {
  2647.             if (!(png_ptr->zstream.avail_out) || png_ptr->zstream.avail_in ||
  2648.                png_ptr->idat_size)
  2649.                png_warning(png_ptr, "Extra compressed data");
  2650.             png_ptr->mode |= PNG_AFTER_IDAT;
  2651.             png_ptr->flags |= PNG_FLAG_ZLIB_FINISHED;
  2652.             break;
  2653.          }
  2654.          if (ret != Z_OK)
  2655.             png_error(png_ptr, png_ptr->zstream.msg ? png_ptr->zstream.msg :
  2656.                       "Decompression Error");
  2657.          if (!(png_ptr->zstream.avail_out))
  2658.          {
  2659.             png_warning(png_ptr, "Extra compressed data.");
  2660.             png_ptr->mode |= PNG_AFTER_IDAT;
  2661.             png_ptr->flags |= PNG_FLAG_ZLIB_FINISHED;
  2662.             break;
  2663.          }
  2664.       }
  2665.       png_ptr->zstream.avail_out = 0;
  2666.    }
  2667.    if (png_ptr->idat_size || png_ptr->zstream.avail_in)
  2668.       png_warning(png_ptr, "Extra compression data");
  2669.    inflateReset(&png_ptr->zstream);
  2670.    png_ptr->mode |= PNG_AFTER_IDAT;
  2671. }
  2672. void /* PRIVATE */
  2673. png_read_start_row(png_structp png_ptr)
  2674. {
  2675. #ifdef PNG_USE_LOCAL_ARRAYS
  2676. #ifdef PNG_READ_INTERLACING_SUPPORTED
  2677.    /* arrays to facilitate easy interlacing - use pass (0 - 6) as index */
  2678.    /* start of interlace block */
  2679.    PNG_CONST int png_pass_start[7] = {0, 4, 0, 2, 0, 1, 0};
  2680.    /* offset to next interlace block */
  2681.    PNG_CONST int png_pass_inc[7] = {8, 8, 4, 4, 2, 2, 1};
  2682.    /* start of interlace block in the y direction */
  2683.    PNG_CONST int png_pass_ystart[7] = {0, 0, 4, 0, 2, 0, 1};
  2684.    /* offset to next interlace block in the y direction */
  2685.    PNG_CONST int png_pass_yinc[7] = {8, 8, 8, 4, 4, 2, 2};
  2686. #endif
  2687. #endif
  2688.    int max_pixel_depth;
  2689.    png_size_t row_bytes;
  2690.    png_debug(1, "in png_read_start_row");
  2691.    png_ptr->zstream.avail_in = 0;
  2692.    png_init_read_transformations(png_ptr);
  2693. #ifdef PNG_READ_INTERLACING_SUPPORTED
  2694.    if (png_ptr->interlaced)
  2695.    {
  2696.       if (!(png_ptr->transformations & PNG_INTERLACE))
  2697.          png_ptr->num_rows = (png_ptr->height + png_pass_yinc[0] - 1 -
  2698.             png_pass_ystart[0]) / png_pass_yinc[0];
  2699.       else
  2700.          png_ptr->num_rows = png_ptr->height;
  2701.       png_ptr->iwidth = (png_ptr->width +
  2702.          png_pass_inc[png_ptr->pass] - 1 -
  2703.          png_pass_start[png_ptr->pass]) /
  2704.          png_pass_inc[png_ptr->pass];
  2705.          png_ptr->irowbytes =
  2706.             PNG_ROWBYTES(png_ptr->pixel_depth, png_ptr->iwidth) + 1;
  2707.    }
  2708.    else
  2709. #endif /* PNG_READ_INTERLACING_SUPPORTED */
  2710.    {
  2711.       png_ptr->num_rows = png_ptr->height;
  2712.       png_ptr->iwidth = png_ptr->width;
  2713.       png_ptr->irowbytes = png_ptr->rowbytes + 1;
  2714.    }
  2715.    max_pixel_depth = png_ptr->pixel_depth;
  2716. #if defined(PNG_READ_PACK_SUPPORTED)
  2717.    if ((png_ptr->transformations & PNG_PACK) && png_ptr->bit_depth < 8)
  2718.       max_pixel_depth = 8;
  2719. #endif
  2720. #if defined(PNG_READ_EXPAND_SUPPORTED)
  2721.    if (png_ptr->transformations & PNG_EXPAND)
  2722.    {
  2723.       if (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE)
  2724.       {
  2725.          if (png_ptr->num_trans)
  2726.             max_pixel_depth = 32;
  2727.          else
  2728.             max_pixel_depth = 24;
  2729.       }
  2730.       else if (png_ptr->color_type == PNG_COLOR_TYPE_GRAY)
  2731.       {
  2732.          if (max_pixel_depth < 8)
  2733.             max_pixel_depth = 8;
  2734.          if (png_ptr->num_trans)
  2735.             max_pixel_depth *= 2;
  2736.       }
  2737.       else if (png_ptr->color_type == PNG_COLOR_TYPE_RGB)
  2738.       {
  2739.          if (png_ptr->num_trans)
  2740.          {
  2741.             max_pixel_depth *= 4;
  2742.             max_pixel_depth /= 3;
  2743.          }
  2744.       }
  2745.    }
  2746. #endif
  2747. #if defined(PNG_READ_FILLER_SUPPORTED)
  2748.    if (png_ptr->transformations & (PNG_FILLER))
  2749.    {
  2750.       if (png_ptr->color_type == PNG_COLOR_TYPE_PALETTE)
  2751.          max_pixel_depth = 32;
  2752.       else if (png_ptr->color_type == PNG_COLOR_TYPE_GRAY)
  2753.       {
  2754.          if (max_pixel_depth <= 8)
  2755.             max_pixel_depth = 16;
  2756.          else
  2757.             max_pixel_depth = 32;
  2758.       }
  2759.       else if (png_ptr->color_type == PNG_COLOR_TYPE_RGB)
  2760.       {
  2761.          if (max_pixel_depth <= 32)
  2762.             max_pixel_depth = 32;
  2763.          else
  2764.             max_pixel_depth = 64;
  2765.       }
  2766.    }
  2767. #endif
  2768. #if defined(PNG_READ_GRAY_TO_RGB_SUPPORTED)
  2769.    if (png_ptr->transformations & PNG_GRAY_TO_RGB)
  2770.    {
  2771.       if (
  2772. #if defined(PNG_READ_EXPAND_SUPPORTED)
  2773.         (png_ptr->num_trans && (png_ptr->transformations & PNG_EXPAND)) ||
  2774. #endif
  2775. #if defined(PNG_READ_FILLER_SUPPORTED)
  2776.         (png_ptr->transformations & (PNG_FILLER)) ||
  2777. #endif
  2778.         png_ptr->color_type == PNG_COLOR_TYPE_GRAY_ALPHA)
  2779.       {
  2780.          if (max_pixel_depth <= 16)
  2781.             max_pixel_depth = 32;
  2782.          else
  2783.             max_pixel_depth = 64;
  2784.       }
  2785.       else
  2786.       {
  2787.          if (max_pixel_depth <= 8)
  2788.            {
  2789.              if (png_ptr->color_type == PNG_COLOR_TYPE_RGB_ALPHA)
  2790.                max_pixel_depth = 32;
  2791.              else
  2792.                max_pixel_depth = 24;
  2793.            }
  2794.          else if (png_ptr->color_type == PNG_COLOR_TYPE_RGB_ALPHA)
  2795.             max_pixel_depth = 64;
  2796.          else
  2797.             max_pixel_depth = 48;
  2798.       }
  2799.    }
  2800. #endif
  2801. #if defined(PNG_READ_USER_TRANSFORM_SUPPORTED) && 
  2802. defined(PNG_USER_TRANSFORM_PTR_SUPPORTED)
  2803.    if (png_ptr->transformations & PNG_USER_TRANSFORM)
  2804.      {
  2805.        int user_pixel_depth = png_ptr->user_transform_depth*
  2806.          png_ptr->user_transform_channels;
  2807.        if (user_pixel_depth > max_pixel_depth)
  2808.          max_pixel_depth=user_pixel_depth;
  2809.      }
  2810. #endif
  2811.    /* align the width on the next larger 8 pixels.  Mainly used
  2812.       for interlacing */
  2813.    row_bytes = ((png_ptr->width + 7) & ~((png_uint_32)7));
  2814.    /* calculate the maximum bytes needed, adding a byte and a pixel
  2815.       for safety's sake */
  2816.    row_bytes = PNG_ROWBYTES(max_pixel_depth, row_bytes) +
  2817.       1 + ((max_pixel_depth + 7) >> 3);
  2818. #ifdef PNG_MAX_MALLOC_64K
  2819.    if (row_bytes > (png_uint_32)65536L)
  2820.       png_error(png_ptr, "This image requires a row greater than 64KB");
  2821. #endif
  2822.    if (row_bytes + 64 > png_ptr->old_big_row_buf_size)
  2823.    {
  2824.      png_free(png_ptr, png_ptr->big_row_buf);
  2825.      png_ptr->big_row_buf = (png_bytep)png_malloc(png_ptr, (png_uint_32)(row_bytes+64));
  2826.      png_ptr->row_buf = png_ptr->big_row_buf+32;
  2827.      png_ptr->old_big_row_buf_size = (png_uint_32)(row_bytes+64);
  2828.    }
  2829. #ifdef PNG_MAX_MALLOC_64K
  2830.    if ((png_uint_32)png_ptr->rowbytes + 1 > (png_uint_32)65536L)
  2831.       png_error(png_ptr, "This image requires a row greater than 64KB");
  2832. #endif
  2833.    if ((png_uint_32)png_ptr->rowbytes > (png_uint_32)(PNG_SIZE_MAX - 1))
  2834.       png_error(png_ptr, "Row has too many bytes to allocate in memory.");
  2835.    if (png_ptr->rowbytes+1 > png_ptr->old_prev_row_size)
  2836.    {
  2837.      png_free(png_ptr, png_ptr->prev_row);
  2838.      png_ptr->prev_row = (png_bytep)png_malloc(png_ptr, (png_uint_32)(
  2839.         png_ptr->rowbytes + 1));
  2840.      png_ptr->old_prev_row_size = png_ptr->rowbytes+1;
  2841.    }
  2842.    png_memset_check(png_ptr, png_ptr->prev_row, 0, png_ptr->rowbytes + 1);
  2843.    png_debug1(3, "width = %lu,", png_ptr->width);
  2844.    png_debug1(3, "height = %lu,", png_ptr->height);
  2845.    png_debug1(3, "iwidth = %lu,", png_ptr->iwidth);
  2846.    png_debug1(3, "num_rows = %lu", png_ptr->num_rows);
  2847.    png_debug1(3, "rowbytes = %lu,", png_ptr->rowbytes);
  2848.    png_debug1(3, "irowbytes = %lu,", png_ptr->irowbytes);
  2849.    png_ptr->flags |= PNG_FLAG_ROW_INIT;
  2850. }
  2851. #endif /* PNG_READ_SUPPORTED */