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

界面编程

开发平台:

Visual C++

  1. /* pngwrite.c - general routines to write a PNG file
  2.  *
  3.  * Last changed in libpng 1.2.31 [August 19, 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. #if ( ! defined _CRT_SECURE_NO_WARNINGS )
  10. #define _CRT_SECURE_NO_WARNINGS
  11. #endif
  12. /* get internal access to png.h */
  13. #define PNG_INTERNAL
  14. #include "png.h"
  15. #ifdef PNG_WRITE_SUPPORTED
  16. /* Writes all the PNG information.  This is the suggested way to use the
  17.  * library.  If you have a new chunk to add, make a function to write it,
  18.  * and put it in the correct location here.  If you want the chunk written
  19.  * after the image data, put it in png_write_end().  I strongly encourage
  20.  * you to supply a PNG_INFO_ flag, and check info_ptr->valid before writing
  21.  * the chunk, as that will keep the code from breaking if you want to just
  22.  * write a plain PNG file.  If you have long comments, I suggest writing
  23.  * them in png_write_end(), and compressing them.
  24.  */
  25. void PNGAPI
  26. png_write_info_before_PLTE(png_structp png_ptr, png_infop info_ptr)
  27. {
  28.    png_debug(1, "in png_write_info_before_PLTE");
  29.    if (png_ptr == NULL || info_ptr == NULL)
  30.       return;
  31.    if (!(png_ptr->mode & PNG_WROTE_INFO_BEFORE_PLTE))
  32.    {
  33.    png_write_sig(png_ptr); /* write PNG signature */
  34. #if defined(PNG_MNG_FEATURES_SUPPORTED)
  35.    if ((png_ptr->mode&PNG_HAVE_PNG_SIGNATURE)&&(png_ptr->mng_features_permitted))
  36.    {
  37.       png_warning(png_ptr, "MNG features are not allowed in a PNG datastream");
  38.       png_ptr->mng_features_permitted=0;
  39.    }
  40. #endif
  41.    /* write IHDR information. */
  42.    png_write_IHDR(png_ptr, info_ptr->width, info_ptr->height,
  43.       info_ptr->bit_depth, info_ptr->color_type, info_ptr->compression_type,
  44.       info_ptr->filter_type,
  45. #if defined(PNG_WRITE_INTERLACING_SUPPORTED)
  46.       info_ptr->interlace_type);
  47. #else
  48.       0);
  49. #endif
  50.    /* the rest of these check to see if the valid field has the appropriate
  51.       flag set, and if it does, writes the chunk. */
  52. #if defined(PNG_WRITE_gAMA_SUPPORTED)
  53.    if (info_ptr->valid & PNG_INFO_gAMA)
  54.    {
  55. #  ifdef PNG_FLOATING_POINT_SUPPORTED
  56.       png_write_gAMA(png_ptr, info_ptr->gamma);
  57. #else
  58. #ifdef PNG_FIXED_POINT_SUPPORTED
  59.       png_write_gAMA_fixed(png_ptr, info_ptr->int_gamma);
  60. #  endif
  61. #endif
  62.    }
  63. #endif
  64. #if defined(PNG_WRITE_sRGB_SUPPORTED)
  65.    if (info_ptr->valid & PNG_INFO_sRGB)
  66.       png_write_sRGB(png_ptr, (int)info_ptr->srgb_intent);
  67. #endif
  68. #if defined(PNG_WRITE_iCCP_SUPPORTED)
  69.    if (info_ptr->valid & PNG_INFO_iCCP)
  70.       png_write_iCCP(png_ptr, info_ptr->iccp_name, PNG_COMPRESSION_TYPE_BASE,
  71.                      info_ptr->iccp_profile, (int)info_ptr->iccp_proflen);
  72. #endif
  73. #if defined(PNG_WRITE_sBIT_SUPPORTED)
  74.    if (info_ptr->valid & PNG_INFO_sBIT)
  75.       png_write_sBIT(png_ptr, &(info_ptr->sig_bit), info_ptr->color_type);
  76. #endif
  77. #if defined(PNG_WRITE_cHRM_SUPPORTED)
  78.    if (info_ptr->valid & PNG_INFO_cHRM)
  79.    {
  80. #ifdef PNG_FLOATING_POINT_SUPPORTED
  81.       png_write_cHRM(png_ptr,
  82.          info_ptr->x_white, info_ptr->y_white,
  83.          info_ptr->x_red, info_ptr->y_red,
  84.          info_ptr->x_green, info_ptr->y_green,
  85.          info_ptr->x_blue, info_ptr->y_blue);
  86. #else
  87. #  ifdef PNG_FIXED_POINT_SUPPORTED
  88.       png_write_cHRM_fixed(png_ptr,
  89.          info_ptr->int_x_white, info_ptr->int_y_white,
  90.          info_ptr->int_x_red, info_ptr->int_y_red,
  91.          info_ptr->int_x_green, info_ptr->int_y_green,
  92.          info_ptr->int_x_blue, info_ptr->int_y_blue);
  93. #  endif
  94. #endif
  95.    }
  96. #endif
  97. #if defined(PNG_WRITE_UNKNOWN_CHUNKS_SUPPORTED)
  98.    if (info_ptr->unknown_chunks_num)
  99.    {
  100.        png_unknown_chunk *up;
  101.        png_debug(5, "writing extra chunks");
  102.        for (up = info_ptr->unknown_chunks;
  103.             up < info_ptr->unknown_chunks + info_ptr->unknown_chunks_num;
  104.             up++)
  105.        {
  106.          int keep=png_handle_as_unknown(png_ptr, up->name);
  107.          if (keep != PNG_HANDLE_CHUNK_NEVER &&
  108.             up->location && !(up->location & PNG_HAVE_PLTE) &&
  109.             !(up->location & PNG_HAVE_IDAT) &&
  110.             ((up->name[3] & 0x20) || keep == PNG_HANDLE_CHUNK_ALWAYS ||
  111.             (png_ptr->flags & PNG_FLAG_KEEP_UNSAFE_CHUNKS)))
  112.          {
  113.             if (up->size == 0)
  114.                png_warning(png_ptr, "Writing zero-length unknown chunk");
  115.             png_write_chunk(png_ptr, up->name, up->data, up->size);
  116.          }
  117.        }
  118.    }
  119. #endif
  120.       png_ptr->mode |= PNG_WROTE_INFO_BEFORE_PLTE;
  121.    }
  122. }
  123. void PNGAPI
  124. png_write_info(png_structp png_ptr, png_infop info_ptr)
  125. {
  126. #if defined(PNG_WRITE_TEXT_SUPPORTED) || defined(PNG_WRITE_sPLT_SUPPORTED)
  127.    int i;
  128. #endif
  129.    png_debug(1, "in png_write_info");
  130.    if (png_ptr == NULL || info_ptr == NULL)
  131.       return;
  132.    png_write_info_before_PLTE(png_ptr, info_ptr);
  133.    if (info_ptr->valid & PNG_INFO_PLTE)
  134.       png_write_PLTE(png_ptr, info_ptr->palette,
  135.          (png_uint_32)info_ptr->num_palette);
  136.    else if (info_ptr->color_type == PNG_COLOR_TYPE_PALETTE)
  137.       png_error(png_ptr, "Valid palette required for paletted images");
  138. #if defined(PNG_WRITE_tRNS_SUPPORTED)
  139.    if (info_ptr->valid & PNG_INFO_tRNS)
  140.       {
  141. #if defined(PNG_WRITE_INVERT_ALPHA_SUPPORTED)
  142.          /* invert the alpha channel (in tRNS) */
  143.          if ((png_ptr->transformations & PNG_INVERT_ALPHA) &&
  144.             info_ptr->color_type == PNG_COLOR_TYPE_PALETTE)
  145.          {
  146.             int j;
  147.             for (j=0; j<(int)info_ptr->num_trans; j++)
  148.                info_ptr->trans[j] = (png_byte)(255 - info_ptr->trans[j]);
  149.          }
  150. #endif
  151.       png_write_tRNS(png_ptr, info_ptr->trans, &(info_ptr->trans_values),
  152.          info_ptr->num_trans, info_ptr->color_type);
  153.       }
  154. #endif
  155. #if defined(PNG_WRITE_bKGD_SUPPORTED)
  156.    if (info_ptr->valid & PNG_INFO_bKGD)
  157.       png_write_bKGD(png_ptr, &(info_ptr->background), info_ptr->color_type);
  158. #endif
  159. #if defined(PNG_WRITE_hIST_SUPPORTED)
  160.    if (info_ptr->valid & PNG_INFO_hIST)
  161.       png_write_hIST(png_ptr, info_ptr->hist, info_ptr->num_palette);
  162. #endif
  163. #if defined(PNG_WRITE_oFFs_SUPPORTED)
  164.    if (info_ptr->valid & PNG_INFO_oFFs)
  165.       png_write_oFFs(png_ptr, info_ptr->x_offset, info_ptr->y_offset,
  166.          info_ptr->offset_unit_type);
  167. #endif
  168. #if defined(PNG_WRITE_pCAL_SUPPORTED)
  169.    if (info_ptr->valid & PNG_INFO_pCAL)
  170.       png_write_pCAL(png_ptr, info_ptr->pcal_purpose, info_ptr->pcal_X0,
  171.          info_ptr->pcal_X1, info_ptr->pcal_type, info_ptr->pcal_nparams,
  172.          info_ptr->pcal_units, info_ptr->pcal_params);
  173. #endif
  174. #if defined(PNG_WRITE_sCAL_SUPPORTED)
  175.    if (info_ptr->valid & PNG_INFO_sCAL)
  176. #if defined(PNG_FLOATING_POINT_SUPPORTED) && !defined(PNG_NO_STDIO)
  177.       png_write_sCAL(png_ptr, (int)info_ptr->scal_unit,
  178.           info_ptr->scal_pixel_width, info_ptr->scal_pixel_height);
  179. #else
  180. #ifdef PNG_FIXED_POINT_SUPPORTED
  181.       png_write_sCAL_s(png_ptr, (int)info_ptr->scal_unit,
  182.           info_ptr->scal_s_width, info_ptr->scal_s_height);
  183. #else
  184.       png_warning(png_ptr,
  185.           "png_write_sCAL not supported; sCAL chunk not written.");
  186. #endif
  187. #endif
  188. #endif
  189. #if defined(PNG_WRITE_pHYs_SUPPORTED)
  190.    if (info_ptr->valid & PNG_INFO_pHYs)
  191.       png_write_pHYs(png_ptr, info_ptr->x_pixels_per_unit,
  192.          info_ptr->y_pixels_per_unit, info_ptr->phys_unit_type);
  193. #endif
  194. #if defined(PNG_WRITE_tIME_SUPPORTED)
  195.    if (info_ptr->valid & PNG_INFO_tIME)
  196.    {
  197.       png_write_tIME(png_ptr, &(info_ptr->mod_time));
  198.       png_ptr->mode |= PNG_WROTE_tIME;
  199.    }
  200. #endif
  201. #if defined(PNG_WRITE_sPLT_SUPPORTED)
  202.    if (info_ptr->valid & PNG_INFO_sPLT)
  203.      for (i = 0; i < (int)info_ptr->splt_palettes_num; i++)
  204.        png_write_sPLT(png_ptr, info_ptr->splt_palettes + i);
  205. #endif
  206. #if defined(PNG_WRITE_TEXT_SUPPORTED)
  207.    /* Check to see if we need to write text chunks */
  208.    for (i = 0; i < info_ptr->num_text; i++)
  209.    {
  210.       png_debug2(2, "Writing header text chunk %d, type %d", i,
  211.          info_ptr->text[i].compression);
  212.       /* an internationalized chunk? */
  213.       if (info_ptr->text[i].compression > 0)
  214.       {
  215. #if defined(PNG_WRITE_iTXt_SUPPORTED)
  216.           /* write international chunk */
  217.           png_write_iTXt(png_ptr,
  218.                          info_ptr->text[i].compression,
  219.                          info_ptr->text[i].key,
  220.                          info_ptr->text[i].lang,
  221.                          info_ptr->text[i].lang_key,
  222.                          info_ptr->text[i].text);
  223. #else
  224.           png_warning(png_ptr, "Unable to write international text");
  225. #endif
  226.           /* Mark this chunk as written */
  227.           info_ptr->text[i].compression = PNG_TEXT_COMPRESSION_NONE_WR;
  228.       }
  229.       /* If we want a compressed text chunk */
  230.       else if (info_ptr->text[i].compression == PNG_TEXT_COMPRESSION_zTXt)
  231.       {
  232. #if defined(PNG_WRITE_zTXt_SUPPORTED)
  233.          /* write compressed chunk */
  234.          png_write_zTXt(png_ptr, info_ptr->text[i].key,
  235.             info_ptr->text[i].text, 0,
  236.             info_ptr->text[i].compression);
  237. #else
  238.          png_warning(png_ptr, "Unable to write compressed text");
  239. #endif
  240.          /* Mark this chunk as written */
  241.          info_ptr->text[i].compression = PNG_TEXT_COMPRESSION_zTXt_WR;
  242.       }
  243.       else if (info_ptr->text[i].compression == PNG_TEXT_COMPRESSION_NONE)
  244.       {
  245. #if defined(PNG_WRITE_tEXt_SUPPORTED)
  246.          /* write uncompressed chunk */
  247.          png_write_tEXt(png_ptr, info_ptr->text[i].key,
  248.                          info_ptr->text[i].text,
  249.                          0);
  250. #else
  251.          png_warning(png_ptr, "Unable to write uncompressed text");
  252. #endif
  253.          /* Mark this chunk as written */
  254.          info_ptr->text[i].compression = PNG_TEXT_COMPRESSION_NONE_WR;
  255.       }
  256.    }
  257. #endif
  258. #if defined(PNG_WRITE_UNKNOWN_CHUNKS_SUPPORTED)
  259.    if (info_ptr->unknown_chunks_num)
  260.    {
  261.        png_unknown_chunk *up;
  262.        png_debug(5, "writing extra chunks");
  263.        for (up = info_ptr->unknown_chunks;
  264.             up < info_ptr->unknown_chunks + info_ptr->unknown_chunks_num;
  265.             up++)
  266.        {
  267.          int keep=png_handle_as_unknown(png_ptr, up->name);
  268.          if (keep != PNG_HANDLE_CHUNK_NEVER &&
  269.             up->location && (up->location & PNG_HAVE_PLTE) &&
  270.             !(up->location & PNG_HAVE_IDAT) &&
  271.             ((up->name[3] & 0x20) || keep == PNG_HANDLE_CHUNK_ALWAYS ||
  272.             (png_ptr->flags & PNG_FLAG_KEEP_UNSAFE_CHUNKS)))
  273.          {
  274.             png_write_chunk(png_ptr, up->name, up->data, up->size);
  275.          }
  276.        }
  277.    }
  278. #endif
  279. }
  280. /* Writes the end of the PNG file.  If you don't want to write comments or
  281.  * time information, you can pass NULL for info.  If you already wrote these
  282.  * in png_write_info(), do not write them again here.  If you have long
  283.  * comments, I suggest writing them here, and compressing them.
  284.  */
  285. void PNGAPI
  286. png_write_end(png_structp png_ptr, png_infop info_ptr)
  287. {
  288.    png_debug(1, "in png_write_end");
  289.    if (png_ptr == NULL)
  290.       return;
  291.    if (!(png_ptr->mode & PNG_HAVE_IDAT))
  292.       png_error(png_ptr, "No IDATs written into file");
  293.    /* see if user wants us to write information chunks */
  294.    if (info_ptr != NULL)
  295.    {
  296. #if defined(PNG_WRITE_TEXT_SUPPORTED)
  297.       int i; /* local index variable */
  298. #endif
  299. #if defined(PNG_WRITE_tIME_SUPPORTED)
  300.       /* check to see if user has supplied a time chunk */
  301.       if ((info_ptr->valid & PNG_INFO_tIME) &&
  302.          !(png_ptr->mode & PNG_WROTE_tIME))
  303.          png_write_tIME(png_ptr, &(info_ptr->mod_time));
  304. #endif
  305. #if defined(PNG_WRITE_TEXT_SUPPORTED)
  306.       /* loop through comment chunks */
  307.       for (i = 0; i < info_ptr->num_text; i++)
  308.       {
  309.          png_debug2(2, "Writing trailer text chunk %d, type %d", i,
  310.             info_ptr->text[i].compression);
  311.          /* an internationalized chunk? */
  312.          if (info_ptr->text[i].compression > 0)
  313.          {
  314. #if defined(PNG_WRITE_iTXt_SUPPORTED)
  315.              /* write international chunk */
  316.              png_write_iTXt(png_ptr,
  317.                          info_ptr->text[i].compression,
  318.                          info_ptr->text[i].key,
  319.                          info_ptr->text[i].lang,
  320.                          info_ptr->text[i].lang_key,
  321.                          info_ptr->text[i].text);
  322. #else
  323.              png_warning(png_ptr, "Unable to write international text");
  324. #endif
  325.              /* Mark this chunk as written */
  326.              info_ptr->text[i].compression = PNG_TEXT_COMPRESSION_NONE_WR;
  327.          }
  328.          else if (info_ptr->text[i].compression >= PNG_TEXT_COMPRESSION_zTXt)
  329.          {
  330. #if defined(PNG_WRITE_zTXt_SUPPORTED)
  331.             /* write compressed chunk */
  332.             png_write_zTXt(png_ptr, info_ptr->text[i].key,
  333.                info_ptr->text[i].text, 0,
  334.                info_ptr->text[i].compression);
  335. #else
  336.             png_warning(png_ptr, "Unable to write compressed text");
  337. #endif
  338.             /* Mark this chunk as written */
  339.             info_ptr->text[i].compression = PNG_TEXT_COMPRESSION_zTXt_WR;
  340.          }
  341.          else if (info_ptr->text[i].compression == PNG_TEXT_COMPRESSION_NONE)
  342.          {
  343. #if defined(PNG_WRITE_tEXt_SUPPORTED)
  344.             /* write uncompressed chunk */
  345.             png_write_tEXt(png_ptr, info_ptr->text[i].key,
  346.                info_ptr->text[i].text, 0);
  347. #else
  348.             png_warning(png_ptr, "Unable to write uncompressed text");
  349. #endif
  350.             /* Mark this chunk as written */
  351.             info_ptr->text[i].compression = PNG_TEXT_COMPRESSION_NONE_WR;
  352.          }
  353.       }
  354. #endif
  355. #if defined(PNG_WRITE_UNKNOWN_CHUNKS_SUPPORTED)
  356.    if (info_ptr->unknown_chunks_num)
  357.    {
  358.        png_unknown_chunk *up;
  359.        png_debug(5, "writing extra chunks");
  360.        for (up = info_ptr->unknown_chunks;
  361.             up < info_ptr->unknown_chunks + info_ptr->unknown_chunks_num;
  362.             up++)
  363.        {
  364.          int keep=png_handle_as_unknown(png_ptr, up->name);
  365.          if (keep != PNG_HANDLE_CHUNK_NEVER &&
  366.             up->location && (up->location & PNG_AFTER_IDAT) &&
  367.             ((up->name[3] & 0x20) || keep == PNG_HANDLE_CHUNK_ALWAYS ||
  368.             (png_ptr->flags & PNG_FLAG_KEEP_UNSAFE_CHUNKS)))
  369.          {
  370.             png_write_chunk(png_ptr, up->name, up->data, up->size);
  371.          }
  372.        }
  373.    }
  374. #endif
  375.    }
  376.    png_ptr->mode |= PNG_AFTER_IDAT;
  377.    /* write end of PNG file */
  378.    png_write_IEND(png_ptr);
  379.    /* This flush, added in libpng-1.0.8, removed from libpng-1.0.9beta03,
  380.     * and restored again in libpng-1.2.30, may cause some applications that
  381.     * do not set png_ptr->output_flush_fn to crash.  If your application
  382.     * experiences a problem, please try building libpng with
  383.     * PNG_WRITE_FLUSH_AFTER_IEND_SUPPORTED defined, and report the event to
  384.     * png-mng-implement at lists.sf.net .  This kludge will be removed
  385.     * from libpng-1.4.0.
  386.     */
  387. #if defined(PNG_WRITE_FLUSH_SUPPORTED) && 
  388.     defined(PNG_WRITE_FLUSH_AFTER_IEND_SUPPORTED)
  389.    png_flush(png_ptr);
  390. #endif
  391. }
  392. #if defined(PNG_WRITE_tIME_SUPPORTED)
  393. #if !defined(_WIN32_WCE)
  394. /* "time.h" functions are not supported on WindowsCE */
  395. void PNGAPI
  396. png_convert_from_struct_tm(png_timep ptime, struct tm FAR * ttime)
  397. {
  398.    png_debug(1, "in png_convert_from_struct_tm");
  399.    ptime->year = (png_uint_16)(1900 + ttime->tm_year);
  400.    ptime->month = (png_byte)(ttime->tm_mon + 1);
  401.    ptime->day = (png_byte)ttime->tm_mday;
  402.    ptime->hour = (png_byte)ttime->tm_hour;
  403.    ptime->minute = (png_byte)ttime->tm_min;
  404.    ptime->second = (png_byte)ttime->tm_sec;
  405. }
  406. void PNGAPI
  407. png_convert_from_time_t(png_timep ptime, time_t ttime)
  408. {
  409.    struct tm *tbuf;
  410.    png_debug(1, "in png_convert_from_time_t");
  411.    tbuf = gmtime(&ttime);
  412.    png_convert_from_struct_tm(ptime, tbuf);
  413. }
  414. #endif
  415. #endif
  416. /* Initialize png_ptr structure, and allocate any memory needed */
  417. png_structp PNGAPI
  418. png_create_write_struct(png_const_charp user_png_ver, png_voidp error_ptr,
  419.    png_error_ptr error_fn, png_error_ptr warn_fn)
  420. {
  421. #ifdef PNG_USER_MEM_SUPPORTED
  422.    return (png_create_write_struct_2(user_png_ver, error_ptr, error_fn,
  423.       warn_fn, png_voidp_NULL, png_malloc_ptr_NULL, png_free_ptr_NULL));
  424. }
  425. /* Alternate initialize png_ptr structure, and allocate any memory needed */
  426. png_structp PNGAPI
  427. png_create_write_struct_2(png_const_charp user_png_ver, png_voidp error_ptr,
  428.    png_error_ptr error_fn, png_error_ptr warn_fn, png_voidp mem_ptr,
  429.    png_malloc_ptr malloc_fn, png_free_ptr free_fn)
  430. {
  431. #endif /* PNG_USER_MEM_SUPPORTED */
  432. #ifdef PNG_SETJMP_SUPPORTED
  433.     volatile
  434. #endif
  435.     png_structp png_ptr;
  436. #ifdef PNG_SETJMP_SUPPORTED
  437. #ifdef USE_FAR_KEYWORD
  438.    jmp_buf jmpbuf;
  439. #endif
  440. #endif
  441.    int i;
  442.    png_debug(1, "in png_create_write_struct");
  443. #ifdef PNG_USER_MEM_SUPPORTED
  444.    png_ptr = (png_structp)png_create_struct_2(PNG_STRUCT_PNG,
  445.       (png_malloc_ptr)malloc_fn, (png_voidp)mem_ptr);
  446. #else
  447.    png_ptr = (png_structp)png_create_struct(PNG_STRUCT_PNG);
  448. #endif /* PNG_USER_MEM_SUPPORTED */
  449.    if (png_ptr == NULL)
  450.       return (NULL);
  451.    /* added at libpng-1.2.6 */
  452. #ifdef PNG_SET_USER_LIMITS_SUPPORTED
  453.    png_ptr->user_width_max=PNG_USER_WIDTH_MAX;
  454.    png_ptr->user_height_max=PNG_USER_HEIGHT_MAX;
  455. #endif
  456. #ifdef PNG_SETJMP_SUPPORTED
  457. #ifdef USE_FAR_KEYWORD
  458.    if (setjmp(jmpbuf))
  459. #else
  460.    if (setjmp(png_ptr->jmpbuf))
  461. #endif
  462.    {
  463.       png_free(png_ptr, png_ptr->zbuf);
  464.        png_ptr->zbuf=NULL;
  465.       png_destroy_struct(png_ptr);
  466.       return (NULL);
  467.    }
  468. #ifdef USE_FAR_KEYWORD
  469.    png_memcpy(png_ptr->jmpbuf, jmpbuf, png_sizeof(jmp_buf));
  470. #endif
  471. #endif
  472. #ifdef PNG_USER_MEM_SUPPORTED
  473.    png_set_mem_fn(png_ptr, mem_ptr, malloc_fn, free_fn);
  474. #endif /* PNG_USER_MEM_SUPPORTED */
  475.    png_set_error_fn(png_ptr, error_ptr, error_fn, warn_fn);
  476.    if (user_png_ver)
  477.    {
  478.      i=0;
  479.      do
  480.      {
  481.        if (user_png_ver[i] != png_libpng_ver[i])
  482.           png_ptr->flags |= PNG_FLAG_LIBRARY_MISMATCH;
  483.      } while (png_libpng_ver[i++]);
  484.    }
  485.    if (png_ptr->flags & PNG_FLAG_LIBRARY_MISMATCH)
  486.    {
  487.      /* Libpng 0.90 and later are binary incompatible with libpng 0.89, so
  488.       * we must recompile any applications that use any older library version.
  489.       * For versions after libpng 1.0, we will be compatible, so we need
  490.       * only check the first digit.
  491.       */
  492.      if (user_png_ver == NULL || user_png_ver[0] != png_libpng_ver[0] ||
  493.          (user_png_ver[0] == '1' && user_png_ver[2] != png_libpng_ver[2]) ||
  494.          (user_png_ver[0] == '0' && user_png_ver[2] < '9'))
  495.      {
  496. #if !defined(PNG_NO_STDIO) && !defined(_WIN32_WCE)
  497.         char msg[80];
  498.         if (user_png_ver)
  499.         {
  500.           png_snprintf(msg, 80,
  501.              "Application was compiled with png.h from libpng-%.20s",
  502.              user_png_ver);
  503.           png_warning(png_ptr, msg);
  504.         }
  505.         png_snprintf(msg, 80,
  506.            "Application  is  running with png.c from libpng-%.20s",
  507.            png_libpng_ver);
  508.         png_warning(png_ptr, msg);
  509. #endif
  510. #ifdef PNG_ERROR_NUMBERS_SUPPORTED
  511.         png_ptr->flags=0;
  512. #endif
  513.         png_error(png_ptr,
  514.            "Incompatible libpng version in application and library");
  515.      }
  516.    }
  517.    /* initialize zbuf - compression buffer */
  518.    png_ptr->zbuf_size = PNG_ZBUF_SIZE;
  519.    png_ptr->zbuf = (png_bytep)png_malloc(png_ptr,
  520.       (png_uint_32)png_ptr->zbuf_size);
  521.    png_set_write_fn(png_ptr, png_voidp_NULL, png_rw_ptr_NULL,
  522.       png_flush_ptr_NULL);
  523. #if defined(PNG_WRITE_WEIGHTED_FILTER_SUPPORTED)
  524.    png_set_filter_heuristics(png_ptr, PNG_FILTER_HEURISTIC_DEFAULT,
  525.       1, png_doublep_NULL, png_doublep_NULL);
  526. #endif
  527. #ifdef PNG_SETJMP_SUPPORTED
  528. /* Applications that neglect to set up their own setjmp() and then encounter
  529.    a png_error() will longjmp here.  Since the jmpbuf is then meaningless we
  530.    abort instead of returning. */
  531. #ifdef USE_FAR_KEYWORD
  532.    if (setjmp(jmpbuf))
  533.       PNG_ABORT();
  534.    png_memcpy(png_ptr->jmpbuf, jmpbuf, png_sizeof(jmp_buf));
  535. #else
  536.    if (setjmp(png_ptr->jmpbuf))
  537.       PNG_ABORT();
  538. #endif
  539. #endif
  540.    return (png_ptr);
  541. }
  542. /* Initialize png_ptr structure, and allocate any memory needed */
  543. #if defined(PNG_1_0_X) || defined(PNG_1_2_X)
  544. /* Deprecated. */
  545. #undef png_write_init
  546. void PNGAPI
  547. png_write_init(png_structp png_ptr)
  548. {
  549.    /* We only come here via pre-1.0.7-compiled applications */
  550.    png_write_init_2(png_ptr, "1.0.6 or earlier", 0, 0);
  551. }
  552. void PNGAPI
  553. png_write_init_2(png_structp png_ptr, png_const_charp user_png_ver,
  554.    png_size_t png_struct_size, png_size_t png_info_size)
  555. {
  556.    /* We only come here via pre-1.0.12-compiled applications */
  557.    if (png_ptr == NULL) return;
  558. #if !defined(PNG_NO_STDIO) && !defined(_WIN32_WCE)
  559.    if (png_sizeof(png_struct) > png_struct_size ||
  560.       png_sizeof(png_info) > png_info_size)
  561.    {
  562.       char msg[80];
  563.       png_ptr->warning_fn=NULL;
  564.       if (user_png_ver)
  565.       {
  566.         png_snprintf(msg, 80,
  567.            "Application was compiled with png.h from libpng-%.20s",
  568.            user_png_ver);
  569.         png_warning(png_ptr, msg);
  570.       }
  571.       png_snprintf(msg, 80,
  572.          "Application  is  running with png.c from libpng-%.20s",
  573.          png_libpng_ver);
  574.       png_warning(png_ptr, msg);
  575.    }
  576. #endif
  577.    if (png_sizeof(png_struct) > png_struct_size)
  578.      {
  579.        png_ptr->error_fn=NULL;
  580. #ifdef PNG_ERROR_NUMBERS_SUPPORTED
  581.        png_ptr->flags=0;
  582. #endif
  583.        png_error(png_ptr,
  584.        "The png struct allocated by the application for writing is too small.");
  585.      }
  586.    if (png_sizeof(png_info) > png_info_size)
  587.      {
  588.        png_ptr->error_fn=NULL;
  589. #ifdef PNG_ERROR_NUMBERS_SUPPORTED
  590.        png_ptr->flags=0;
  591. #endif
  592.        png_error(png_ptr,
  593.        "The info struct allocated by the application for writing is too small.");
  594.      }
  595.    png_write_init_3(&png_ptr, user_png_ver, png_struct_size);
  596. }
  597. #endif /* PNG_1_0_X || PNG_1_2_X */
  598. void PNGAPI
  599. png_write_init_3(png_structpp ptr_ptr, png_const_charp user_png_ver,
  600.    png_size_t png_struct_size)
  601. {
  602.    png_structp png_ptr=*ptr_ptr;
  603. #ifdef PNG_SETJMP_SUPPORTED
  604.    jmp_buf tmp_jmp; /* to save current jump buffer */
  605. #endif
  606.    int i = 0;
  607.    if (png_ptr == NULL)
  608.       return;
  609.    do
  610.    {
  611.      if (user_png_ver[i] != png_libpng_ver[i])
  612.      {
  613. #ifdef PNG_LEGACY_SUPPORTED
  614.        png_ptr->flags |= PNG_FLAG_LIBRARY_MISMATCH;
  615. #else
  616.        png_ptr->warning_fn=NULL;
  617.        png_warning(png_ptr,
  618.  "Application uses deprecated png_write_init() and should be recompiled.");
  619.        break;
  620. #endif
  621.      }
  622.    } while (png_libpng_ver[i++]);
  623.    png_debug(1, "in png_write_init_3");
  624. #ifdef PNG_SETJMP_SUPPORTED
  625.    /* save jump buffer and error functions */
  626.    png_memcpy(tmp_jmp, png_ptr->jmpbuf, png_sizeof(jmp_buf));
  627. #endif
  628.    if (png_sizeof(png_struct) > png_struct_size)
  629.      {
  630.        png_destroy_struct(png_ptr);
  631.        png_ptr = (png_structp)png_create_struct(PNG_STRUCT_PNG);
  632.        *ptr_ptr = png_ptr;
  633.      }
  634.    /* reset all variables to 0 */
  635.    png_memset(png_ptr, 0, png_sizeof(png_struct));
  636.    /* added at libpng-1.2.6 */
  637. #ifdef PNG_SET_USER_LIMITS_SUPPORTED
  638.    png_ptr->user_width_max=PNG_USER_WIDTH_MAX;
  639.    png_ptr->user_height_max=PNG_USER_HEIGHT_MAX;
  640. #endif
  641. #ifdef PNG_SETJMP_SUPPORTED
  642.    /* restore jump buffer */
  643.    png_memcpy(png_ptr->jmpbuf, tmp_jmp, png_sizeof(jmp_buf));
  644. #endif
  645.    png_set_write_fn(png_ptr, png_voidp_NULL, png_rw_ptr_NULL,
  646.       png_flush_ptr_NULL);
  647.    /* initialize zbuf - compression buffer */
  648.    png_ptr->zbuf_size = PNG_ZBUF_SIZE;
  649.    png_ptr->zbuf = (png_bytep)png_malloc(png_ptr,
  650.       (png_uint_32)png_ptr->zbuf_size);
  651. #if defined(PNG_WRITE_WEIGHTED_FILTER_SUPPORTED)
  652.    png_set_filter_heuristics(png_ptr, PNG_FILTER_HEURISTIC_DEFAULT,
  653.       1, png_doublep_NULL, png_doublep_NULL);
  654. #endif
  655. }
  656. /* Write a few rows of image data.  If the image is interlaced,
  657.  * either you will have to write the 7 sub images, or, if you
  658.  * have called png_set_interlace_handling(), you will have to
  659.  * "write" the image seven times.
  660.  */
  661. void PNGAPI
  662. png_write_rows(png_structp png_ptr, png_bytepp row,
  663.    png_uint_32 num_rows)
  664. {
  665.    png_uint_32 i; /* row counter */
  666.    png_bytepp rp; /* row pointer */
  667.    png_debug(1, "in png_write_rows");
  668.    if (png_ptr == NULL)
  669.       return;
  670.    /* loop through the rows */
  671.    for (i = 0, rp = row; i < num_rows; i++, rp++)
  672.    {
  673.       png_write_row(png_ptr, *rp);
  674.    }
  675. }
  676. /* Write the image.  You only need to call this function once, even
  677.  * if you are writing an interlaced image.
  678.  */
  679. void PNGAPI
  680. png_write_image(png_structp png_ptr, png_bytepp image)
  681. {
  682.    png_uint_32 i; /* row index */
  683.    int pass, num_pass; /* pass variables */
  684.    png_bytepp rp; /* points to current row */
  685.    if (png_ptr == NULL)
  686.       return;
  687.    png_debug(1, "in png_write_image");
  688. #if defined(PNG_WRITE_INTERLACING_SUPPORTED)
  689.    /* intialize interlace handling.  If image is not interlaced,
  690.       this will set pass to 1 */
  691.    num_pass = png_set_interlace_handling(png_ptr);
  692. #else
  693.    num_pass = 1;
  694. #endif
  695.    /* loop through passes */
  696.    for (pass = 0; pass < num_pass; pass++)
  697.    {
  698.       /* loop through image */
  699.       for (i = 0, rp = image; i < png_ptr->height; i++, rp++)
  700.       {
  701.          png_write_row(png_ptr, *rp);
  702.       }
  703.    }
  704. }
  705. /* called by user to write a row of image data */
  706. void PNGAPI
  707. png_write_row(png_structp png_ptr, png_bytep row)
  708. {
  709.    if (png_ptr == NULL)
  710.       return;
  711.    png_debug2(1, "in png_write_row (row %ld, pass %d)",
  712.       png_ptr->row_number, png_ptr->pass);
  713.    /* initialize transformations and other stuff if first time */
  714.    if (png_ptr->row_number == 0 && png_ptr->pass == 0)
  715.    {
  716.    /* make sure we wrote the header info */
  717.    if (!(png_ptr->mode & PNG_WROTE_INFO_BEFORE_PLTE))
  718.       png_error(png_ptr,
  719.          "png_write_info was never called before png_write_row.");
  720.    /* check for transforms that have been set but were defined out */
  721. #if !defined(PNG_WRITE_INVERT_SUPPORTED) && defined(PNG_READ_INVERT_SUPPORTED)
  722.    if (png_ptr->transformations & PNG_INVERT_MONO)
  723.       png_warning(png_ptr, "PNG_WRITE_INVERT_SUPPORTED is not defined.");
  724. #endif
  725. #if !defined(PNG_WRITE_FILLER_SUPPORTED) && defined(PNG_READ_FILLER_SUPPORTED)
  726.    if (png_ptr->transformations & PNG_FILLER)
  727.       png_warning(png_ptr, "PNG_WRITE_FILLER_SUPPORTED is not defined.");
  728. #endif
  729. #if !defined(PNG_WRITE_PACKSWAP_SUPPORTED) && defined(PNG_READ_PACKSWAP_SUPPORTED)
  730.    if (png_ptr->transformations & PNG_PACKSWAP)
  731.       png_warning(png_ptr, "PNG_WRITE_PACKSWAP_SUPPORTED is not defined.");
  732. #endif
  733. #if !defined(PNG_WRITE_PACK_SUPPORTED) && defined(PNG_READ_PACK_SUPPORTED)
  734.    if (png_ptr->transformations & PNG_PACK)
  735.       png_warning(png_ptr, "PNG_WRITE_PACK_SUPPORTED is not defined.");
  736. #endif
  737. #if !defined(PNG_WRITE_SHIFT_SUPPORTED) && defined(PNG_READ_SHIFT_SUPPORTED)
  738.    if (png_ptr->transformations & PNG_SHIFT)
  739.       png_warning(png_ptr, "PNG_WRITE_SHIFT_SUPPORTED is not defined.");
  740. #endif
  741. #if !defined(PNG_WRITE_BGR_SUPPORTED) && defined(PNG_READ_BGR_SUPPORTED)
  742.    if (png_ptr->transformations & PNG_BGR)
  743.       png_warning(png_ptr, "PNG_WRITE_BGR_SUPPORTED is not defined.");
  744. #endif
  745. #if !defined(PNG_WRITE_SWAP_SUPPORTED) && defined(PNG_READ_SWAP_SUPPORTED)
  746.    if (png_ptr->transformations & PNG_SWAP_BYTES)
  747.       png_warning(png_ptr, "PNG_WRITE_SWAP_SUPPORTED is not defined.");
  748. #endif
  749.       png_write_start_row(png_ptr);
  750.    }
  751. #if defined(PNG_WRITE_INTERLACING_SUPPORTED)
  752.    /* if interlaced and not interested in row, return */
  753.    if (png_ptr->interlaced && (png_ptr->transformations & PNG_INTERLACE))
  754.    {
  755.       switch (png_ptr->pass)
  756.       {
  757.          case 0:
  758.             if (png_ptr->row_number & 0x07)
  759.             {
  760.                png_write_finish_row(png_ptr);
  761.                return;
  762.             }
  763.             break;
  764.          case 1:
  765.             if ((png_ptr->row_number & 0x07) || png_ptr->width < 5)
  766.             {
  767.                png_write_finish_row(png_ptr);
  768.                return;
  769.             }
  770.             break;
  771.          case 2:
  772.             if ((png_ptr->row_number & 0x07) != 4)
  773.             {
  774.                png_write_finish_row(png_ptr);
  775.                return;
  776.             }
  777.             break;
  778.          case 3:
  779.             if ((png_ptr->row_number & 0x03) || png_ptr->width < 3)
  780.             {
  781.                png_write_finish_row(png_ptr);
  782.                return;
  783.             }
  784.             break;
  785.          case 4:
  786.             if ((png_ptr->row_number & 0x03) != 2)
  787.             {
  788.                png_write_finish_row(png_ptr);
  789.                return;
  790.             }
  791.             break;
  792.          case 5:
  793.             if ((png_ptr->row_number & 0x01) || png_ptr->width < 2)
  794.             {
  795.                png_write_finish_row(png_ptr);
  796.                return;
  797.             }
  798.             break;
  799.          case 6:
  800.             if (!(png_ptr->row_number & 0x01))
  801.             {
  802.                png_write_finish_row(png_ptr);
  803.                return;
  804.             }
  805.             break;
  806.       }
  807.    }
  808. #endif
  809.    /* set up row info for transformations */
  810.    png_ptr->row_info.color_type = png_ptr->color_type;
  811.    png_ptr->row_info.width = png_ptr->usr_width;
  812.    png_ptr->row_info.channels = png_ptr->usr_channels;
  813.    png_ptr->row_info.bit_depth = png_ptr->usr_bit_depth;
  814.    png_ptr->row_info.pixel_depth = (png_byte)(png_ptr->row_info.bit_depth *
  815.       png_ptr->row_info.channels);
  816.    png_ptr->row_info.rowbytes = PNG_ROWBYTES(png_ptr->row_info.pixel_depth,
  817.       png_ptr->row_info.width);
  818.    png_debug1(3, "row_info->color_type = %d", png_ptr->row_info.color_type);
  819.    png_debug1(3, "row_info->width = %lu", png_ptr->row_info.width);
  820.    png_debug1(3, "row_info->channels = %d", png_ptr->row_info.channels);
  821.    png_debug1(3, "row_info->bit_depth = %d", png_ptr->row_info.bit_depth);
  822.    png_debug1(3, "row_info->pixel_depth = %d", png_ptr->row_info.pixel_depth);
  823.    png_debug1(3, "row_info->rowbytes = %lu", png_ptr->row_info.rowbytes);
  824.    /* Copy user's row into buffer, leaving room for filter byte. */
  825.    png_memcpy_check(png_ptr, png_ptr->row_buf + 1, row,
  826.       png_ptr->row_info.rowbytes);
  827. #if defined(PNG_WRITE_INTERLACING_SUPPORTED)
  828.    /* handle interlacing */
  829.    if (png_ptr->interlaced && png_ptr->pass < 6 &&
  830.       (png_ptr->transformations & PNG_INTERLACE))
  831.    {
  832.       png_do_write_interlace(&(png_ptr->row_info),
  833.          png_ptr->row_buf + 1, png_ptr->pass);
  834.       /* this should always get caught above, but still ... */
  835.       if (!(png_ptr->row_info.width))
  836.       {
  837.          png_write_finish_row(png_ptr);
  838.          return;
  839.       }
  840.    }
  841. #endif
  842.    /* handle other transformations */
  843.    if (png_ptr->transformations)
  844.       png_do_write_transformations(png_ptr);
  845. #if defined(PNG_MNG_FEATURES_SUPPORTED)
  846.    /* Write filter_method 64 (intrapixel differencing) only if
  847.     * 1. Libpng was compiled with PNG_MNG_FEATURES_SUPPORTED and
  848.     * 2. Libpng did not write a PNG signature (this filter_method is only
  849.     *    used in PNG datastreams that are embedded in MNG datastreams) and
  850.     * 3. The application called png_permit_mng_features with a mask that
  851.     *    included PNG_FLAG_MNG_FILTER_64 and
  852.     * 4. The filter_method is 64 and
  853.     * 5. The color_type is RGB or RGBA
  854.     */
  855.    if ((png_ptr->mng_features_permitted & PNG_FLAG_MNG_FILTER_64) &&
  856.       (png_ptr->filter_type == PNG_INTRAPIXEL_DIFFERENCING))
  857.    {
  858.       /* Intrapixel differencing */
  859.       png_do_write_intrapixel(&(png_ptr->row_info), png_ptr->row_buf + 1);
  860.    }
  861. #endif
  862.    /* Find a filter if necessary, filter the row and write it out. */
  863.    png_write_find_filter(png_ptr, &(png_ptr->row_info));
  864.    if (png_ptr->write_row_fn != NULL)
  865.       (*(png_ptr->write_row_fn))(png_ptr, png_ptr->row_number, png_ptr->pass);
  866. }
  867. #if defined(PNG_WRITE_FLUSH_SUPPORTED)
  868. /* Set the automatic flush interval or 0 to turn flushing off */
  869. void PNGAPI
  870. png_set_flush(png_structp png_ptr, int nrows)
  871. {
  872.    png_debug(1, "in png_set_flush");
  873.    if (png_ptr == NULL)
  874.       return;
  875.    png_ptr->flush_dist = (nrows < 0 ? 0 : nrows);
  876. }
  877. /* flush the current output buffers now */
  878. void PNGAPI
  879. png_write_flush(png_structp png_ptr)
  880. {
  881.    int wrote_IDAT;
  882.    png_debug(1, "in png_write_flush");
  883.    if (png_ptr == NULL)
  884.       return;
  885.    /* We have already written out all of the data */
  886.    if (png_ptr->row_number >= png_ptr->num_rows)
  887.      return;
  888.    do
  889.    {
  890.       int ret;
  891.       /* compress the data */
  892.       ret = deflate(&png_ptr->zstream, Z_SYNC_FLUSH);
  893.       wrote_IDAT = 0;
  894.       /* check for compression errors */
  895.       if (ret != Z_OK)
  896.       {
  897.          if (png_ptr->zstream.msg != NULL)
  898.             png_error(png_ptr, png_ptr->zstream.msg);
  899.          else
  900.             png_error(png_ptr, "zlib error");
  901.       }
  902.       if (!(png_ptr->zstream.avail_out))
  903.       {
  904.          /* write the IDAT and reset the zlib output buffer */
  905.          png_write_IDAT(png_ptr, png_ptr->zbuf,
  906.                         png_ptr->zbuf_size);
  907.          png_ptr->zstream.next_out = png_ptr->zbuf;
  908.          png_ptr->zstream.avail_out = (uInt)png_ptr->zbuf_size;
  909.          wrote_IDAT = 1;
  910.       }
  911.    } while(wrote_IDAT == 1);
  912.    /* If there is any data left to be output, write it into a new IDAT */
  913.    if (png_ptr->zbuf_size != png_ptr->zstream.avail_out)
  914.    {
  915.       /* write the IDAT and reset the zlib output buffer */
  916.       png_write_IDAT(png_ptr, png_ptr->zbuf,
  917.                      png_ptr->zbuf_size - png_ptr->zstream.avail_out);
  918.       png_ptr->zstream.next_out = png_ptr->zbuf;
  919.       png_ptr->zstream.avail_out = (uInt)png_ptr->zbuf_size;
  920.    }
  921.    png_ptr->flush_rows = 0;
  922.    png_flush(png_ptr);
  923. }
  924. #endif /* PNG_WRITE_FLUSH_SUPPORTED */
  925. /* free all memory used by the write */
  926. void PNGAPI
  927. png_destroy_write_struct(png_structpp png_ptr_ptr, png_infopp info_ptr_ptr)
  928. {
  929.    png_structp png_ptr = NULL;
  930.    png_infop info_ptr = NULL;
  931. #ifdef PNG_USER_MEM_SUPPORTED
  932.    png_free_ptr free_fn = NULL;
  933.    png_voidp mem_ptr = NULL;
  934. #endif
  935.    png_debug(1, "in png_destroy_write_struct");
  936.    if (png_ptr_ptr != NULL)
  937.    {
  938.       png_ptr = *png_ptr_ptr;
  939. #ifdef PNG_USER_MEM_SUPPORTED
  940.       free_fn = png_ptr->free_fn;
  941.       mem_ptr = png_ptr->mem_ptr;
  942. #endif
  943.    }
  944. #ifdef PNG_USER_MEM_SUPPORTED
  945.    if (png_ptr != NULL)
  946.    {
  947.       free_fn = png_ptr->free_fn;
  948.       mem_ptr = png_ptr->mem_ptr;
  949.    }
  950. #endif
  951.    if (info_ptr_ptr != NULL)
  952.       info_ptr = *info_ptr_ptr;
  953.    if (info_ptr != NULL)
  954.    {
  955.       if (png_ptr != NULL)
  956.       {
  957.         png_free_data(png_ptr, info_ptr, PNG_FREE_ALL, -1);
  958. #if defined(PNG_UNKNOWN_CHUNKS_SUPPORTED)
  959.         if (png_ptr->num_chunk_list)
  960.         {
  961.            png_free(png_ptr, png_ptr->chunk_list);
  962.            png_ptr->chunk_list=NULL;
  963.            png_ptr->num_chunk_list = 0;
  964.         }
  965. #endif
  966.       }
  967. #ifdef PNG_USER_MEM_SUPPORTED
  968.       png_destroy_struct_2((png_voidp)info_ptr, (png_free_ptr)free_fn,
  969.          (png_voidp)mem_ptr);
  970. #else
  971.       png_destroy_struct((png_voidp)info_ptr);
  972. #endif
  973.       *info_ptr_ptr = NULL;
  974.    }
  975.    if (png_ptr != NULL)
  976.    {
  977.       png_write_destroy(png_ptr);
  978. #ifdef PNG_USER_MEM_SUPPORTED
  979.       png_destroy_struct_2((png_voidp)png_ptr, (png_free_ptr)free_fn,
  980.          (png_voidp)mem_ptr);
  981. #else
  982.       png_destroy_struct((png_voidp)png_ptr);
  983. #endif
  984.       *png_ptr_ptr = NULL;
  985.    }
  986. }
  987. /* Free any memory used in png_ptr struct (old method) */
  988. void /* PRIVATE */
  989. png_write_destroy(png_structp png_ptr)
  990. {
  991. #ifdef PNG_SETJMP_SUPPORTED
  992.    jmp_buf tmp_jmp; /* save jump buffer */
  993. #endif
  994.    png_error_ptr error_fn;
  995.    png_error_ptr warning_fn;
  996.    png_voidp error_ptr;
  997. #ifdef PNG_USER_MEM_SUPPORTED
  998.    png_free_ptr free_fn;
  999. #endif
  1000.    png_debug(1, "in png_write_destroy");
  1001.    /* free any memory zlib uses */
  1002.    deflateEnd(&png_ptr->zstream);
  1003.    /* free our memory.  png_free checks NULL for us. */
  1004.    png_free(png_ptr, png_ptr->zbuf);
  1005.    png_free(png_ptr, png_ptr->row_buf);
  1006. #ifndef PNG_NO_WRITE_FILTER
  1007.    png_free(png_ptr, png_ptr->prev_row);
  1008.    png_free(png_ptr, png_ptr->sub_row);
  1009.    png_free(png_ptr, png_ptr->up_row);
  1010.    png_free(png_ptr, png_ptr->avg_row);
  1011.    png_free(png_ptr, png_ptr->paeth_row);
  1012. #endif
  1013. #if defined(PNG_TIME_RFC1123_SUPPORTED)
  1014.    png_free(png_ptr, png_ptr->time_buffer);
  1015. #endif
  1016. #if defined(PNG_WRITE_WEIGHTED_FILTER_SUPPORTED)
  1017.    png_free(png_ptr, png_ptr->prev_filters);
  1018.    png_free(png_ptr, png_ptr->filter_weights);
  1019.    png_free(png_ptr, png_ptr->inv_filter_weights);
  1020.    png_free(png_ptr, png_ptr->filter_costs);
  1021.    png_free(png_ptr, png_ptr->inv_filter_costs);
  1022. #endif
  1023. #ifdef PNG_SETJMP_SUPPORTED
  1024.    /* reset structure */
  1025.    png_memcpy(tmp_jmp, png_ptr->jmpbuf, png_sizeof(jmp_buf));
  1026. #endif
  1027.    error_fn = png_ptr->error_fn;
  1028.    warning_fn = png_ptr->warning_fn;
  1029.    error_ptr = png_ptr->error_ptr;
  1030. #ifdef PNG_USER_MEM_SUPPORTED
  1031.    free_fn = png_ptr->free_fn;
  1032. #endif
  1033.    png_memset(png_ptr, 0, png_sizeof(png_struct));
  1034.    png_ptr->error_fn = error_fn;
  1035.    png_ptr->warning_fn = warning_fn;
  1036.    png_ptr->error_ptr = error_ptr;
  1037. #ifdef PNG_USER_MEM_SUPPORTED
  1038.    png_ptr->free_fn = free_fn;
  1039. #endif
  1040. #ifdef PNG_SETJMP_SUPPORTED
  1041.    png_memcpy(png_ptr->jmpbuf, tmp_jmp, png_sizeof(jmp_buf));
  1042. #endif
  1043. }
  1044. /* Allow the application to select one or more row filters to use. */
  1045. void PNGAPI
  1046. png_set_filter(png_structp png_ptr, int method, int filters)
  1047. {
  1048.    png_debug(1, "in png_set_filter");
  1049.    if (png_ptr == NULL)
  1050.       return;
  1051. #if defined(PNG_MNG_FEATURES_SUPPORTED)
  1052.    if ((png_ptr->mng_features_permitted & PNG_FLAG_MNG_FILTER_64) &&
  1053.       (method == PNG_INTRAPIXEL_DIFFERENCING))
  1054.          method = PNG_FILTER_TYPE_BASE;
  1055. #endif
  1056.    if (method == PNG_FILTER_TYPE_BASE)
  1057.    {
  1058.       switch (filters & (PNG_ALL_FILTERS | 0x07))
  1059.       {
  1060. #ifndef PNG_NO_WRITE_FILTER
  1061.          case 5:
  1062.          case 6:
  1063.          case 7: png_warning(png_ptr, "Unknown row filter for method 0");
  1064. #endif /* PNG_NO_WRITE_FILTER */
  1065.          case PNG_FILTER_VALUE_NONE:
  1066.               png_ptr->do_filter=PNG_FILTER_NONE; break;
  1067. #ifndef PNG_NO_WRITE_FILTER
  1068.          case PNG_FILTER_VALUE_SUB:
  1069.               png_ptr->do_filter=PNG_FILTER_SUB; break;
  1070.          case PNG_FILTER_VALUE_UP:
  1071.               png_ptr->do_filter=PNG_FILTER_UP; break;
  1072.          case PNG_FILTER_VALUE_AVG:
  1073.               png_ptr->do_filter=PNG_FILTER_AVG; break;
  1074.          case PNG_FILTER_VALUE_PAETH:
  1075.               png_ptr->do_filter=PNG_FILTER_PAETH; break;
  1076.          default: png_ptr->do_filter = (png_byte)filters; break;
  1077. #else
  1078.          default: png_warning(png_ptr, "Unknown row filter for method 0");
  1079. #endif /* PNG_NO_WRITE_FILTER */
  1080.       }
  1081.       /* If we have allocated the row_buf, this means we have already started
  1082.        * with the image and we should have allocated all of the filter buffers
  1083.        * that have been selected.  If prev_row isn't already allocated, then
  1084.        * it is too late to start using the filters that need it, since we
  1085.        * will be missing the data in the previous row.  If an application
  1086.        * wants to start and stop using particular filters during compression,
  1087.        * it should start out with all of the filters, and then add and
  1088.        * remove them after the start of compression.
  1089.        */
  1090.       if (png_ptr->row_buf != NULL)
  1091.       {
  1092. #ifndef PNG_NO_WRITE_FILTER
  1093.          if ((png_ptr->do_filter & PNG_FILTER_SUB) && png_ptr->sub_row == NULL)
  1094.          {
  1095.             png_ptr->sub_row = (png_bytep)png_malloc(png_ptr,
  1096.               (png_ptr->rowbytes + 1));
  1097.             png_ptr->sub_row[0] = PNG_FILTER_VALUE_SUB;
  1098.          }
  1099.          if ((png_ptr->do_filter & PNG_FILTER_UP) && png_ptr->up_row == NULL)
  1100.          {
  1101.             if (png_ptr->prev_row == NULL)
  1102.             {
  1103.                png_warning(png_ptr, "Can't add Up filter after starting");
  1104.                png_ptr->do_filter &= ~PNG_FILTER_UP;
  1105.             }
  1106.             else
  1107.             {
  1108.                png_ptr->up_row = (png_bytep)png_malloc(png_ptr,
  1109.                   (png_ptr->rowbytes + 1));
  1110.                png_ptr->up_row[0] = PNG_FILTER_VALUE_UP;
  1111.             }
  1112.          }
  1113.          if ((png_ptr->do_filter & PNG_FILTER_AVG) && png_ptr->avg_row == NULL)
  1114.          {
  1115.             if (png_ptr->prev_row == NULL)
  1116.             {
  1117.                png_warning(png_ptr, "Can't add Average filter after starting");
  1118.                png_ptr->do_filter &= ~PNG_FILTER_AVG;
  1119.             }
  1120.             else
  1121.             {
  1122.                png_ptr->avg_row = (png_bytep)png_malloc(png_ptr,
  1123.                   (png_ptr->rowbytes + 1));
  1124.                png_ptr->avg_row[0] = PNG_FILTER_VALUE_AVG;
  1125.             }
  1126.          }
  1127.          if ((png_ptr->do_filter & PNG_FILTER_PAETH) &&
  1128.              png_ptr->paeth_row == NULL)
  1129.          {
  1130.             if (png_ptr->prev_row == NULL)
  1131.             {
  1132.                png_warning(png_ptr, "Can't add Paeth filter after starting");
  1133.                png_ptr->do_filter &= (png_byte)(~PNG_FILTER_PAETH);
  1134.             }
  1135.             else
  1136.             {
  1137.                png_ptr->paeth_row = (png_bytep)png_malloc(png_ptr,
  1138.                   (png_ptr->rowbytes + 1));
  1139.                png_ptr->paeth_row[0] = PNG_FILTER_VALUE_PAETH;
  1140.             }
  1141.          }
  1142.          if (png_ptr->do_filter == PNG_NO_FILTERS)
  1143. #endif /* PNG_NO_WRITE_FILTER */
  1144.             png_ptr->do_filter = PNG_FILTER_NONE;
  1145.       }
  1146.    }
  1147.    else
  1148.       png_error(png_ptr, "Unknown custom filter method");
  1149. }
  1150. /* This allows us to influence the way in which libpng chooses the "best"
  1151.  * filter for the current scanline.  While the "minimum-sum-of-absolute-
  1152.  * differences metric is relatively fast and effective, there is some
  1153.  * question as to whether it can be improved upon by trying to keep the
  1154.  * filtered data going to zlib more consistent, hopefully resulting in
  1155.  * better compression.
  1156.  */
  1157. #if defined(PNG_WRITE_WEIGHTED_FILTER_SUPPORTED)      /* GRR 970116 */
  1158. void PNGAPI
  1159. png_set_filter_heuristics(png_structp png_ptr, int heuristic_method,
  1160.    int num_weights, png_doublep filter_weights,
  1161.    png_doublep filter_costs)
  1162. {
  1163.    int i;
  1164.    png_debug(1, "in png_set_filter_heuristics");
  1165.    if (png_ptr == NULL)
  1166.       return;
  1167.    if (heuristic_method >= PNG_FILTER_HEURISTIC_LAST)
  1168.    {
  1169.       png_warning(png_ptr, "Unknown filter heuristic method");
  1170.       return;
  1171.    }
  1172.    if (heuristic_method == PNG_FILTER_HEURISTIC_DEFAULT)
  1173.    {
  1174.       heuristic_method = PNG_FILTER_HEURISTIC_UNWEIGHTED;
  1175.    }
  1176.    if (num_weights < 0 || filter_weights == NULL ||
  1177.       heuristic_method == PNG_FILTER_HEURISTIC_UNWEIGHTED)
  1178.    {
  1179.       num_weights = 0;
  1180.    }
  1181.    png_ptr->num_prev_filters = (png_byte)num_weights;
  1182.    png_ptr->heuristic_method = (png_byte)heuristic_method;
  1183.    if (num_weights > 0)
  1184.    {
  1185.       if (png_ptr->prev_filters == NULL)
  1186.       {
  1187.          png_ptr->prev_filters = (png_bytep)png_malloc(png_ptr,
  1188.             (png_uint_32)(png_sizeof(png_byte) * num_weights));
  1189.          /* To make sure that the weighting starts out fairly */
  1190.          for (i = 0; i < num_weights; i++)
  1191.          {
  1192.             png_ptr->prev_filters[i] = 255;
  1193.          }
  1194.       }
  1195.       if (png_ptr->filter_weights == NULL)
  1196.       {
  1197.          png_ptr->filter_weights = (png_uint_16p)png_malloc(png_ptr,
  1198.             (png_uint_32)(png_sizeof(png_uint_16) * num_weights));
  1199.          png_ptr->inv_filter_weights = (png_uint_16p)png_malloc(png_ptr,
  1200.             (png_uint_32)(png_sizeof(png_uint_16) * num_weights));
  1201.          for (i = 0; i < num_weights; i++)
  1202.          {
  1203.             png_ptr->inv_filter_weights[i] =
  1204.             png_ptr->filter_weights[i] = PNG_WEIGHT_FACTOR;
  1205.          }
  1206.       }
  1207.       for (i = 0; i < num_weights; i++)
  1208.       {
  1209.          if (filter_weights[i] < 0.0)
  1210.          {
  1211.             png_ptr->inv_filter_weights[i] =
  1212.             png_ptr->filter_weights[i] = PNG_WEIGHT_FACTOR;
  1213.          }
  1214.          else
  1215.          {
  1216.             png_ptr->inv_filter_weights[i] =
  1217.                (png_uint_16)((double)PNG_WEIGHT_FACTOR*filter_weights[i]+0.5);
  1218.             png_ptr->filter_weights[i] =
  1219.                (png_uint_16)((double)PNG_WEIGHT_FACTOR/filter_weights[i]+0.5);
  1220.          }
  1221.       }
  1222.    }
  1223.    /* If, in the future, there are other filter methods, this would
  1224.     * need to be based on png_ptr->filter.
  1225.     */
  1226.    if (png_ptr->filter_costs == NULL)
  1227.    {
  1228.       png_ptr->filter_costs = (png_uint_16p)png_malloc(png_ptr,
  1229.          (png_uint_32)(png_sizeof(png_uint_16) * PNG_FILTER_VALUE_LAST));
  1230.       png_ptr->inv_filter_costs = (png_uint_16p)png_malloc(png_ptr,
  1231.          (png_uint_32)(png_sizeof(png_uint_16) * PNG_FILTER_VALUE_LAST));
  1232.       for (i = 0; i < PNG_FILTER_VALUE_LAST; i++)
  1233.       {
  1234.          png_ptr->inv_filter_costs[i] =
  1235.          png_ptr->filter_costs[i] = PNG_COST_FACTOR;
  1236.       }
  1237.    }
  1238.    /* Here is where we set the relative costs of the different filters.  We
  1239.     * should take the desired compression level into account when setting
  1240.     * the costs, so that Paeth, for instance, has a high relative cost at low
  1241.     * compression levels, while it has a lower relative cost at higher
  1242.     * compression settings.  The filter types are in order of increasing
  1243.     * relative cost, so it would be possible to do this with an algorithm.
  1244.     */
  1245.    for (i = 0; i < PNG_FILTER_VALUE_LAST; i++)
  1246.    {
  1247.       if (filter_costs == NULL || filter_costs[i] < 0.0)
  1248.       {
  1249.          png_ptr->inv_filter_costs[i] =
  1250.          png_ptr->filter_costs[i] = PNG_COST_FACTOR;
  1251.       }
  1252.       else if (filter_costs[i] >= 1.0)
  1253.       {
  1254.          png_ptr->inv_filter_costs[i] =
  1255.             (png_uint_16)((double)PNG_COST_FACTOR / filter_costs[i] + 0.5);
  1256.          png_ptr->filter_costs[i] =
  1257.             (png_uint_16)((double)PNG_COST_FACTOR * filter_costs[i] + 0.5);
  1258.       }
  1259.    }
  1260. }
  1261. #endif /* PNG_WRITE_WEIGHTED_FILTER_SUPPORTED */
  1262. void PNGAPI
  1263. png_set_compression_level(png_structp png_ptr, int level)
  1264. {
  1265.    png_debug(1, "in png_set_compression_level");
  1266.    if (png_ptr == NULL)
  1267.       return;
  1268.    png_ptr->flags |= PNG_FLAG_ZLIB_CUSTOM_LEVEL;
  1269.    png_ptr->zlib_level = level;
  1270. }
  1271. void PNGAPI
  1272. png_set_compression_mem_level(png_structp png_ptr, int mem_level)
  1273. {
  1274.    png_debug(1, "in png_set_compression_mem_level");
  1275.    if (png_ptr == NULL)
  1276.       return;
  1277.    png_ptr->flags |= PNG_FLAG_ZLIB_CUSTOM_MEM_LEVEL;
  1278.    png_ptr->zlib_mem_level = mem_level;
  1279. }
  1280. void PNGAPI
  1281. png_set_compression_strategy(png_structp png_ptr, int strategy)
  1282. {
  1283.    png_debug(1, "in png_set_compression_strategy");
  1284.    if (png_ptr == NULL)
  1285.       return;
  1286.    png_ptr->flags |= PNG_FLAG_ZLIB_CUSTOM_STRATEGY;
  1287.    png_ptr->zlib_strategy = strategy;
  1288. }
  1289. void PNGAPI
  1290. png_set_compression_window_bits(png_structp png_ptr, int window_bits)
  1291. {
  1292.    if (png_ptr == NULL)
  1293.       return;
  1294.    if (window_bits > 15)
  1295.       png_warning(png_ptr, "Only compression windows <= 32k supported by PNG");
  1296.    else if (window_bits < 8)
  1297.       png_warning(png_ptr, "Only compression windows >= 256 supported by PNG");
  1298. #ifndef WBITS_8_OK
  1299.    /* avoid libpng bug with 256-byte windows */
  1300.    if (window_bits == 8)
  1301.      {
  1302.        png_warning(png_ptr, "Compression window is being reset to 512");
  1303.        window_bits=9;
  1304.      }
  1305. #endif
  1306.    png_ptr->flags |= PNG_FLAG_ZLIB_CUSTOM_WINDOW_BITS;
  1307.    png_ptr->zlib_window_bits = window_bits;
  1308. }
  1309. void PNGAPI
  1310. png_set_compression_method(png_structp png_ptr, int method)
  1311. {
  1312.    png_debug(1, "in png_set_compression_method");
  1313.    if (png_ptr == NULL)
  1314.       return;
  1315.    if (method != 8)
  1316.       png_warning(png_ptr, "Only compression method 8 is supported by PNG");
  1317.    png_ptr->flags |= PNG_FLAG_ZLIB_CUSTOM_METHOD;
  1318.    png_ptr->zlib_method = method;
  1319. }
  1320. void PNGAPI
  1321. png_set_write_status_fn(png_structp png_ptr, png_write_status_ptr write_row_fn)
  1322. {
  1323.    if (png_ptr == NULL)
  1324.       return;
  1325.    png_ptr->write_row_fn = write_row_fn;
  1326. }
  1327. #if defined(PNG_WRITE_USER_TRANSFORM_SUPPORTED)
  1328. void PNGAPI
  1329. png_set_write_user_transform_fn(png_structp png_ptr, png_user_transform_ptr
  1330.    write_user_transform_fn)
  1331. {
  1332.    png_debug(1, "in png_set_write_user_transform_fn");
  1333.    if (png_ptr == NULL)
  1334.       return;
  1335.    png_ptr->transformations |= PNG_USER_TRANSFORM;
  1336.    png_ptr->write_user_transform_fn = write_user_transform_fn;
  1337. }
  1338. #endif
  1339. #if defined(PNG_INFO_IMAGE_SUPPORTED)
  1340. void PNGAPI
  1341. png_write_png(png_structp png_ptr, png_infop info_ptr,
  1342.               int transforms, voidp params)
  1343. {
  1344.    if (png_ptr == NULL || info_ptr == NULL)
  1345.       return;
  1346. #if defined(PNG_WRITE_INVERT_ALPHA_SUPPORTED)
  1347.    /* invert the alpha channel from opacity to transparency */
  1348.    if (transforms & PNG_TRANSFORM_INVERT_ALPHA)
  1349.        png_set_invert_alpha(png_ptr);
  1350. #endif
  1351.    /* Write the file header information. */
  1352.    png_write_info(png_ptr, info_ptr);
  1353.    /* ------ these transformations don't touch the info structure ------- */
  1354. #if defined(PNG_WRITE_INVERT_SUPPORTED)
  1355.    /* invert monochrome pixels */
  1356.    if (transforms & PNG_TRANSFORM_INVERT_MONO)
  1357.        png_set_invert_mono(png_ptr);
  1358. #endif
  1359. #if defined(PNG_WRITE_SHIFT_SUPPORTED)
  1360.    /* Shift the pixels up to a legal bit depth and fill in
  1361.     * as appropriate to correctly scale the image.
  1362.     */
  1363.    if ((transforms & PNG_TRANSFORM_SHIFT)
  1364.                && (info_ptr->valid & PNG_INFO_sBIT))
  1365.        png_set_shift(png_ptr, &info_ptr->sig_bit);
  1366. #endif
  1367. #if defined(PNG_WRITE_PACK_SUPPORTED)
  1368.    /* pack pixels into bytes */
  1369.    if (transforms & PNG_TRANSFORM_PACKING)
  1370.        png_set_packing(png_ptr);
  1371. #endif
  1372. #if defined(PNG_WRITE_SWAP_ALPHA_SUPPORTED)
  1373.    /* swap location of alpha bytes from ARGB to RGBA */
  1374.    if (transforms & PNG_TRANSFORM_SWAP_ALPHA)
  1375.        png_set_swap_alpha(png_ptr);
  1376. #endif
  1377. #if defined(PNG_WRITE_FILLER_SUPPORTED)
  1378.    /* Pack XRGB/RGBX/ARGB/RGBA into * RGB (4 channels -> 3 channels) */
  1379.   if (transforms & PNG_TRANSFORM_STRIP_FILLER_AFTER)
  1380.       png_set_filler(png_ptr, 0, PNG_FILLER_AFTER);
  1381.   else if (transforms & PNG_TRANSFORM_STRIP_FILLER_BEFORE)
  1382.       png_set_filler(png_ptr, 0, PNG_FILLER_BEFORE);
  1383. #endif
  1384. #if defined(PNG_WRITE_BGR_SUPPORTED)
  1385.    /* flip BGR pixels to RGB */
  1386.    if (transforms & PNG_TRANSFORM_BGR)
  1387.        png_set_bgr(png_ptr);
  1388. #endif
  1389. #if defined(PNG_WRITE_SWAP_SUPPORTED)
  1390.    /* swap bytes of 16-bit files to most significant byte first */
  1391.    if (transforms & PNG_TRANSFORM_SWAP_ENDIAN)
  1392.        png_set_swap(png_ptr);
  1393. #endif
  1394. #if defined(PNG_WRITE_PACKSWAP_SUPPORTED)
  1395.    /* swap bits of 1, 2, 4 bit packed pixel formats */
  1396.    if (transforms & PNG_TRANSFORM_PACKSWAP)
  1397.        png_set_packswap(png_ptr);
  1398. #endif
  1399.    /* ----------------------- end of transformations ------------------- */
  1400.    /* write the bits */
  1401.    if (info_ptr->valid & PNG_INFO_IDAT)
  1402.        png_write_image(png_ptr, info_ptr->row_pointers);
  1403.    /* It is REQUIRED to call this to finish writing the rest of the file */
  1404.    png_write_end(png_ptr, info_ptr);
  1405.    transforms = transforms; /* quiet compiler warnings */
  1406.    params = params;
  1407. }
  1408. #endif
  1409. #endif /* PNG_WRITE_SUPPORTED */