mirror of
https://github.com/RaySollium99/libpicofe.git
synced 2025-09-03 06:17:44 -04:00
309 lines
7.6 KiB
C
309 lines
7.6 KiB
C
/*
|
|
* (C) Gražvydas "notaz" Ignotas, 2008-2011
|
|
*
|
|
* This work is licensed under the terms of any of these licenses
|
|
* (at your option):
|
|
* - GNU GPL, version 2 or later.
|
|
* - GNU LGPL, version 2.1 or later.
|
|
* - MAME license.
|
|
* See the COPYING file in the top-level directory.
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <png.h>
|
|
#include "plat.h"
|
|
#include "readpng.h"
|
|
#include "lprintf.h"
|
|
|
|
int readpng(void *dest, const char *fname, readpng_what what, int req_w, int req_h)
|
|
{
|
|
FILE *fp;
|
|
png_structp png_ptr = NULL;
|
|
png_infop info_ptr = NULL;
|
|
png_bytepp row_ptr = NULL;
|
|
int ret = -1;
|
|
|
|
if (dest == NULL || fname == NULL)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
fp = fopen(fname, "rb");
|
|
if (fp == NULL)
|
|
{
|
|
lprintf(__FILE__ ": failed to open: %s\n", fname);
|
|
return -1;
|
|
}
|
|
|
|
png_ptr = png_create_read_struct(PNG_LIBPNG_VER_STRING, NULL, NULL, NULL);
|
|
if (!png_ptr)
|
|
{
|
|
lprintf(__FILE__ ": png_create_read_struct() failed\n");
|
|
fclose(fp);
|
|
return -1;
|
|
}
|
|
|
|
info_ptr = png_create_info_struct(png_ptr);
|
|
if (!info_ptr)
|
|
{
|
|
lprintf(__FILE__ ": png_create_info_struct() failed\n");
|
|
goto done;
|
|
}
|
|
|
|
// Start reading
|
|
png_init_io(png_ptr, fp);
|
|
png_read_png(png_ptr, info_ptr, PNG_TRANSFORM_STRIP_16 | PNG_TRANSFORM_STRIP_ALPHA | PNG_TRANSFORM_PACKING, NULL);
|
|
row_ptr = png_get_rows(png_ptr, info_ptr);
|
|
if (row_ptr == NULL)
|
|
{
|
|
lprintf(__FILE__ ": png_get_rows() failed\n");
|
|
goto done;
|
|
}
|
|
|
|
// lprintf("%s: %ix%i @ %ibpp\n", fname, (int)png_get_image_width(png_ptr, info_ptr),
|
|
// (int)png_get_image_height(png_ptr, info_ptr), png_get_bit_depth(png_ptr, info_ptr));
|
|
|
|
switch (what)
|
|
{
|
|
case READPNG_SCALE:
|
|
{
|
|
int height, width, x_ofs = 0, y_ofs = 0;
|
|
unsigned short *dst = dest;
|
|
int x_scale, y_scale, x_pos, y_pos; // Q16
|
|
|
|
if (png_get_bit_depth(png_ptr, info_ptr) != 8)
|
|
{
|
|
lprintf(__FILE__ ": scaled image uses %ibpc, needed 8bpc\n", png_get_bit_depth(png_ptr, info_ptr));
|
|
goto done;
|
|
}
|
|
width = png_get_image_width(png_ptr, info_ptr);
|
|
x_scale = width*65536 / req_w;
|
|
height = png_get_image_height(png_ptr, info_ptr);
|
|
y_scale = height*65536 / req_h;
|
|
if (x_scale < y_scale)
|
|
x_scale = y_scale;
|
|
else y_scale = x_scale;
|
|
x_ofs = req_w - width*65536 / x_scale;
|
|
y_ofs = req_h - height*65536 / y_scale;
|
|
|
|
dst += y_ofs/2*req_w + x_ofs/2;
|
|
for (y_pos = 0; y_pos < height*65536; y_pos += y_scale+1)
|
|
{
|
|
unsigned char *src = row_ptr[y_pos >> 16];
|
|
int len = 0;
|
|
for (x_pos = 0; x_pos < width*65536; x_pos += x_scale+1, len++)
|
|
{
|
|
int o = 3*(x_pos >> 16);
|
|
// TODO: could use bilinear if upsampling?
|
|
*dst++ = PXMAKE(src[o], src[o+1], src[o+2]);
|
|
}
|
|
dst += req_w - len;
|
|
}
|
|
break;
|
|
}
|
|
|
|
case READPNG_BG:
|
|
{
|
|
int height, width, h, x_ofs = 0, y_ofs = 0;
|
|
unsigned short *dst = dest;
|
|
|
|
if (png_get_bit_depth(png_ptr, info_ptr) != 8)
|
|
{
|
|
lprintf(__FILE__ ": bg image uses %ibpc, needed 8bpc\n", png_get_bit_depth(png_ptr, info_ptr));
|
|
goto done;
|
|
}
|
|
width = png_get_image_width(png_ptr, info_ptr);
|
|
if (width > req_w) {
|
|
x_ofs = (width - req_w) / 2;
|
|
width = req_w;
|
|
} else
|
|
dst += (req_w - width) / 2;
|
|
height = png_get_image_height(png_ptr, info_ptr);
|
|
if (height > req_h) {
|
|
y_ofs = (height - req_h) / 2;
|
|
height = req_h;
|
|
} else
|
|
dst += (req_h - height) / 2 * req_w;
|
|
|
|
for (h = 0; h < height; h++)
|
|
{
|
|
unsigned char *src = row_ptr[h + y_ofs] + x_ofs * 3;
|
|
int len = width;
|
|
while (len--)
|
|
{
|
|
*dst++ = PXMAKE(src[0], src[1], src[2]);
|
|
src += 3;
|
|
}
|
|
dst += req_w - width;
|
|
}
|
|
break;
|
|
}
|
|
|
|
case READPNG_FONT:
|
|
{
|
|
int x, y, x1, y1;
|
|
unsigned char *dst = dest;
|
|
if (png_get_image_width(png_ptr, info_ptr) != req_w || png_get_image_height(png_ptr, info_ptr) != req_h)
|
|
{
|
|
lprintf(__FILE__ ": unexpected font image size %dx%d, needed %dx%d\n",
|
|
(int)png_get_image_width(png_ptr, info_ptr), (int)png_get_image_height(png_ptr, info_ptr), req_w, req_h);
|
|
goto done;
|
|
}
|
|
if (png_get_bit_depth(png_ptr, info_ptr) != 8)
|
|
{
|
|
lprintf(__FILE__ ": font image uses %ibpp, needed 8bpp\n", png_get_bit_depth(png_ptr, info_ptr));
|
|
goto done;
|
|
}
|
|
for (y = 0; y < 16; y++)
|
|
{
|
|
for (x = 0; x < 16; x++)
|
|
{
|
|
/* 16x16 grid of syms */
|
|
int sym_w = req_w / 16;
|
|
int sym_h = req_h / 16;
|
|
for (y1 = 0; y1 < sym_h; y1++)
|
|
{
|
|
unsigned char *src = row_ptr[y*sym_h + y1] + x*sym_w;
|
|
for (x1 = sym_w/2; x1 > 0; x1--, src+=2)
|
|
*dst++ = ((src[0]^0xff) & 0xf0) | ((src[1]^0xff) >> 4);
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
case READPNG_SELECTOR:
|
|
{
|
|
int x1, y1;
|
|
unsigned char *dst = dest;
|
|
if (png_get_image_width(png_ptr, info_ptr) != req_w || png_get_image_height(png_ptr, info_ptr) != req_h)
|
|
{
|
|
lprintf(__FILE__ ": unexpected selector image size %ix%i, needed %dx%d\n",
|
|
(int)png_get_image_width(png_ptr, info_ptr), (int)png_get_image_height(png_ptr, info_ptr), req_w, req_h);
|
|
goto done;
|
|
}
|
|
if (png_get_bit_depth(png_ptr, info_ptr) != 8)
|
|
{
|
|
lprintf(__FILE__ ": selector image uses %ibpp, needed 8bpp\n", png_get_bit_depth(png_ptr, info_ptr));
|
|
goto done;
|
|
}
|
|
for (y1 = 0; y1 < req_h; y1++)
|
|
{
|
|
unsigned char *src = row_ptr[y1];
|
|
for (x1 = req_w/2; x1 > 0; x1--, src+=2)
|
|
*dst++ = ((src[0]^0xff) & 0xf0) | ((src[1]^0xff) >> 4);
|
|
}
|
|
break;
|
|
}
|
|
|
|
case READPNG_24:
|
|
{
|
|
int height, width, h;
|
|
unsigned char *dst = dest;
|
|
if (png_get_bit_depth(png_ptr, info_ptr) != 8)
|
|
{
|
|
lprintf(__FILE__ ": image uses %ibpc, needed 8bpc\n", png_get_bit_depth(png_ptr, info_ptr));
|
|
goto done;
|
|
}
|
|
width = png_get_image_width(png_ptr, info_ptr);
|
|
if (width > req_w)
|
|
width = req_w;
|
|
height = png_get_image_height(png_ptr, info_ptr);
|
|
if (height > req_h)
|
|
height = req_h;
|
|
|
|
for (h = 0; h < height; h++)
|
|
{
|
|
int len = width;
|
|
unsigned char *src = row_ptr[h];
|
|
dst += (req_w - width) * 3;
|
|
for (len = width; len > 0; len--, dst+=3, src+=3)
|
|
dst[0] = src[2], dst[1] = src[1], dst[2] = src[0];
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
ret = 0;
|
|
done:
|
|
png_destroy_read_struct(&png_ptr, info_ptr ? &info_ptr : NULL, (png_infopp)NULL);
|
|
fclose(fp);
|
|
return ret;
|
|
}
|
|
|
|
int writepng(const char *fname, unsigned short *src, int w, int h)
|
|
{
|
|
png_structp png_ptr = NULL;
|
|
png_infop info_ptr = NULL;
|
|
png_bytepp row_pointers;
|
|
int i, j, ret = -1;
|
|
FILE *f;
|
|
|
|
f = fopen(fname, "wb");
|
|
if (f == NULL) {
|
|
lprintf(__FILE__ ": failed to open \"%s\"\n", fname);
|
|
return -1;
|
|
}
|
|
|
|
row_pointers = calloc(h, sizeof(row_pointers[0]));
|
|
if (row_pointers == NULL)
|
|
goto end1;
|
|
|
|
for (i = 0; i < h; i++) {
|
|
unsigned char *dst = malloc(w * 3);
|
|
if (dst == NULL)
|
|
goto end2;
|
|
row_pointers[i] = dst;
|
|
for (j = 0; j < w; j++, src++, dst += 3) {
|
|
dst[0] = PXGETR(*src);
|
|
dst[1] = PXGETG(*src);
|
|
dst[2] = PXGETB(*src);
|
|
}
|
|
}
|
|
|
|
/* initialize stuff */
|
|
png_ptr = png_create_write_struct(PNG_LIBPNG_VER_STRING, NULL, NULL, NULL);
|
|
if (png_ptr == NULL) {
|
|
fprintf(stderr, "png_create_write_struct() failed");
|
|
goto end2;
|
|
}
|
|
|
|
info_ptr = png_create_info_struct(png_ptr);
|
|
if (info_ptr == NULL) {
|
|
fprintf(stderr, "png_create_info_struct() failed");
|
|
goto end3;
|
|
}
|
|
|
|
if (setjmp(png_jmpbuf(png_ptr)) != 0) {
|
|
fprintf(stderr, "error in png code\n");
|
|
goto end4;
|
|
}
|
|
|
|
png_init_io(png_ptr, f);
|
|
|
|
png_set_IHDR(png_ptr, info_ptr, w, h,
|
|
8, PNG_COLOR_TYPE_RGB, PNG_INTERLACE_NONE,
|
|
PNG_COMPRESSION_TYPE_DEFAULT, PNG_FILTER_TYPE_DEFAULT);
|
|
|
|
png_write_info(png_ptr, info_ptr);
|
|
png_write_image(png_ptr, row_pointers);
|
|
png_write_end(png_ptr, NULL);
|
|
|
|
ret = 0;
|
|
|
|
end4:
|
|
// png_destroy_info_struct(png_ptr, &info_ptr); // freed below
|
|
end3:
|
|
png_destroy_write_struct(&png_ptr, &info_ptr);
|
|
end2:
|
|
for (i = 0; i < h; i++)
|
|
free(row_pointers[i]);
|
|
free(row_pointers);
|
|
end1:
|
|
fclose(f);
|
|
return ret;
|
|
}
|
|
|