2006-04-01 10:38:54 +00:00
|
|
|
// Copyright (c) 2006, Rodrigo Braz Monteiro
|
|
|
|
// All rights reserved.
|
|
|
|
//
|
|
|
|
// Redistribution and use in source and binary forms, with or without
|
|
|
|
// modification, are permitted provided that the following conditions are met:
|
|
|
|
//
|
|
|
|
// * Redistributions of source code must retain the above copyright notice,
|
|
|
|
// this list of conditions and the following disclaimer.
|
|
|
|
// * Redistributions in binary form must reproduce the above copyright notice,
|
|
|
|
// this list of conditions and the following disclaimer in the documentation
|
|
|
|
// and/or other materials provided with the distribution.
|
|
|
|
// * Neither the name of the Aegisub Group nor the names of its contributors
|
|
|
|
// may be used to endorse or promote products derived from this software
|
|
|
|
// without specific prior written permission.
|
|
|
|
//
|
|
|
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
|
|
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
// ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
|
|
|
|
// LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
|
|
// CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
|
|
// SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
|
|
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
|
|
// CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
|
|
// ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
|
|
// POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
//
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
// AEGISUB
|
|
|
|
//
|
|
|
|
// Website: http://aegisub.cellosoft.com
|
|
|
|
// Contact: mailto:zeratul@cellosoft.com
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
|
|
///////////
|
|
|
|
// Headers
|
|
|
|
#include <png.h>
|
|
|
|
#include "png_wrap.h"
|
2006-04-01 11:00:55 +00:00
|
|
|
#include "prs_video_frame.h"
|
2006-04-01 10:38:54 +00:00
|
|
|
|
|
|
|
|
|
|
|
///////////////
|
|
|
|
// Constructor
|
|
|
|
PNGWrapper::PNGWrapper() {
|
|
|
|
initialized = false;
|
|
|
|
pos = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//////////////
|
|
|
|
// Destructor
|
|
|
|
PNGWrapper::~PNGWrapper() {
|
|
|
|
if (initialized) End();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//////////////
|
|
|
|
// Read image
|
2006-04-01 11:00:55 +00:00
|
|
|
void PNGWrapper::Read(PRSVideoFrame *frame) {
|
|
|
|
// Begin
|
|
|
|
Begin();
|
2006-04-01 10:38:54 +00:00
|
|
|
|
|
|
|
// Initialize libpng structures
|
|
|
|
png_structp png_ptr = png_create_read_struct (PNG_LIBPNG_VER_STRING, NULL, NULL, NULL);
|
|
|
|
if (!png_ptr) throw 1;
|
|
|
|
png_infop info_ptr = png_create_info_struct(png_ptr);
|
|
|
|
if (!info_ptr) {
|
|
|
|
png_destroy_read_struct(&png_ptr, (png_infopp)NULL, (png_infopp)NULL);
|
|
|
|
throw 1;
|
|
|
|
}
|
|
|
|
png_infop end_info = png_create_info_struct(png_ptr);
|
|
|
|
if (!end_info) {
|
|
|
|
png_destroy_read_struct(&png_ptr, &info_ptr, (png_infopp)NULL);
|
|
|
|
throw 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set jump for error handling (man, I hate this lib)
|
|
|
|
if (setjmp(png_jmpbuf(png_ptr))) {
|
|
|
|
png_destroy_read_struct(&png_ptr, &info_ptr, &end_info);
|
|
|
|
throw 1;
|
|
|
|
}
|
|
|
|
|
2006-04-01 11:00:55 +00:00
|
|
|
// Set data reading
|
2006-04-03 13:45:47 +00:00
|
|
|
png_set_read_fn(png_ptr,this,(png_rw_ptr) memory_read_data);
|
2006-04-01 11:00:55 +00:00
|
|
|
|
|
|
|
// Read data
|
2006-04-01 14:34:03 +00:00
|
|
|
//png_read_png(png_ptr, info_ptr, PNG_TRANSFORM_EXPAND | PNG_TRANSFORM_PACKING | PNG_TRANSFORM_STRIP_16, NULL);
|
2006-04-01 11:45:38 +00:00
|
|
|
|
|
|
|
// Get image size
|
2006-04-01 14:34:03 +00:00
|
|
|
//int w = png_get_image_width(png_ptr,info_ptr);
|
|
|
|
//int h = png_get_image_height(png_ptr,info_ptr);
|
|
|
|
|
|
|
|
// Read info
|
|
|
|
png_uint_32 w, h;
|
|
|
|
int bit_depth, color_type, interlace_type, compression_type, filter_method;
|
|
|
|
png_read_info(png_ptr, info_ptr);
|
|
|
|
png_get_IHDR(png_ptr, info_ptr, &w, &h, &bit_depth, &color_type, &interlace_type, &compression_type, &filter_method);
|
|
|
|
|
|
|
|
// Transformations
|
|
|
|
color_type = png_get_color_type(png_ptr, info_ptr);
|
|
|
|
if (color_type == PNG_COLOR_TYPE_PALETTE) png_set_palette_to_rgb(png_ptr);
|
|
|
|
if (color_type == PNG_COLOR_TYPE_GRAY && bit_depth < 8) png_set_gray_1_2_4_to_8(png_ptr);
|
2006-04-01 14:40:17 +00:00
|
|
|
else if (color_type == PNG_COLOR_TYPE_GRAY || color_type == PNG_COLOR_TYPE_GRAY_ALPHA) png_set_gray_to_rgb(png_ptr);
|
2006-04-01 14:34:03 +00:00
|
|
|
if (png_get_valid(png_ptr, info_ptr, PNG_INFO_tRNS)) png_set_tRNS_to_alpha(png_ptr);
|
|
|
|
if (bit_depth > 8) png_set_strip_16(png_ptr);
|
2006-04-01 14:40:17 +00:00
|
|
|
if (bit_depth < 8) png_set_packing(png_ptr);
|
|
|
|
if (color_type == PNG_COLOR_TYPE_RGB) png_set_filler(png_ptr, 0xFFFF, PNG_FILLER_AFTER);
|
2006-04-01 14:34:03 +00:00
|
|
|
png_read_update_info(png_ptr, info_ptr);
|
|
|
|
int rowLen = png_get_rowbytes(png_ptr,info_ptr);
|
2006-04-01 11:45:38 +00:00
|
|
|
|
|
|
|
// Allocate frame data
|
|
|
|
int bpp = 4;
|
|
|
|
frame->ownData = true;
|
2006-04-01 14:34:03 +00:00
|
|
|
frame->data[0] = new char[h*rowLen];
|
|
|
|
frame->w = rowLen/4;
|
2006-04-01 11:45:38 +00:00
|
|
|
frame->h = h;
|
2006-04-01 14:34:03 +00:00
|
|
|
frame->pitch = rowLen;
|
2006-04-01 11:45:38 +00:00
|
|
|
frame->colorSpace = ColorSpace_RGB32;
|
|
|
|
|
2006-04-01 14:34:03 +00:00
|
|
|
// Get rows
|
|
|
|
png_bytepp row_pointers = (png_bytepp) png_malloc (png_ptr,sizeof(png_bytep) * h);
|
2006-04-01 11:45:38 +00:00
|
|
|
char *dst = frame->data[0];
|
2006-04-01 14:34:03 +00:00
|
|
|
for (size_t i=0;i<h;i++) row_pointers[i] = (png_bytep) (dst+rowLen*i);
|
|
|
|
png_read_image(png_ptr, row_pointers);
|
|
|
|
png_read_end(png_ptr, end_info);
|
2006-04-01 11:00:55 +00:00
|
|
|
|
|
|
|
// Clean up
|
|
|
|
png_destroy_read_struct(&png_ptr, &info_ptr, &end_info);
|
|
|
|
End();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//////////////
|
|
|
|
// Initialize
|
|
|
|
void PNGWrapper::Begin() {
|
|
|
|
// Check initialization
|
|
|
|
if (initialized) End();
|
|
|
|
initialized = true;
|
2006-04-01 10:38:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
////////////
|
|
|
|
// Clean up
|
|
|
|
void PNGWrapper::End() {
|
|
|
|
// Check initialization
|
|
|
|
if (!initialized) return;
|
|
|
|
initialized = false;
|
|
|
|
}
|
2006-04-01 11:00:55 +00:00
|
|
|
|
|
|
|
|
|
|
|
/////////////
|
|
|
|
// Read data
|
|
|
|
void PNGWrapper::memory_read_data(png_structp png_ptr, png_bytep dstData, png_size_t length) {
|
|
|
|
PNGWrapper *wrapper = (PNGWrapper*) png_get_io_ptr(png_ptr);
|
|
|
|
wrapper->ReadData(dstData,length);
|
|
|
|
}
|
|
|
|
|
|
|
|
void PNGWrapper::ReadData(png_bytep dstData, png_size_t length) {
|
|
|
|
memcpy(dstData,((char*)data)+pos,length);
|
2006-04-01 11:45:38 +00:00
|
|
|
pos += (int) length;
|
2006-04-01 11:00:55 +00:00
|
|
|
}
|