2008-11-05 09:21:04 -08:00
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
# include <config.h>
|
|
|
|
#endif
|
|
|
|
|
2006-10-27 20:02:23 -07:00
|
|
|
#ifndef _GNU_SOURCE
|
|
|
|
#define _GNU_SOURCE
|
|
|
|
#endif
|
|
|
|
|
2008-11-05 09:21:04 -08:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <png.h>
|
2008-11-05 12:39:10 -08:00
|
|
|
#include <setjmp.h>
|
2008-11-05 09:21:04 -08:00
|
|
|
|
|
|
|
#ifdef HAVE_EVIL
|
|
|
|
# include <Evil.h>
|
|
|
|
#endif
|
2006-01-14 12:03:42 -08:00
|
|
|
|
2008-11-09 11:32:12 -08:00
|
|
|
#ifdef _WIN32_WCE
|
|
|
|
# define E_FOPEN(file, mode) evil_fopen_native((file), (mode))
|
|
|
|
# define E_FREAD(buffer, size, count, stream) evil_fread_native(buffer, size, count, stream)
|
|
|
|
# define E_FCLOSE(stream) evil_fclose_native(stream)
|
|
|
|
#else
|
|
|
|
# define E_FOPEN(file, mode) fopen((file), (mode))
|
|
|
|
# define E_FREAD(buffer, size, count, stream) fread(buffer, size, count, stream)
|
|
|
|
# define E_FCLOSE(stream) fclose(stream)
|
|
|
|
#endif
|
|
|
|
|
2006-01-14 12:03:42 -08:00
|
|
|
#include "evas_common.h"
|
|
|
|
#include "evas_private.h"
|
|
|
|
|
|
|
|
|
|
|
|
#define PNG_BYTES_TO_CHECK 4
|
|
|
|
|
|
|
|
|
2008-06-03 02:09:39 -07:00
|
|
|
int evas_image_load_file_head_png(Image_Entry *ie, const char *file, const char *key);
|
|
|
|
int evas_image_load_file_data_png(Image_Entry *ie, const char *file, const char *key);
|
2006-01-14 12:03:42 -08:00
|
|
|
|
|
|
|
Evas_Image_Load_Func evas_image_load_png_func =
|
|
|
|
{
|
|
|
|
evas_image_load_file_head_png,
|
|
|
|
evas_image_load_file_data_png
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2008-06-03 02:09:39 -07:00
|
|
|
evas_image_load_file_head_png(Image_Entry *ie, const char *file, const char *key)
|
2006-01-14 12:03:42 -08:00
|
|
|
{
|
|
|
|
png_uint_32 w32, h32;
|
|
|
|
FILE *f;
|
|
|
|
png_structp png_ptr = NULL;
|
|
|
|
png_infop info_ptr = NULL;
|
|
|
|
int bit_depth, color_type, interlace_type;
|
|
|
|
unsigned char buf[PNG_BYTES_TO_CHECK];
|
2007-08-24 19:36:18 -07:00
|
|
|
char hasa;
|
2006-01-14 12:03:42 -08:00
|
|
|
|
|
|
|
if ((!file)) return 0;
|
|
|
|
hasa = 0;
|
2008-11-09 11:32:12 -08:00
|
|
|
f = E_FOPEN(file, "rb");
|
2006-01-14 12:03:42 -08:00
|
|
|
if (!f) return 0;
|
|
|
|
|
|
|
|
/* if we havent read the header before, set the header data */
|
2008-11-09 11:32:12 -08:00
|
|
|
if (E_FREAD(buf, PNG_BYTES_TO_CHECK, 1, f) != 1)
|
|
|
|
goto close_file;
|
|
|
|
|
2006-01-14 12:03:42 -08:00
|
|
|
if (!png_check_sig(buf, PNG_BYTES_TO_CHECK))
|
2008-11-09 11:32:12 -08:00
|
|
|
goto close_file;
|
|
|
|
|
2006-01-14 12:03:42 -08:00
|
|
|
png_ptr = png_create_read_struct(PNG_LIBPNG_VER_STRING, NULL, NULL, NULL);
|
|
|
|
if (!png_ptr)
|
2008-11-09 11:32:12 -08:00
|
|
|
goto close_file;
|
|
|
|
|
2006-01-14 12:03:42 -08:00
|
|
|
info_ptr = png_create_info_struct(png_ptr);
|
|
|
|
if (!info_ptr)
|
|
|
|
{
|
|
|
|
png_destroy_read_struct(&png_ptr, NULL, NULL);
|
2008-11-09 11:32:12 -08:00
|
|
|
goto close_file;
|
2006-01-14 12:03:42 -08:00
|
|
|
}
|
2008-11-06 10:54:19 -08:00
|
|
|
if (setjmp(png_jmpbuf(png_ptr)))
|
2006-01-14 12:03:42 -08:00
|
|
|
{
|
|
|
|
png_destroy_read_struct(&png_ptr, &info_ptr, NULL);
|
2008-11-09 11:32:12 -08:00
|
|
|
goto close_file;
|
2006-01-14 12:03:42 -08:00
|
|
|
}
|
|
|
|
png_init_io(png_ptr, f);
|
2008-11-06 10:54:19 -08:00
|
|
|
png_set_sig_bytes(png_ptr, PNG_BYTES_TO_CHECK);
|
2006-01-14 12:03:42 -08:00
|
|
|
png_read_info(png_ptr, info_ptr);
|
|
|
|
png_get_IHDR(png_ptr, info_ptr, (png_uint_32 *) (&w32),
|
|
|
|
(png_uint_32 *) (&h32), &bit_depth, &color_type,
|
|
|
|
&interlace_type, NULL, NULL);
|
2006-11-04 21:07:53 -08:00
|
|
|
if ((w32 < 1) || (h32 < 1) || (w32 > 8192) || (h32 > 8192))
|
|
|
|
{
|
2008-11-09 11:32:12 -08:00
|
|
|
png_destroy_read_struct(&png_ptr, &info_ptr, NULL);
|
|
|
|
goto close_file;
|
2006-11-04 21:07:53 -08:00
|
|
|
}
|
2008-06-03 02:09:39 -07:00
|
|
|
ie->w = (int) w32;
|
|
|
|
ie->h = (int) h32;
|
2007-08-24 19:36:18 -07:00
|
|
|
if (png_get_valid(png_ptr, info_ptr, PNG_INFO_tRNS)) hasa = 1;
|
|
|
|
if (color_type == PNG_COLOR_TYPE_RGB_ALPHA) hasa = 1;
|
|
|
|
if (color_type == PNG_COLOR_TYPE_GRAY_ALPHA) hasa = 1;
|
2008-06-03 02:09:39 -07:00
|
|
|
if (hasa) ie->flags.alpha = 1;
|
2008-11-09 11:32:12 -08:00
|
|
|
png_destroy_read_struct(&png_ptr, &info_ptr, NULL);
|
|
|
|
E_FCLOSE(f);
|
2006-01-14 12:03:42 -08:00
|
|
|
return 1;
|
2008-11-09 11:32:12 -08:00
|
|
|
|
|
|
|
close_file:
|
|
|
|
E_FCLOSE(f);
|
|
|
|
return 0;
|
|
|
|
|
2006-01-14 12:03:42 -08:00
|
|
|
key = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2008-06-03 02:09:39 -07:00
|
|
|
evas_image_load_file_data_png(Image_Entry *ie, const char *file, const char *key)
|
2006-01-14 12:03:42 -08:00
|
|
|
{
|
2008-06-03 02:09:39 -07:00
|
|
|
unsigned char *surface;
|
2006-01-14 12:03:42 -08:00
|
|
|
png_uint_32 w32, h32;
|
|
|
|
int w, h;
|
|
|
|
FILE *f;
|
|
|
|
png_structp png_ptr = NULL;
|
|
|
|
png_infop info_ptr = NULL;
|
|
|
|
int bit_depth, color_type, interlace_type;
|
|
|
|
unsigned char buf[PNG_BYTES_TO_CHECK];
|
|
|
|
unsigned char **lines;
|
2007-08-24 19:36:18 -07:00
|
|
|
char hasa;
|
2006-01-14 12:03:42 -08:00
|
|
|
int i;
|
|
|
|
|
|
|
|
if ((!file)) return 0;
|
|
|
|
hasa = 0;
|
2008-11-09 11:32:12 -08:00
|
|
|
f = E_FOPEN(file, "rb");
|
2006-01-14 12:03:42 -08:00
|
|
|
if (!f) return 0;
|
|
|
|
|
|
|
|
/* if we havent read the header before, set the header data */
|
2008-11-09 11:32:12 -08:00
|
|
|
E_FREAD(buf, 1, PNG_BYTES_TO_CHECK, f);
|
2006-01-14 12:03:42 -08:00
|
|
|
if (!png_check_sig(buf, PNG_BYTES_TO_CHECK))
|
2008-11-09 11:32:12 -08:00
|
|
|
goto close_file;
|
2006-01-14 12:03:42 -08:00
|
|
|
png_ptr = png_create_read_struct(PNG_LIBPNG_VER_STRING, NULL, NULL, NULL);
|
|
|
|
if (!png_ptr)
|
2008-11-09 11:32:12 -08:00
|
|
|
goto close_file;
|
|
|
|
|
2006-01-14 12:03:42 -08:00
|
|
|
info_ptr = png_create_info_struct(png_ptr);
|
|
|
|
if (!info_ptr)
|
|
|
|
{
|
|
|
|
png_destroy_read_struct(&png_ptr, NULL, NULL);
|
2008-11-09 11:32:12 -08:00
|
|
|
goto close_file;
|
2006-01-14 12:03:42 -08:00
|
|
|
}
|
2008-11-06 10:54:19 -08:00
|
|
|
if (setjmp(png_jmpbuf(png_ptr)))
|
2006-01-14 12:03:42 -08:00
|
|
|
{
|
|
|
|
png_destroy_read_struct(&png_ptr, &info_ptr, NULL);
|
2008-11-09 11:32:12 -08:00
|
|
|
goto close_file;
|
2006-01-14 12:03:42 -08:00
|
|
|
}
|
|
|
|
png_init_io(png_ptr, f);
|
2008-11-06 10:54:19 -08:00
|
|
|
png_set_sig_bytes(png_ptr, PNG_BYTES_TO_CHECK);
|
2006-01-14 12:03:42 -08:00
|
|
|
png_read_info(png_ptr, info_ptr);
|
|
|
|
png_get_IHDR(png_ptr, info_ptr, (png_uint_32 *) (&w32),
|
|
|
|
(png_uint_32 *) (&h32), &bit_depth, &color_type,
|
|
|
|
&interlace_type, NULL, NULL);
|
2008-06-03 02:09:39 -07:00
|
|
|
evas_cache_image_surface_alloc(ie, w32, h32);
|
|
|
|
surface = (unsigned char *) evas_cache_image_pixels(ie);
|
|
|
|
if (!surface)
|
2008-04-11 17:32:30 -07:00
|
|
|
{
|
2008-11-09 11:32:12 -08:00
|
|
|
png_destroy_read_struct(&png_ptr, &info_ptr, NULL);
|
|
|
|
goto close_file;
|
2008-04-11 17:32:30 -07:00
|
|
|
}
|
2008-06-03 02:09:39 -07:00
|
|
|
if ((w32 != ie->w) || (h32 != ie->h))
|
2006-11-04 21:07:53 -08:00
|
|
|
{
|
2008-11-09 11:32:12 -08:00
|
|
|
png_destroy_read_struct(&png_ptr, &info_ptr, NULL);
|
|
|
|
goto close_file;
|
2006-11-04 21:07:53 -08:00
|
|
|
}
|
2007-08-24 19:36:18 -07:00
|
|
|
if (png_get_valid(png_ptr, info_ptr, PNG_INFO_tRNS)) hasa = 1;
|
|
|
|
if (color_type == PNG_COLOR_TYPE_RGB_ALPHA) hasa = 1;
|
|
|
|
if (color_type == PNG_COLOR_TYPE_GRAY_ALPHA) hasa = 1;
|
2008-06-03 02:09:39 -07:00
|
|
|
if (hasa) ie->flags.alpha = 1;
|
2006-01-14 12:03:42 -08:00
|
|
|
|
2007-08-24 19:36:18 -07:00
|
|
|
/* Prep for transformations... ultimately we want ARGB */
|
|
|
|
/* expand palette -> RGB if necessary */
|
|
|
|
if (color_type == PNG_COLOR_TYPE_PALETTE) png_set_palette_to_rgb(png_ptr);
|
|
|
|
/* expand gray (w/reduced bits) -> 8-bit RGB if necessary */
|
|
|
|
if ((color_type == PNG_COLOR_TYPE_GRAY) ||
|
|
|
|
(color_type == PNG_COLOR_TYPE_GRAY_ALPHA))
|
|
|
|
{
|
|
|
|
png_set_gray_to_rgb(png_ptr);
|
|
|
|
if (bit_depth < 8) png_set_expand_gray_1_2_4_to_8(png_ptr);
|
|
|
|
}
|
|
|
|
/* expand transparency entry -> alpha channel if present */
|
|
|
|
if (png_get_valid(png_ptr, info_ptr, PNG_INFO_tRNS))
|
|
|
|
png_set_tRNS_to_alpha(png_ptr);
|
|
|
|
/* reduce 16bit color -> 8bit color if necessary */
|
|
|
|
if (bit_depth > 8) png_set_strip_16(png_ptr);
|
|
|
|
/* pack all pixels to byte boundaries */
|
|
|
|
png_set_packing(png_ptr);
|
2008-04-11 17:32:30 -07:00
|
|
|
|
2008-06-03 02:09:39 -07:00
|
|
|
w = ie->w;
|
|
|
|
h = ie->h;
|
2006-01-14 12:03:42 -08:00
|
|
|
/* we want ARGB */
|
|
|
|
#ifdef WORDS_BIGENDIAN
|
|
|
|
png_set_swap_alpha(png_ptr);
|
2007-08-24 19:36:18 -07:00
|
|
|
if (!hasa) png_set_filler(png_ptr, 0xff, PNG_FILLER_BEFORE);
|
2006-01-14 12:03:42 -08:00
|
|
|
#else
|
|
|
|
png_set_bgr(png_ptr);
|
2007-08-24 19:36:18 -07:00
|
|
|
if (!hasa) png_set_filler(png_ptr, 0xff, PNG_FILLER_AFTER);
|
2006-01-14 12:03:42 -08:00
|
|
|
#endif
|
|
|
|
lines = (unsigned char **) alloca(h * sizeof(unsigned char *));
|
|
|
|
|
|
|
|
for (i = 0; i < h; i++)
|
2008-06-03 02:09:39 -07:00
|
|
|
lines[i] = surface + (i * w * sizeof(DATA32));
|
2006-01-14 12:03:42 -08:00
|
|
|
png_read_image(png_ptr, lines);
|
|
|
|
png_read_end(png_ptr, info_ptr);
|
2008-11-09 11:32:12 -08:00
|
|
|
png_destroy_read_struct(&png_ptr, &info_ptr, NULL);
|
|
|
|
E_FCLOSE(f);
|
2008-06-03 02:09:39 -07:00
|
|
|
evas_common_image_premul(ie);
|
2008-04-11 17:32:30 -07:00
|
|
|
|
2006-01-14 12:03:42 -08:00
|
|
|
return 1;
|
2008-11-09 11:32:12 -08:00
|
|
|
|
|
|
|
close_file:
|
|
|
|
E_FCLOSE(f);
|
|
|
|
return 0;
|
|
|
|
|
2006-01-14 12:03:42 -08:00
|
|
|
key = 0;
|
|
|
|
}
|
|
|
|
|
2006-09-06 00:28:46 -07:00
|
|
|
EAPI int
|
|
|
|
module_open(Evas_Module *em)
|
2006-01-14 12:03:42 -08:00
|
|
|
{
|
|
|
|
if (!em) return 0;
|
|
|
|
em->functions = (void *)(&evas_image_load_png_func);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2006-09-06 00:28:46 -07:00
|
|
|
EAPI void
|
|
|
|
module_close(void)
|
2006-01-14 12:03:42 -08:00
|
|
|
{
|
2008-11-09 11:32:12 -08:00
|
|
|
|
2006-01-14 12:03:42 -08:00
|
|
|
}
|
|
|
|
|
2006-09-06 00:28:46 -07:00
|
|
|
EAPI Evas_Module_Api evas_modapi =
|
2006-01-14 12:03:42 -08:00
|
|
|
{
|
|
|
|
EVAS_MODULE_API_VERSION,
|
|
|
|
EVAS_MODULE_TYPE_IMAGE_LOADER,
|
|
|
|
"png",
|
|
|
|
"none"
|
|
|
|
};
|