You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
539 lines
16 KiB
539 lines
16 KiB
/*****************************************************************************
|
|
* png.c: png decoder module making use of libpng.
|
|
*****************************************************************************
|
|
* Copyright (C) 1999-2001 VLC authors and VideoLAN
|
|
*
|
|
* Authors: Gildas Bazin <gbazin@videolan.org>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
* under the terms of the GNU Lesser General Public License as published by
|
|
* the Free Software Foundation; either version 2.1 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public License
|
|
* along with this program; if not, write to the Free Software Foundation,
|
|
* Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
|
|
*****************************************************************************/
|
|
|
|
/*****************************************************************************
|
|
* Preamble
|
|
*****************************************************************************/
|
|
#ifdef HAVE_CONFIG_H
|
|
# include "config.h"
|
|
#endif
|
|
|
|
#include <vlc_common.h>
|
|
#include <vlc_plugin.h>
|
|
#include <vlc_codec.h>
|
|
#include <vlc_rand.h>
|
|
#include <vlc_ancillary.h>
|
|
#include <png.h>
|
|
|
|
/* PNG_SYS_COMMON_MEMBERS:
|
|
* members common to encoder and decoder descriptors
|
|
*/
|
|
#define PNG_SYS_COMMON_MEMBERS \
|
|
/**@{*/ \
|
|
bool b_error; \
|
|
vlc_object_t *p_obj; \
|
|
/**@}*/ \
|
|
|
|
/*
|
|
* png common descriptor
|
|
*/
|
|
struct png_sys_t
|
|
{
|
|
PNG_SYS_COMMON_MEMBERS
|
|
};
|
|
|
|
typedef struct png_sys_t png_sys_t;
|
|
|
|
/*****************************************************************************
|
|
* decoder_sys_t : png decoder descriptor
|
|
*****************************************************************************/
|
|
typedef struct
|
|
{
|
|
PNG_SYS_COMMON_MEMBERS
|
|
} decoder_sys_t;
|
|
|
|
/*****************************************************************************
|
|
* Local prototypes
|
|
*****************************************************************************/
|
|
static int OpenDecoder ( vlc_object_t * );
|
|
|
|
static int DecodeBlock ( decoder_t *, block_t * );
|
|
|
|
/*
|
|
* png encoder descriptor
|
|
*/
|
|
typedef struct
|
|
{
|
|
PNG_SYS_COMMON_MEMBERS
|
|
int i_blocksize;
|
|
} encoder_sys_t;
|
|
|
|
static int OpenEncoder(vlc_object_t *);
|
|
|
|
static block_t *EncodeBlock(encoder_t *, picture_t *);
|
|
|
|
/*****************************************************************************
|
|
* Module descriptor
|
|
*****************************************************************************/
|
|
vlc_module_begin ()
|
|
set_subcategory( SUBCAT_INPUT_VCODEC )
|
|
set_description( N_("PNG video decoder") )
|
|
set_capability( "video decoder", 1000 )
|
|
set_callback( OpenDecoder )
|
|
add_shortcut( "png" )
|
|
|
|
/* video encoder submodule */
|
|
add_submodule()
|
|
add_shortcut("png")
|
|
set_section(N_("Encoding"), NULL)
|
|
set_description(N_("PNG video encoder"))
|
|
set_capability("video encoder", 1000)
|
|
set_callback(OpenEncoder)
|
|
|
|
/* image encoder submodule */
|
|
add_submodule()
|
|
add_shortcut("png")
|
|
set_section(N_("Encoding"), NULL)
|
|
set_description(N_("PNG image encoder"))
|
|
set_capability("image encoder", 1000)
|
|
set_callback(OpenEncoder)
|
|
vlc_module_end ()
|
|
|
|
/*****************************************************************************
|
|
* OpenDecoder: probe the decoder and return score
|
|
*****************************************************************************/
|
|
static int OpenDecoder( vlc_object_t *p_this )
|
|
{
|
|
decoder_t *p_dec = (decoder_t*)p_this;
|
|
|
|
if( p_dec->fmt_in->i_codec != VLC_CODEC_PNG &&
|
|
p_dec->fmt_in->i_codec != VLC_FOURCC('M','P','N','G') )
|
|
{
|
|
return VLC_EGENERIC;
|
|
}
|
|
|
|
/* Allocate the memory needed to store the decoder's structure */
|
|
decoder_sys_t *p_sys = vlc_obj_malloc( p_this, sizeof(decoder_sys_t) );
|
|
if( p_sys == NULL )
|
|
return VLC_ENOMEM;
|
|
p_dec->p_sys = p_sys;
|
|
|
|
p_sys->p_obj = p_this;
|
|
|
|
/* Set output properties */
|
|
p_dec->fmt_out.i_codec = VLC_CODEC_RGBA;
|
|
p_dec->fmt_out.video.transfer = TRANSFER_FUNC_SRGB;
|
|
p_dec->fmt_out.video.space = COLOR_SPACE_SRGB;
|
|
p_dec->fmt_out.video.primaries = COLOR_PRIMARIES_SRGB;
|
|
p_dec->fmt_out.video.color_range = COLOR_RANGE_FULL;
|
|
|
|
/* Set callbacks */
|
|
p_dec->pf_decode = DecodeBlock;
|
|
|
|
return VLC_SUCCESS;
|
|
}
|
|
|
|
static void user_read( png_structp p_png, png_bytep data, png_size_t i_length )
|
|
{
|
|
block_t *p_block = (block_t *)png_get_io_ptr( p_png );
|
|
if( i_length > p_block->i_buffer ) {
|
|
png_error( p_png, "not enough data" );
|
|
return;
|
|
}
|
|
|
|
memcpy( data, p_block->p_buffer, i_length );
|
|
p_block->p_buffer += i_length;
|
|
p_block->i_buffer -= i_length;
|
|
}
|
|
|
|
static void user_flush( png_structp p_png )
|
|
{
|
|
/* noop */
|
|
(void) p_png;
|
|
}
|
|
|
|
static void user_write( png_structp p_png, png_bytep data, png_size_t i_length )
|
|
{
|
|
block_t *p_block = (block_t *)png_get_io_ptr( p_png );
|
|
if( i_length > p_block->i_buffer ) {
|
|
char err_str[64];
|
|
snprintf( err_str, sizeof(err_str),
|
|
"block size %zu too small for %zu encoded bytes",
|
|
p_block->i_buffer, i_length );
|
|
png_error( p_png, err_str );
|
|
return;
|
|
}
|
|
|
|
memcpy( p_block->p_buffer, data, i_length );
|
|
p_block->p_buffer += i_length;
|
|
p_block->i_buffer -= i_length;
|
|
}
|
|
|
|
static void user_error( png_structp p_png, png_const_charp error_msg )
|
|
{
|
|
png_sys_t *p_sys = (png_sys_t *)png_get_error_ptr( p_png );
|
|
p_sys->b_error = true;
|
|
msg_Err( p_sys->p_obj, "%s", error_msg );
|
|
}
|
|
|
|
static void user_warning( png_structp p_png, png_const_charp warning_msg )
|
|
{
|
|
png_sys_t *p_sys = (png_sys_t *)png_get_error_ptr( p_png );
|
|
msg_Warn( p_sys->p_obj, "%s", warning_msg );
|
|
}
|
|
|
|
#ifdef PNG_TEXT_SUPPORTED
|
|
static void process_text_chunk( decoder_t *p_dec, const png_textp chunk )
|
|
{
|
|
if( chunk->compression != PNG_ITXT_COMPRESSION_NONE ||
|
|
memcmp( chunk->key, "XML:com.adobe.xmp", 17 ) ||
|
|
chunk->itxt_length < 20 )
|
|
return;
|
|
|
|
const char *exifxmp = (const char *) chunk->text;
|
|
const char *orient = strnstr( exifxmp, ":Orientation>", chunk->itxt_length );
|
|
if(orient && orient - exifxmp > 14)
|
|
p_dec->fmt_out.video.orientation = ORIENT_FROM_EXIF( orient[13] - '0' );
|
|
}
|
|
|
|
static int make_xmp_packet( const video_format_t *fmt, png_textp chunk )
|
|
{
|
|
unsigned char id[9];
|
|
vlc_rand_bytes(id, 8);
|
|
for(int i=0; i<8; i++)
|
|
id[i] = (id[i] % 26) + 'A';
|
|
id[8] = '\0';
|
|
int len = asprintf( &chunk->text,
|
|
"<?xpacket begin='' id='%s'?>"
|
|
"<x:xmpmeta xmlns:x='adobe:ns:meta/' x:xmptk='VLC " VERSION "'>"
|
|
"<rdf:RDF xmlns:rdf='http://www.w3.org/1999/02/22-rdf-syntax-ns#'>"
|
|
"<rdf:Description rdf:about='' xmlns:tiff='http://ns.adobe.com/tiff/1.0/'>"
|
|
"<tiff:Orientation>%" PRIu8 "</tiff:Orientation>"
|
|
"</rdf:Description>"
|
|
"</rdf:RDF>"
|
|
"</x:xmpmeta>"
|
|
"<?xpacket end='r'?>", id, ORIENT_TO_EXIF(fmt->orientation) );
|
|
if(len == 0)
|
|
{
|
|
free(chunk->text);
|
|
chunk->text = NULL;
|
|
}
|
|
chunk->itxt_length = (len <= 0) ? 0 : len;
|
|
chunk->compression = PNG_ITXT_COMPRESSION_NONE;
|
|
chunk->key = len > 0 ? strdup( "XML:com.adobe.xmp" ) : NULL;
|
|
chunk->lang_key = NULL;
|
|
chunk->lang = NULL;
|
|
chunk->text_length = 0;
|
|
return len > 0 ? VLC_SUCCESS : VLC_EGENERIC;
|
|
}
|
|
|
|
#endif
|
|
|
|
/****************************************************************************
|
|
* DecodeBlock: the whole thing
|
|
****************************************************************************
|
|
* This function must be fed with a complete compressed frame.
|
|
****************************************************************************/
|
|
static int DecodeBlock( decoder_t *p_dec, block_t *p_block )
|
|
{
|
|
decoder_sys_t *p_sys = p_dec->p_sys;
|
|
picture_t *p_pic = 0;
|
|
|
|
png_uint_32 i_width, i_height;
|
|
int i_color_type, i_interlace_type, i_compression_type, i_filter_type;
|
|
int i_bit_depth, i;
|
|
|
|
png_structp p_png;
|
|
png_infop p_info, p_end_info;
|
|
png_bytep *volatile p_row_pointers = NULL;
|
|
|
|
if( !p_block ) /* No Drain */
|
|
return VLCDEC_SUCCESS;
|
|
|
|
p_sys->b_error = false;
|
|
|
|
if( p_block->i_flags & BLOCK_FLAG_CORRUPTED )
|
|
{
|
|
block_Release( p_block );
|
|
return VLCDEC_SUCCESS;
|
|
}
|
|
|
|
p_png = png_create_read_struct( PNG_LIBPNG_VER_STRING, 0, 0, 0 );
|
|
if( p_png == NULL )
|
|
{
|
|
block_Release( p_block );
|
|
return VLCDEC_SUCCESS;
|
|
}
|
|
|
|
p_info = png_create_info_struct( p_png );
|
|
if( p_info == NULL )
|
|
{
|
|
png_destroy_read_struct( &p_png, NULL, NULL );
|
|
block_Release( p_block );
|
|
return VLCDEC_SUCCESS;
|
|
}
|
|
|
|
p_end_info = png_create_info_struct( p_png );
|
|
if( p_end_info == NULL )
|
|
{
|
|
png_destroy_read_struct( &p_png, &p_info, NULL );
|
|
block_Release( p_block );
|
|
return VLCDEC_SUCCESS;
|
|
}
|
|
|
|
/* libpng longjmp's there in case of error */
|
|
if( setjmp( png_jmpbuf( p_png ) ) )
|
|
goto error;
|
|
|
|
png_set_read_fn( p_png, p_block, user_read );
|
|
png_set_error_fn( p_png, p_sys, user_error, user_warning );
|
|
|
|
png_read_info( p_png, p_info );
|
|
if( p_sys->b_error ) goto error;
|
|
|
|
png_get_IHDR( p_png, p_info, &i_width, &i_height,
|
|
&i_bit_depth, &i_color_type, &i_interlace_type,
|
|
&i_compression_type, &i_filter_type);
|
|
if( p_sys->b_error ) goto error;
|
|
|
|
/* Set output properties */
|
|
p_dec->fmt_out.i_codec = VLC_CODEC_RGBA;
|
|
p_dec->fmt_out.video.i_visible_width = p_dec->fmt_out.video.i_width = i_width;
|
|
p_dec->fmt_out.video.i_visible_height = p_dec->fmt_out.video.i_height = i_height;
|
|
p_dec->fmt_out.video.i_sar_num = 1;
|
|
p_dec->fmt_out.video.i_sar_den = 1;
|
|
|
|
#ifdef PNG_TEXT_SUPPORTED
|
|
png_textp textp;
|
|
int numtextp;
|
|
if( png_get_text( p_png, p_info, &textp, &numtextp ) > 0 )
|
|
for( int ii=0; ii<numtextp; ii++ )
|
|
process_text_chunk( p_dec, &textp[ii] );
|
|
#endif
|
|
|
|
if( i_color_type == PNG_COLOR_TYPE_PALETTE )
|
|
png_set_palette_to_rgb( p_png );
|
|
|
|
if( i_color_type == PNG_COLOR_TYPE_GRAY ||
|
|
i_color_type == PNG_COLOR_TYPE_GRAY_ALPHA )
|
|
png_set_gray_to_rgb( p_png );
|
|
if( i_color_type & PNG_COLOR_MASK_ALPHA )
|
|
png_set_alpha_mode( p_png, PNG_ALPHA_OPTIMIZED, PNG_DEFAULT_sRGB );
|
|
|
|
/* Strip to 8 bits per channel */
|
|
if( i_bit_depth == 16 )
|
|
{
|
|
#if PNG_LIBPNG_VER >= 10504
|
|
png_set_scale_16( p_png );
|
|
#else
|
|
png_set_strip_16( p_png );
|
|
#endif
|
|
}
|
|
|
|
if( png_get_valid( p_png, p_info, PNG_INFO_tRNS ) )
|
|
{
|
|
png_set_tRNS_to_alpha( p_png );
|
|
}
|
|
else if( !(i_color_type & PNG_COLOR_MASK_ALPHA) )
|
|
{
|
|
p_dec->fmt_out.i_codec = VLC_CODEC_RGB24;
|
|
}
|
|
|
|
/* Get a new picture */
|
|
if( decoder_UpdateVideoFormat( p_dec ) )
|
|
goto error;
|
|
p_pic = decoder_NewPicture( p_dec );
|
|
if( !p_pic ) goto error;
|
|
|
|
/* Decode ICC profile */
|
|
#ifdef PNG_iCCP_SUPPORTED
|
|
if (png_get_valid( p_png, p_info, PNG_INFO_iCCP ))
|
|
{
|
|
vlc_icc_profile_t *icc;
|
|
png_charp name;
|
|
int compression;
|
|
# if PNG_LIBPNG_VER < 10500
|
|
png_charp iccdata;
|
|
# else
|
|
png_bytep iccdata;
|
|
# endif
|
|
png_uint_32 icclen;
|
|
png_get_iCCP( p_png, p_info, &name, &compression, &iccdata, &icclen);
|
|
if( compression != PNG_COMPRESSION_TYPE_BASE )
|
|
goto error; /* impossible with current libpng */
|
|
icc = picture_AttachNewAncillary( p_pic, VLC_ANCILLARY_ID_ICC, sizeof(*icc) + icclen );
|
|
if ( !icc )
|
|
goto error;
|
|
memcpy( icc->data, iccdata, icclen );
|
|
icc->size = icclen;
|
|
}
|
|
#endif
|
|
|
|
|
|
/* Decode picture */
|
|
p_row_pointers = vlc_alloc( i_height, sizeof(png_bytep) );
|
|
if( !p_row_pointers )
|
|
goto error;
|
|
for( i = 0; i < (int)i_height; i++ )
|
|
p_row_pointers[i] = p_pic->p->p_pixels + p_pic->p->i_pitch * i;
|
|
|
|
png_read_image( p_png, p_row_pointers );
|
|
if( p_sys->b_error ) goto error;
|
|
png_read_end( p_png, p_end_info );
|
|
if( p_sys->b_error ) goto error;
|
|
|
|
png_destroy_read_struct( &p_png, &p_info, &p_end_info );
|
|
free( p_row_pointers );
|
|
|
|
p_pic->date = p_block->i_pts != VLC_TICK_INVALID ? p_block->i_pts : p_block->i_dts;
|
|
|
|
block_Release( p_block );
|
|
decoder_QueueVideo( p_dec, p_pic );
|
|
return VLCDEC_SUCCESS;
|
|
|
|
error:
|
|
|
|
if( p_pic )
|
|
picture_Release( p_pic );
|
|
free( p_row_pointers );
|
|
png_destroy_read_struct( &p_png, &p_info, &p_end_info );
|
|
block_Release( p_block );
|
|
return VLCDEC_SUCCESS;
|
|
}
|
|
|
|
static int OpenEncoder(vlc_object_t *p_this)
|
|
{
|
|
encoder_t *p_enc = (encoder_t *) p_this;
|
|
|
|
if( p_enc->fmt_out.i_codec != VLC_CODEC_PNG )
|
|
return VLC_EGENERIC;
|
|
|
|
/* Allocate the memory needed to store the encoder's structure */
|
|
encoder_sys_t *p_sys = vlc_obj_malloc(p_this, sizeof(encoder_sys_t));
|
|
if( p_sys == NULL )
|
|
return VLC_ENOMEM;
|
|
p_enc->p_sys = p_sys;
|
|
|
|
p_sys->p_obj = p_this;
|
|
|
|
p_sys->i_blocksize = 3 * p_enc->fmt_in.video.i_visible_width *
|
|
p_enc->fmt_in.video.i_visible_height;
|
|
|
|
p_enc->fmt_in.i_codec =
|
|
p_enc->fmt_in.video.i_chroma = VLC_CODEC_RGB24;
|
|
|
|
static const struct vlc_encoder_operations ops =
|
|
{ .encode_video = EncodeBlock };
|
|
|
|
p_enc->ops = &ops;
|
|
|
|
return VLC_SUCCESS;
|
|
}
|
|
|
|
/*
|
|
* EncodeBlock
|
|
*/
|
|
static block_t *EncodeBlock(encoder_t *p_enc, picture_t *p_pic)
|
|
{
|
|
encoder_sys_t *p_sys = p_enc->p_sys;
|
|
|
|
if( unlikely( !p_pic ) )
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
block_t *p_block = block_Alloc( p_sys->i_blocksize );
|
|
if( p_block == NULL )
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
png_structp p_png = png_create_write_struct( PNG_LIBPNG_VER_STRING, 0, 0, 0 );
|
|
if( p_png == NULL )
|
|
{
|
|
block_Release( p_block );
|
|
return NULL;
|
|
}
|
|
|
|
/* Disable filtering to speed-up encoding */
|
|
png_set_filter( p_png, 0, PNG_NO_FILTERS );
|
|
/* 1 == best speed */
|
|
png_set_compression_level( p_png, 1 );
|
|
|
|
/* save buffer start */
|
|
uint8_t *p_start = p_block->p_buffer;
|
|
size_t i_start = p_block->i_buffer;
|
|
|
|
p_sys->b_error = false;
|
|
png_infop p_info = NULL;
|
|
|
|
/* libpng longjmp's there in case of error */
|
|
if( setjmp( png_jmpbuf( p_png ) ) )
|
|
goto error;
|
|
|
|
png_set_write_fn( p_png, p_block, user_write, user_flush );
|
|
png_set_error_fn( p_png, p_sys, user_error, user_warning );
|
|
|
|
p_info = png_create_info_struct( p_png );
|
|
if( p_info == NULL )
|
|
goto error;
|
|
|
|
png_set_IHDR( p_png, p_info,
|
|
p_enc->fmt_in.video.i_visible_width,
|
|
p_enc->fmt_in.video.i_visible_height,
|
|
8, PNG_COLOR_TYPE_RGB, PNG_INTERLACE_NONE,
|
|
PNG_COMPRESSION_TYPE_DEFAULT, PNG_FILTER_TYPE_DEFAULT );
|
|
if( p_sys->b_error ) goto error;
|
|
|
|
#ifdef PNG_TEXT_SUPPORTED
|
|
png_text text;
|
|
if( make_xmp_packet( &p_pic->format, &text ) == VLC_SUCCESS )
|
|
{
|
|
png_set_text( p_png, p_info, &text, 1 );
|
|
png_write_info( p_png, p_info );
|
|
free( text.key );
|
|
free( text.text );
|
|
}
|
|
else
|
|
#endif
|
|
png_write_info( p_png, p_info );
|
|
|
|
if( p_sys->b_error ) goto error;
|
|
|
|
/* Encode picture */
|
|
|
|
for( int i = 0; i < p_pic->p->i_visible_lines; i++ )
|
|
{
|
|
png_write_row( p_png, p_pic->p->p_pixels + (i * p_pic->p->i_pitch) );
|
|
if( p_sys->b_error ) goto error;
|
|
}
|
|
|
|
png_write_end( p_png, p_info );
|
|
if( p_sys->b_error ) goto error;
|
|
|
|
png_destroy_write_struct( &p_png, &p_info );
|
|
|
|
/* restore original buffer position */
|
|
p_block->p_buffer = p_start;
|
|
p_block->i_buffer = i_start - p_block->i_buffer;
|
|
|
|
p_block->i_dts = p_block->i_pts = p_pic->date;
|
|
|
|
return p_block;
|
|
|
|
error:
|
|
|
|
png_destroy_write_struct( &p_png, p_info ? &p_info : NULL );
|
|
|
|
block_Release(p_block);
|
|
return NULL;
|
|
}
|
|
|