bernard.xiong 520c1cdc08 add HDC conversion tool
git-svn-id: https://rt-thread.googlecode.com/svn/trunk@154 bbd45198-f89e-11dd-88c7-29a3b14d5316
2009-11-06 00:09:19 +00:00

196 lines
4.5 KiB
C

#include <sys/types.h>
#include <stdio.h>
#include <stdlib.h>
#include "libpng/png.h"
typedef unsigned char uint8_t;
typedef unsigned short uint16_t;
typedef unsigned int uint32_t;
typedef unsigned int hdc_color_t;
typedef unsigned short hdc_pixel_t;
#define HDC_ARGB(a, r, g, b) \
((hdc_color_t)(((uint8_t)(r)|\
(((unsigned)(uint8_t)(g))<<8))|\
(((unsigned long)(uint8_t)(b))<<16)|\
(((unsigned long)(uint8_t)(a))<<24)))
#define HDC_RGB(r, g, b) HDC_ARGB(255, (r), (g), (b))
#define HDC_RGB_R(c) ((c) & 0xff)
#define HDC_RGB_G(c) (((c) >> 8) & 0xff)
#define HDC_RGB_B(c) (((c) >> 16) & 0xff)
#define HDC_RGB_A(c) (((c) >> 24) & 0xff)
/* convert HDC color to BBBBBGGGGGGRRRRR */
uint16_t hdc_color_to_565(hdc_color_t c)
{
uint16_t pixel;
pixel = ((HDC_RGB_B(c)>> 3) << 11) | ((HDC_RGB_G(c) >> 2) << 5) | (HDC_RGB_R(c) >> 3);
return pixel;
}
/* convert HDC color to RRRRRGGGGGGBBBBB */
uint16_t hdc_color_to_565p(hdc_color_t c)
{
uint16_t pixel;
pixel = ((HDC_RGB_R(c) >> 3) << 11) | ((HDC_RGB_G(c) >> 2) << 5) | (HDC_RGB_B(c)>> 3);
return pixel;
}
/* png image information */
png_structp png_ptr;
png_infop info_ptr;
struct hdc_image_header
{
uint32_t magic;
uint32_t width;
uint32_t height;
/* transparent color */
uint32_t tmask;
/* reserved */
uint32_t reserved;
};
struct hdc_image_header header;
int main(int argc, char** argv)
{
int bit_depth;
int color_type;
double gamma;
int width, height;
FILE *src_fp, *dst_fp;
png_uint_32 x, y;
png_bytep row;
png_bytep data;
hdc_pixel_t *ptr;
src_fp = fopen(argv[1], "rb");
dst_fp = fopen(argv[2], "wb");
if (src_fp == NULL || dst_fp == NULL)
{
printf("can't open file for read or write\n");
return -1;
}
png_ptr = png_create_read_struct(PNG_LIBPNG_VER_STRING, NULL, NULL, NULL);
if (png_ptr == NULL)
{
return -1;
}
info_ptr = png_create_info_struct(png_ptr);
if (info_ptr == NULL)
{
png_destroy_read_struct(&png_ptr, NULL, NULL);
return -1;
}
if (setjmp(png_jmpbuf(png_ptr)))
{
png_read_end(png_ptr, NULL);
/* destroy png struct */
png_destroy_info_struct(png_ptr, &info_ptr);
png_destroy_read_struct(&png_ptr, NULL, NULL);
fclose(src_fp);fclose(dst_fp);
return (-1);
}
png_init_io(png_ptr, src_fp);
png_read_info(png_ptr, info_ptr);
png_get_IHDR(png_ptr, info_ptr, &width, &height, &bit_depth,
&color_type, NULL, NULL, NULL);
if (bit_depth == 16)
png_set_strip_16(png_ptr);
if (color_type == PNG_COLOR_TYPE_PALETTE)
png_set_expand(png_ptr);
if (bit_depth < 8)
png_set_expand(png_ptr);
if (png_get_valid(png_ptr, info_ptr, PNG_INFO_tRNS))
png_set_expand(png_ptr);
if (color_type == PNG_COLOR_TYPE_GRAY ||
color_type == PNG_COLOR_TYPE_GRAY_ALPHA)
png_set_gray_to_rgb(png_ptr);
/* set gamma conversion */
if (png_get_gAMA(png_ptr, info_ptr, &gamma))
png_set_gamma(png_ptr, (double)2.2, gamma);
png_read_update_info(png_ptr, info_ptr);
/* write HDC file header */
sprintf((char *)&header.magic, "HDC%c", '\0');
header.width = info_ptr->width;
header.height = info_ptr->height;
header.tmask = 0;
header.reserved = 0;
fwrite(&header, 1, sizeof(struct hdc_image_header), dst_fp);
row = (png_bytep) malloc (png_get_rowbytes(png_ptr, info_ptr));
if (row == NULL) goto _return;
ptr = (hdc_pixel_t*) malloc (header.width * header.height * sizeof(hdc_pixel_t));
switch (info_ptr->color_type)
{
case PNG_COLOR_TYPE_RGBA:
for (y = 0; y < info_ptr->height; y++)
{
png_read_row(png_ptr, row, png_bytep_NULL);
for (x = 0; x < info_ptr->width; x++)
{
data = &(row[x * 4]);
ptr[x + y * info_ptr->width] = hdc_color_to_565p(HDC_ARGB((255 - data[3]), data[0], data[1], data[2]));
}
}
break;
case PNG_COLOR_TYPE_PALETTE:
for (y = 0; y < info_ptr->height; y++)
{
png_read_row(png_ptr, row, png_bytep_NULL);
for (x = 0; x < info_ptr->width; x++)
{
data = &(row[x]);
ptr[x] = hdc_color_to_565p(HDC_ARGB(0, info_ptr->palette[data[0]].red,
info_ptr->palette[data[0]].green,
info_ptr->palette[data[0]].blue));
}
}
default:
break;
};
fwrite(ptr, 1, header.height * header.width * sizeof(hdc_pixel_t), dst_fp);
_return:
png_read_end(png_ptr, NULL);
/* destroy png struct */
png_destroy_info_struct(png_ptr, &info_ptr);
png_destroy_read_struct(&png_ptr, NULL, NULL);
fclose(src_fp);
fclose(dst_fp);
free(ptr);
return 0;
}