libbpg/bpgdec.c

355 lines
9.3 KiB
C

/*
* BPG decoder command line utility
*
* Copyright (c) 2014 Fabrice Bellard
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/
#include <stdlib.h>
#include <stdio.h>
#include <math.h>
#include <getopt.h>
#include <inttypes.h>
/* define it to include PNG output */
#define USE_PNG
#ifdef USE_PNG
#include <png.h>
#endif
#include "libbpg.h"
static void ppm_save(BPGDecoderContext *img, const char *filename)
{
BPGImageInfo img_info_s, *img_info = &img_info_s;
FILE *f;
int w, h, y;
uint8_t *rgb_line;
bpg_decoder_get_info(img, img_info);
w = img_info->width;
h = img_info->height;
rgb_line = malloc(3 * w);
f = fopen(filename,"wb");
if (!f) {
fprintf(stderr, "%s: I/O error\n", filename);
exit(1);
}
fprintf(f, "P6\n%d %d\n%d\n", w, h, 255);
bpg_decoder_start(img, BPG_OUTPUT_FORMAT_RGB24);
for (y = 0; y < h; y++) {
bpg_decoder_get_line(img, rgb_line);
fwrite(rgb_line, 1, w * 3, f);
}
fclose(f);
free(rgb_line);
}
#ifdef USE_PNG
static void png_write_data (png_structp png_ptr, png_bytep data,
png_size_t length)
{
FILE *f;
int ret;
f = png_get_io_ptr(png_ptr);
ret = fwrite(data, 1, length, f);
if (ret != length)
png_error(png_ptr, "PNG Write Error");
}
static void png_save(BPGDecoderContext *img, const char *filename, int bit_depth)
{
BPGImageInfo img_info_s, *img_info = &img_info_s;
FILE *f;
png_structp png_ptr;
png_infop info_ptr;
png_bytep row_pointer;
int y, color_type, bpp;
BPGDecoderOutputFormat out_fmt;
if (bit_depth != 8 && bit_depth != 16) {
fprintf(stderr, "Only bit_depth = 8 or 16 are supported for PNG output\n");
exit(1);
}
bpg_decoder_get_info(img, img_info);
f = fopen(filename, "wb");
if (!f) {
fprintf(stderr, "%s: I/O error\n", filename);
exit(1);
}
png_ptr = png_create_write_struct_2(PNG_LIBPNG_VER_STRING,
NULL,
NULL, /* error */
NULL, /* warning */
NULL,
NULL,
NULL);
info_ptr = png_create_info_struct(png_ptr);
png_set_write_fn(png_ptr, (png_voidp)f, &png_write_data, NULL);
if (setjmp(png_jmpbuf(png_ptr)) != 0) {
fprintf(stderr, "PNG write error\n");
exit(1);
}
if (img_info->has_alpha)
color_type = PNG_COLOR_TYPE_RGB_ALPHA;
else
color_type = PNG_COLOR_TYPE_RGB;
png_set_IHDR(png_ptr, info_ptr, img_info->width, img_info->height,
bit_depth, color_type, PNG_INTERLACE_NONE,
PNG_COMPRESSION_TYPE_DEFAULT, PNG_FILTER_TYPE_DEFAULT);
png_write_info(png_ptr, info_ptr);
#if __BYTE_ORDER__ != __ORDER_BIG_ENDIAN__
if (bit_depth == 16) {
png_set_swap(png_ptr);
}
#endif
if (bit_depth == 16) {
if (img_info->has_alpha)
out_fmt = BPG_OUTPUT_FORMAT_RGBA64;
else
out_fmt = BPG_OUTPUT_FORMAT_RGB48;
} else {
if (img_info->has_alpha)
out_fmt = BPG_OUTPUT_FORMAT_RGBA32;
else
out_fmt = BPG_OUTPUT_FORMAT_RGB24;
}
bpg_decoder_start(img, out_fmt);
bpp = (3 + img_info->has_alpha) * (bit_depth / 8);
row_pointer = (png_bytep)png_malloc(png_ptr, img_info->width * bpp);
for (y = 0; y < img_info->height; y++) {
bpg_decoder_get_line(img, row_pointer);
png_write_row(png_ptr, row_pointer);
}
png_free(png_ptr, row_pointer);
png_write_end(png_ptr, NULL);
png_destroy_write_struct(&png_ptr, &info_ptr);
fclose(f);
}
#endif /* USE_PNG */
static void bpg_show_info(const char *filename, int show_extensions)
{
uint8_t *buf;
int buf_len, ret, buf_len_max;
FILE *f;
BPGImageInfo p_s, *p = &p_s;
BPGExtensionData *first_md, *md;
static const char *format_str[6] = {
"Gray",
"4:2:0",
"4:2:2",
"4:4:4",
"4:2:0_video",
"4:2:2_video",
};
static const char *color_space_str[BPG_CS_COUNT] = {
"YCbCr",
"RGB",
"YCgCo",
"YCbCr_BT709",
"YCbCr_BT2020",
};
static const char *extension_tag_str[] = {
"Unknown",
"EXIF",
"ICC profile",
"XMP",
"Thumbnail",
"Animation control",
};
f = fopen(filename, "rb");
if (!f) {
fprintf(stderr, "Could not open %s\n", filename);
exit(1);
}
if (show_extensions) {
fseek(f, 0, SEEK_END);
buf_len_max = ftell(f);
fseek(f, 0, SEEK_SET);
} else {
/* if no extension are shown, just need the header */
buf_len_max = BPG_DECODER_INFO_BUF_SIZE;
}
buf = malloc(buf_len_max);
buf_len = fread(buf, 1, buf_len_max, f);
ret = bpg_decoder_get_info_from_buf(p, show_extensions ? &first_md : NULL,
buf, buf_len);
free(buf);
fclose(f);
if (ret < 0) {
fprintf(stderr, "Not a BPG image\n");
exit(1);
}
printf("size=%dx%d color_space=%s",
p->width, p->height,
p->format == BPG_FORMAT_GRAY ? "Gray" : color_space_str[p->color_space]);
if (p->has_w_plane) {
printf(" w_plane=%d", p->has_w_plane);
}
if (p->has_alpha) {
printf(" alpha=%d premul=%d",
p->has_alpha, p->premultiplied_alpha);
}
printf(" format=%s limited_range=%d bit_depth=%d animation=%d\n",
format_str[p->format],
p->limited_range,
p->bit_depth,
p->has_animation);
if (first_md) {
const char *tag_name;
printf("Extension data:\n");
for(md = first_md; md != NULL; md = md->next) {
if (md->tag <= 5)
tag_name = extension_tag_str[md->tag];
else
tag_name = extension_tag_str[0];
printf(" tag=%d (%s) length=%d\n",
md->tag, tag_name, md->buf_len);
}
bpg_decoder_free_extension_data(first_md);
}
}
static void help(void)
{
printf("BPG Image Decoder version " CONFIG_BPG_VERSION "\n"
"usage: bpgdec [options] infile\n"
"Options:\n"
"-o outfile.[ppm|png] set the output filename (default = out.png)\n"
"-b bit_depth PNG output only: use bit_depth per component (8 or 16, default = 8)\n"
"-i display information about the image\n");
exit(1);
}
int main(int argc, char **argv)
{
FILE *f;
BPGDecoderContext *img;
uint8_t *buf;
int buf_len, bit_depth, c, show_info;
const char *outfilename, *filename, *p;
outfilename = "out.png";
bit_depth = 8;
show_info = 0;
for(;;) {
c = getopt(argc, argv, "ho:b:i");
if (c == -1)
break;
switch(c) {
case 'h':
show_help:
help();
break;
case 'o':
outfilename = optarg;
break;
case 'b':
bit_depth = atoi(optarg);
break;
case 'i':
show_info = 1;
break;
default:
exit(1);
}
}
if (optind >= argc)
goto show_help;
filename = argv[optind++];
if (show_info) {
bpg_show_info(filename, 1);
return 0;
}
f = fopen(filename, "rb");
if (!f) {
fprintf(stderr, "Could not open %s\n", filename);
exit(1);
}
fseek(f, 0, SEEK_END);
buf_len = ftell(f);
fseek(f, 0, SEEK_SET);
buf = malloc(buf_len);
if (fread(buf, 1, buf_len, f) != buf_len) {
fprintf(stderr, "Error while reading file\n");
exit(1);
}
fclose(f);
img = bpg_decoder_open();
if (bpg_decoder_decode(img, buf, buf_len) < 0) {
fprintf(stderr, "Could not decode image\n");
exit(1);
}
free(buf);
#ifdef USE_PNG
p = strrchr(outfilename, '.');
if (p)
p++;
if (p && strcasecmp(p, "ppm") != 0) {
png_save(img, outfilename, bit_depth);
} else
#endif
{
ppm_save(img, outfilename);
}
bpg_decoder_close(img);
return 0;
}