png.c
上传用户:zlh9724
上传日期:2007-01-04
资源大小:1991k
文件大小:18k
源码类别:

浏览器

开发平台:

Unix_Linux

  1. #include <stdio.h>
  2. #define Byte WWWByte
  3. #include "www.h"
  4. #undef Byte
  5. #ifdef PNG
  6. #include "png.h"
  7. extern Visual *visual;
  8. extern int RPixelShift;   /* GRR */
  9. extern int GPixelShift;   /* GRR */
  10. extern int BPixelShift;   /* GRR */
  11. extern int RPixelMask, GPixelMask, BPixelMask;
  12. extern int depth;         /* Depth of display in bits */
  13. extern Colormap colormap;
  14. extern int imaging;       /* set to COLOR888, COLOR232, GREY4 or MONO */
  15. extern int tileWidth, tileHeight;
  16. extern unsigned char *tileData;
  17. extern unsigned long windowColor;
  18. extern unsigned long stdcmap[128];  /* 2/3/2 color maps for gifs etc */
  19. extern unsigned long greymap[16];
  20. extern debug;
  21. extern unsigned long transparent;
  22. extern double Gamma;
  23. extern XColor papercols[256];
  24. extern unsigned char *gamma_table; /* For gamma correcting paper */
  25. /* defined in module dither */
  26. extern int Magic256[256];
  27. extern int Magic16[256];
  28. extern int Magic32[256];
  29. extern int Magic64[256];
  30. /*png_rw_ptr */
  31. static void
  32. arena_png_read_data(png_structp png_ptr, png_bytep data, png_uint_32 length)
  33. {
  34.   Block *bp=(Block*)png_get_io_ptr(png_ptr);
  35.   
  36.   if (bp->size > bp->next) {
  37.     if (bp->next + length > bp->size)
  38.       length = bp->size - bp->next;
  39.     
  40.     memcpy(data, bp->buffer + bp->next, length);
  41.     bp->next += length;
  42.     return;
  43.   } else {
  44.     png_error(png_ptr, "Read Error");
  45.   }
  46. }
  47. void
  48. png_error(png_struct *png_ptr, char *message)
  49. {
  50.   fprintf(stderr,"libpng error: %sn", message);
  51.   /* Warn("libpng error: %sn", message); */
  52.   longjmp(png_ptr->jmpbuf, 1);
  53. }
  54. void
  55. png_warning(png_struct *png_ptr, char *message)
  56. {
  57.   if (!png_ptr)
  58.     return;
  59.   fprintf(stderr,"libpng warning: %sn", message);
  60.   /* Warn("libpng warning: %sn", message); */
  61. }
  62. /* Read image into displays 1,2 or 4 bit deep */
  63. unsigned char *ReadPNGImage_1_2_4(png_struct *png_ptr, png_info *info_ptr, png_byte *png_image) 
  64. {
  65.   png_byte *pp;
  66.   unsigned char *data, *dp;
  67.   int ypos;
  68.   int ppb, bpl;
  69.   int len=info_ptr->width;
  70.   int color_type=info_ptr->color_type;
  71.   int alpha;
  72.   int newbyte;
  73.   
  74.   ppb = 8/depth; /* pixels per byte */
  75.   bpl = len/ppb + (len%ppb ? 1 : 0); /* bytes per line */
  76.   
  77.   data = (unsigned char *)malloc(bpl * info_ptr->height);
  78.   if (data == NULL)
  79.     png_error (png_ptr, "couldn't alloc space for X image data");
  80.   
  81.   /* Remove alpha channel from type,  but remember it */
  82.   alpha=color_type & PNG_COLOR_MASK_ALPHA;
  83.   color_type &= ~PNG_COLOR_MASK_ALPHA;
  84.   
  85.   pp=png_image;
  86.   dp=data;
  87.   newbyte = 1;
  88.   for (ypos = 0; ypos < png_ptr->height; ypos++)
  89.     {
  90.       int col = ypos & 0x0f;
  91.       int shift=0;
  92.       int xpos;
  93.       
  94.       for (xpos = 0; xpos < len; ++xpos)
  95. {
  96.   int row = xpos & 0x0f;
  97.   int gr;
  98.   int cr=0, cg=0, cb=0, cgr, a;
  99.   int isgrey = 0;
  100.   
  101.   if (newbyte) {
  102.     *dp = 0;
  103.     newbyte = 0;
  104.   }
  105.   
  106. /*
  107. (From gif.c)
  108. Some quick notes to remember how shift is calculated:
  109. 0 1 2 3 4 5 6 7 8 9  xpos
  110. 0 1 2 3 4 5 6 7 0 1     xpos %8
  111. 7 6 5 4 3 2 1 0 7 6     7 - (xpos % 8)
  112. 1 0 1 0 1 0 1 0 1 0     (7 - (xpos % 8)) % ppb               pixels perl byte = 2
  113. 4 0 4 0 4 0 4 0 4 0     ((7 - (xpos % 8)) % ppb) * depth     depth = 4
  114. 3 2 1 0 3 2 1                                                pixels per byte = 4
  115. 6 4 2 0 6 4 2                                                depth = 2
  116. */
  117.   shift = (((7 - (xpos % 8)) % ppb) * depth);
  118.   
  119.   if (color_type == PNG_COLOR_TYPE_GRAY) {
  120.     cr=cg=cb=cgr=(*pp++);
  121.     isgrey=1;
  122.   } else if (color_type == PNG_COLOR_TYPE_RGB) {
  123.     int drb, dgb, dbr;
  124.     
  125.     cr= (*pp++);
  126.     cg= (*pp++);
  127.     cb= (*pp++);
  128.     cgr = (0.59*cr) + (0.20*cg) + (0.11*cb);
  129.     drb = abs(cr - cb);
  130.     dgb = abs(cg - cb);
  131.     dbr = abs(cb - cr);
  132.     if (drb < 20 && dgb < 20 && dbr < 20)
  133.       isgrey=1;
  134.   } else {
  135.     png_error(png_ptr, "Unknown PNG color type seen (ReadPNGImage_1_2_4)");
  136.   }
  137.   
  138.   /* Process alpha channel if present */
  139.   if (alpha)
  140.     a=(*pp++);
  141.   
  142.   if (alpha && a != 0xff) {
  143.     int tr,tg,tb;
  144.     int drb, dgb, dbr;
  145.     
  146.     /* From www.c - sorry! */
  147.     if(depth==1) {
  148.       tr=255; tg=255; tb=255;
  149.     } else {
  150.       tr=220; tg=209; tb=186;
  151.     }
  152.       
  153.             /* Add picture pixel to background pixel */
  154.     cr=(a/255.0)*cr + ((255.0-a)/255.0) * tr;
  155.     cg=(a/255.0)*cg + ((255.0-a)/255.0) * tg;
  156.     cb=(a/255.0)*cb + ((255.0-a)/255.0) * tb;
  157.     
  158.     /* Recalculate grayness */
  159.     cgr = (0.59*cr) + (0.20*cg) + (0.11*cb);
  160.     isgrey=0;
  161.     drb = abs(cr - cb);
  162.     dgb = abs(cg - cb);
  163.     dbr = abs(cb - cr);
  164.     if (drb < 20 && dgb < 20 && dbr < 20)
  165.       isgrey=1;
  166.   }
  167.   
  168.   if (alpha && !a)
  169.     *dp |= transparent << shift;
  170.   else if (imaging == COLOR232)
  171.     {
  172.       if (isgrey) {
  173.                 int gr = cgr & 0xF0;
  174.                 if (cgr - gr > Magic16[(row << 4) + col])
  175.                     gr += 16;
  176.                 gr = min(gr, 0xF0);
  177.                 
  178. *dp |= greymap[gr >> 4] << shift;
  179.       } else {
  180. int r = cr & 0xC0;
  181. int g = cg & 0xE0;
  182. int b = cb & 0xC0;
  183. int v = (row << 4) + col;
  184. if (cr - r > Magic64[v])
  185.   r += 64;
  186. if (cg - g > Magic32[v])
  187.   g += 32;
  188. if (cb - b > Magic64[v])
  189.   b += 64;
  190. /* clamp error to keep color in range 0 to 255 */
  191. r = min(r, 255) & 0xC0;
  192. g = min(g, 255) & 0xE0;
  193. b = min(b, 255) & 0xC0;
  194.   
  195. *dp |= stdcmap[(r >> 6) | (g >> 3) | (b >> 1)] << shift;
  196.       }
  197.     }
  198.   else if (imaging == MONO)
  199.     {
  200.       if (cgr < Magic256[(row << 4) + col])
  201. *dp |= greymap[0] << shift;
  202.       else
  203. *dp |= greymap[15] << shift;
  204.     }
  205.   else /* Not COLOR232 or MONO */
  206.     {
  207.       gr = cgr & 0xF0;
  208.       if (cgr - gr > Magic16[(row << 4) + col])
  209. gr += 16;
  210.       gr = min(gr, 0xF0);
  211.       *dp |= greymap[gr >> 4] << shift;
  212.     }
  213.   
  214.   if (shift == 0) {
  215.     dp++;
  216.     newbyte = 1;
  217.   }
  218.   
  219. } /* xpos */
  220.       
  221.       if (shift) {
  222.   dp++;   /* make sure we start on a new byte for the next line */
  223.   newbyte = 1;
  224. }
  225.     } /* ypos */
  226.   
  227.   return(data);
  228.   
  229. }
  230. /* Read image into displays 24 bit deep */
  231. unsigned char *ReadPNGImage_24(png_struct *png_ptr, png_info *info_ptr, png_byte *png_image) 
  232. {
  233.   png_byte *pp;
  234.   unsigned char *data, *dp;
  235.   unsigned long int ulp;
  236.   int ypos;
  237.   int len=info_ptr->width;
  238.   int color_type=info_ptr->color_type;
  239.   int alpha;
  240.   
  241.   data = (unsigned char *)malloc(4 * len * info_ptr->height);
  242.   if (data == NULL)
  243.     png_error (png_ptr, "couldn't alloc space for X image data");
  244.   
  245.   /* Remove alpha channel from type,  but remember it */
  246.   alpha=color_type & PNG_COLOR_MASK_ALPHA;
  247.   color_type &= ~PNG_COLOR_MASK_ALPHA;
  248.   
  249.   pp=png_image;
  250.   dp=data;
  251.   for (ypos = 0; ypos < png_ptr->height; ypos++)
  252.     {
  253.       int xpos;
  254.       
  255.       for (xpos = 0; xpos < len; ++xpos)
  256. {
  257.   int cr, cg, cb, cgr, a;
  258.   
  259.   if (color_type == PNG_COLOR_TYPE_GRAY) {
  260.     cr=cg=cb=cgr=(*pp++);
  261.   } else if (color_type == PNG_COLOR_TYPE_RGB) {
  262.     cr= (*pp++);
  263.     cg= (*pp++);
  264.     cb= (*pp++);
  265.     cgr = (0.59*cr) + (0.20*cg) + (0.11*cb);
  266.   } else {
  267.     png_error(png_ptr, "Unknown PNG color type seen (ReadPNGImage_24)");
  268.   }
  269.   
  270.   /* Process alpha channel if present */
  271.   if (alpha)
  272.     a=(*pp++);
  273.   
  274.    /* Assume COLOR888 */
  275.   if (alpha && a != 0xff)
  276.     {
  277.       int tr,tg,tb;
  278.     
  279.       /* Get tile colour without gamma - from www.c */
  280.       if (tileData)
  281. {
  282.   int x = xpos % tileWidth;
  283.   int y = ypos % tileHeight;
  284.   int i = y * tileWidth + x;
  285.   unsigned char *s = tileData + 4 * i;
  286.   tr=s[(RPixelShift) ? 1 : 3];
  287.   tg=s[2];
  288.   tb=s[(RPixelShift) ? 3 : 1];
  289. }
  290.       else
  291. {
  292.   tr = (transparent  >> 16) & 0xFF;
  293.   tg = (transparent  >> 8) & 0xFF;
  294.   tb = transparent & 0xFF;
  295. }
  296.       
  297.       /* Add picture pixel to background pixel */
  298.       cr=(a/255.0)*cr + ((255.0-a)/255.0) * tr;
  299.       cg=(a/255.0)*cg + ((255.0-a)/255.0) * tg;
  300.       cb=(a/255.0)*cb + ((255.0-a)/255.0) * tb;
  301.     }
  302.   GetColor(cr, cg, cb, &ulp);
  303.   *dp++ = '';
  304.   *dp++ = (ulp >> 16) & 0xff; 
  305.   *dp++ = (ulp >> 8) & 0xff; 
  306.   *dp++ = ulp & 0xff; 
  307.   
  308. } /* xpos */
  309.       
  310.     } /* ypos */      
  311.   
  312.   return data;
  313. }
  314. /* Read image into displays 16 bit deep */
  315. unsigned char *ReadPNGImage_16(png_struct *png_ptr, png_info *info_ptr, png_byte *png_image) 
  316. {
  317.   png_byte *pp;
  318.   unsigned char *data, *dp;
  319.   unsigned long int ulp;
  320.   int ypos;
  321.   int len=info_ptr->width;
  322.   int color_type=info_ptr->color_type;
  323.   int alpha;
  324.   
  325.   data = (unsigned char *)malloc(2 * len * info_ptr->height);
  326.   if (data == NULL)
  327.     png_error (png_ptr, "couldn't alloc space for X image data");
  328.   
  329.   /* Remove alpha channel from type,  but remember it */
  330.   alpha=color_type & PNG_COLOR_MASK_ALPHA;
  331.   color_type &= ~PNG_COLOR_MASK_ALPHA;
  332.   
  333.   pp=png_image;
  334.   dp=data;
  335.   for (ypos = 0; ypos < png_ptr->height; ypos++)
  336.     {
  337.       int xpos;
  338.       
  339.       for (xpos = 0; xpos < len; ++xpos)
  340. {
  341.   int cr=0, cg=0, cb=0, cgr, a;
  342.   
  343.   if (color_type == PNG_COLOR_TYPE_GRAY) {
  344.     cr=cg=cb=cgr=(*pp++);
  345.   } else if (color_type == PNG_COLOR_TYPE_RGB) {
  346.     cr= (*pp++);
  347.     cg= (*pp++);
  348.     cb= (*pp++);
  349.     cgr = (0.59*cr) + (0.20*cg) + (0.11*cb);
  350.   } else {
  351.     png_error(png_ptr, "Unknown PNG color type seen (ReadPNGImage_16)");
  352.   }
  353.   
  354.   /* Process alpha channel if present */
  355.   if (alpha)
  356.     a=(*pp++);
  357.   
  358.   if (alpha && a != 0xff)
  359.     {
  360.       int tr,tg,tb;
  361.     
  362.       /* Get tile colour without gamma - from www.c */
  363.       if (tileData)
  364.       {
  365.   int x = xpos % tileWidth;
  366.   int y = ypos % tileHeight;
  367.   int i = y * tileWidth + x;
  368.   unsigned char *s = (tileData + 2 * i);
  369.   unsigned short val;
  370.   val = *s * 256  + *(s+1);
  371.   tr= (((val & visual->red_mask) >> RPixelShift) << 8) / (RPixelMask+1);
  372.   tg= (((val & visual->green_mask) >> GPixelShift) << 8) /  (GPixelMask+1);
  373.   tb= (((val & visual->blue_mask) >> BPixelShift) << 8 ) / (BPixelMask+1);
  374.       }
  375.       else
  376.       {
  377.     tr = (((transparent & visual->red_mask) >> RPixelShift) << 8) / (RPixelMask+1);
  378.     tg = (((transparent & visual->green_mask) >> GPixelShift) << 8) /  (GPixelMask+1);
  379.   tb = (((transparent & visual->blue_mask)  >> BPixelShift) << 8 ) / (BPixelMask+1);
  380.       }
  381.       
  382.       /* Add picture pixel to background pixel */
  383.       cr=(a/255.0)*cr + ((255.0-a)/255.0) * tr;
  384.       cg=(a/255.0)*cg + ((255.0-a)/255.0) * tg;
  385.       cb=(a/255.0)*cb + ((255.0-a)/255.0) * tb;
  386.     }
  387.   
  388.   GetColor(cr, cg, cb, &ulp);
  389.   *dp++ = (unsigned char)((ulp >> 8) & 0xff);
  390.   *dp++ = (unsigned char)((ulp) & 0xff);
  391.   
  392. } /* xpos */
  393.       
  394.     } /* ypos */      
  395.   
  396.   return data;
  397. }
  398. /* Read image into displays, not 1,2,4,16 or 24 bit deep */
  399. unsigned char *ReadPNGImage(png_struct *png_ptr, png_info *info_ptr, png_byte *png_image) 
  400. {
  401.   png_byte *pp;
  402.   unsigned char *data, *dp;
  403.   int ypos;
  404.   int len=info_ptr->width;
  405.   int color_type=info_ptr->color_type;
  406.   int alpha;
  407.   
  408.   data = (unsigned char *)malloc(len * info_ptr->height);
  409.   if (data == NULL)
  410.     png_error (png_ptr, "couldn't alloc space for X image data");
  411.   
  412.   /* Remove alpha channel from type,  but remember it */
  413.   alpha=color_type & PNG_COLOR_MASK_ALPHA;
  414.   color_type &= ~PNG_COLOR_MASK_ALPHA;
  415.   
  416.   pp=png_image;
  417.   dp=data;
  418.   for (ypos = 0; ypos < png_ptr->height; ypos++)
  419.     {
  420.       int col = ypos & 0x0f;
  421.       int xpos;
  422.       
  423.       for (xpos = 0; xpos < len; ++xpos)
  424. {
  425.   int row = xpos & 0x0f;
  426.   int cr=0, cg=0, cb=0, cgr, a;
  427.   int gr;
  428.   int isgrey = 0;
  429.   
  430.   if (color_type == PNG_COLOR_TYPE_GRAY) {
  431.     cr=cg=cb=cgr=(*pp++);
  432.     isgrey=1;
  433.   } else if (color_type == PNG_COLOR_TYPE_RGB) {
  434.     int drb, dgb, dbr;
  435.     
  436.     cr= (*pp++);
  437.     cg= (*pp++);
  438.     cb= (*pp++);
  439.     cgr = (0.59*cr) + (0.20*cg) + (0.11*cb);
  440.     drb = abs(cr - cb);
  441.     dgb = abs(cg - cb);
  442.     dbr = abs(cb - cr);
  443.     if (drb < 20 && dgb < 20 && dbr < 20)
  444.       isgrey=1;
  445.   } else {
  446.     png_error(png_ptr, "Unknown PNG color type seen (ReadPNGImage)");
  447.   }
  448.   
  449.   
  450.   /* Process alpha channel if present */
  451.   if (alpha) {
  452.     a=(*pp++);
  453.   
  454.     if(!a) {
  455.       dp=Transparent(dp, xpos, ypos);
  456.       continue;
  457.     } else if(a != 0xff) {
  458.       int tcolor, tr,tg,tb;
  459.       int drb, dgb, dbr;
  460.       
  461.       /* Get tile colour without gamma - from www.c */
  462.       if (tileData)
  463. {
  464.   unsigned int x = xpos % tileWidth;
  465.   unsigned int y = ypos % tileHeight;
  466.   unsigned int i = y * tileWidth + x;
  467.   tcolor=tileData[i];
  468. }
  469.       else
  470. tcolor=transparent;
  471.       
  472.       tr=papercols[tcolor].red;
  473.       tg=papercols[tcolor].green;
  474.       tb=papercols[tcolor].blue;
  475.       /* Add picture pixel to background pixel */
  476.       cr=(a/255.0)*cr + ((255.0-a)/255.0) * tr;
  477.       cg=(a/255.0)*cg + ((255.0-a)/255.0) * tg;
  478.       cb=(a/255.0)*cb + ((255.0-a)/255.0) * tb;
  479.       
  480.       /* Recalculate grayness */
  481.       cgr = (0.59*cr) + (0.20*cg) + (0.11*cb);
  482.       isgrey=0;
  483.       drb = abs(cr - cb);
  484.       dgb = abs(cg - cb);
  485.       dbr = abs(cb - cr);
  486.       if (drb < 20 && dgb < 20 && dbr < 20)
  487. isgrey=1;
  488.     }
  489.   }
  490.   
  491.   if (imaging == COLOR232)
  492.     {
  493.       if (isgrey) {
  494.                 int gr = cgr & 0xF0;
  495.                 if (cgr - gr > Magic16[(row << 4) + col])
  496.                     gr += 16;
  497.                 gr = min(gr, 0xF0);
  498.                 
  499. *dp++ = greymap[gr >> 4];
  500.       } else {
  501.       int r = cr & 0xC0;
  502.       int g = cg & 0xE0;
  503.       int b = cb & 0xC0;
  504.       int v = (row << 4) + col;
  505.       if (cr - r > Magic64[v])
  506. r += 64;
  507.       if (cg - g > Magic32[v])
  508. g += 32;
  509.       if (cb - b > Magic64[v])
  510. b += 64;
  511.       /* clamp error to keep color in range 0 to 255 */
  512.       r = min(r, 255) & 0xC0;
  513.       g = min(g, 255) & 0xE0;
  514.       b = min(b, 255) & 0xC0;
  515.     
  516.       *dp++ = stdcmap[(r >> 6) | (g >> 3) | (b >> 1)];
  517.       }
  518.     }
  519.   else if (imaging == MONO)
  520.     {
  521.       if (cgr < Magic256[(row << 4) + col])
  522. *dp++ = greymap[0];
  523.       else
  524. *dp++ = greymap[15];
  525.     }
  526.   else /* Not COLOR232 or MONO */
  527.     {
  528.       gr = cgr & 0xF0;
  529.       if (cgr - gr > Magic16[(row << 4) + col])
  530. gr += 16;
  531.       
  532.       gr = min(gr, 0xF0);
  533.       *dp++ = greymap[gr >> 4];
  534.     }
  535. } /* xpos */
  536.       
  537.     } /* ypos */      
  538.   
  539.   return data;
  540. }
  541. static char *png_color_type[] = {"grayscale", "undefined type", "RGB",
  542.  "colormap", "grayscale+alpha",
  543.  "undefined type", "RGB+alpha"};
  544. unsigned char *LoadPNGImage(Image *image, Block *bp, unsigned int depth) {
  545.   png_struct *png_ptr; 
  546.   png_info *info_ptr;
  547.   png_byte *row_pointer;
  548.   png_byte *png_image=NULL;
  549.   int number_passes, pass, row, bit_depth, image_width;
  550.   unsigned char *ximage=NULL;
  551.   int color_type, orig_color_type;
  552.   double image_gamma;
  553.   
  554.   png_ptr = malloc(sizeof (png_struct));
  555.   if (!png_ptr)
  556.     return 0;
  557.   info_ptr = malloc(sizeof (png_info));
  558.   if (!info_ptr) {
  559.     free(png_ptr);
  560.     return 0;
  561.   }
  562.    
  563.   if (setjmp(png_ptr->jmpbuf)) {
  564.     png_read_destroy(png_ptr, info_ptr, (png_info *)0);
  565.     free(png_ptr);
  566.     free(info_ptr);
  567.     if (ximage)
  568.       free(ximage);
  569.     if (png_image)
  570.       free(png_image);
  571.     return 0;
  572.   }
  573.   
  574.   png_read_init(png_ptr);
  575.   png_info_init(info_ptr);
  576.   /* No need to init PNG I/O - only ever do a read */
  577.   png_set_read_fn(png_ptr, bp, &arena_png_read_data);
  578.   png_read_info(png_ptr, info_ptr);
  579.   bit_depth=info_ptr->bit_depth;
  580.   orig_color_type = color_type = info_ptr->color_type;
  581.   if (IMAGE_TRACE)
  582.     fprintf(stderr, "PNG Image: %ld x %ld, %d-bit%s, %s, %sinterlacedn",
  583.     info_ptr->width, info_ptr->height,
  584.     bit_depth, bit_depth>1 ? "s" : "",
  585.     (color_type>6) ? png_color_type[1] : png_color_type[color_type],
  586.     info_ptr->interlace_type ? "" : "non-");
  587.   
  588.   /* expand to RGB or 8 bit grayscale (with alpha if present) */
  589.   png_set_expand(png_ptr);
  590.   
  591.   /* tell libpng to handle the gamma conversion */
  592.   if (info_ptr->valid & PNG_INFO_gAMA)
  593.     image_gamma=info_ptr->gamma;
  594.   else
  595.     image_gamma=(double)0.45; /* FIXME: OR: 1/Gamma ? */
  596.   
  597.   /* Only gamma correct if >1.1% difference */
  598.   if (abs((Gamma * image_gamma) -1) > 0.011)
  599.     png_set_gamma(png_ptr, Gamma, image_gamma);
  600.   
  601.   if (info_ptr->valid & PNG_INFO_bKGD)
  602.     png_set_background(png_ptr, &(png_ptr->background),
  603.        PNG_BACKGROUND_GAMMA_FILE, 1, image_gamma);
  604.   /* tell libpng to convert 16 bits per channel to 8 */
  605.   if (info_ptr->bit_depth == 16)
  606.       png_set_strip_16(png_ptr); 
  607.   
  608.   /* read all the image into the rows allocated above */
  609.   /* (Automatically handles interlacing) */
  610.   number_passes = png_set_interlace_handling(png_ptr);
  611.   
  612.   /* Optional call to update the info_ptr area with the new settings */
  613.   png_read_update_info(png_ptr, info_ptr);
  614.   bit_depth=info_ptr->bit_depth;
  615.   color_type=info_ptr->color_type;
  616.   image_width= info_ptr->width * info_ptr->channels;
  617.   png_image = malloc (info_ptr->height * image_width);
  618.   if (png_image == NULL)
  619.     png_error (png_ptr, "couldn't alloc space for PNG image");
  620.   /* optional call to update palette with transformations, except
  621.      it doesn't do any! */
  622.   png_start_read_image(png_ptr);
  623.   
  624.   for (pass = 0; pass < number_passes; pass++)
  625.     {
  626. /*      if ((depth == 24)||(depth == 16))
  627. row_pointer=(png_byte*)ximage;
  628. else    --patch dsr 24-bits display 8-Dec-95 */
  629. row_pointer=png_image;
  630. for (row = 0; row < info_ptr->height; row++)
  631. {
  632.     png_read_row(png_ptr, NULL, row_pointer);
  633.     row_pointer += image_width;
  634. }
  635.     }
  636.   
  637.   /* Now convert PNG image to X one */
  638.   if (depth == 1 || depth == 2 || depth == 4)
  639.       ximage = ReadPNGImage_1_2_4(png_ptr, info_ptr, png_image);
  640.   else if (depth == 24)
  641.       ximage = ReadPNGImage_24(png_ptr, info_ptr, png_image);
  642.   else if (depth == 16) 
  643.       ximage = ReadPNGImage_16(png_ptr, info_ptr, png_image);
  644.   else
  645.       ximage = ReadPNGImage(png_ptr, info_ptr, png_image);
  646.   
  647.   image->npixels = 0; /* Set to 0 in gif.c, icon.c, image.c  */
  648.   image->width = info_ptr->width;
  649.   image->height = info_ptr->height;
  650.   /* Finish */
  651.   png_read_end(png_ptr, info_ptr);
  652.   png_read_destroy(png_ptr, info_ptr, (png_info *)0);
  653.   free(png_ptr);   /* howcome 23/10/95: applied patch from Dave Beckett */
  654.   free(info_ptr);  /* howcome 23/10/95: applied patch from Dave Beckett */
  655.   free(png_image);
  656.   
  657.   return(ximage);
  658. }
  659. #endif /* PNG */