@ -435,3 +435,170 @@ int av_image_copy_to_buffer(uint8_t *dst, int dst_size,
return size ;
return size ;
}
}
// Fill dst[0..dst_size] with the bytes in clear[0..clear_size]. The clear
// bytes are repeated until dst_size is reached. If dst_size is unaligned (i.e.
// dst_size%clear_size!=0), the remaining data will be filled with the beginning
// of the clear data only.
static void memset_bytes ( uint8_t * dst , size_t dst_size , uint8_t * clear ,
size_t clear_size )
{
size_t pos = 0 ;
int same = 1 ;
int i ;
if ( ! clear_size )
return ;
// Reduce to memset() if possible.
for ( i = 0 ; i < clear_size ; i + + ) {
if ( clear [ i ] ! = clear [ 0 ] ) {
same = 0 ;
break ;
}
}
if ( same )
clear_size = 1 ;
if ( clear_size = = 1 ) {
memset ( dst , clear [ 0 ] , dst_size ) ;
dst_size = 0 ;
} else if ( clear_size = = 2 ) {
uint16_t val = AV_RN16 ( clear ) ;
for ( ; dst_size > = 2 ; dst_size - = 2 ) {
AV_WN16 ( dst , val ) ;
dst + = 2 ;
}
} else if ( clear_size = = 4 ) {
uint32_t val = AV_RN32 ( clear ) ;
for ( ; dst_size > = 4 ; dst_size - = 4 ) {
AV_WN32 ( dst , val ) ;
dst + = 4 ;
}
} else if ( clear_size = = 8 ) {
uint32_t val = AV_RN64 ( clear ) ;
for ( ; dst_size > = 8 ; dst_size - = 8 ) {
AV_WN64 ( dst , val ) ;
dst + = 8 ;
}
}
for ( ; dst_size ; dst_size - - )
* dst + + = clear [ pos + + % clear_size ] ;
}
// Maximum size in bytes of a plane element (usually a pixel, or multiple pixels
// if it's a subsampled packed format).
# define MAX_BLOCK_SIZE 32
int av_image_fill_black ( uint8_t * dst_data [ 4 ] , const ptrdiff_t dst_linesize [ 4 ] ,
enum AVPixelFormat pix_fmt , enum AVColorRange range ,
int width , int height )
{
const AVPixFmtDescriptor * desc = av_pix_fmt_desc_get ( pix_fmt ) ;
int nb_planes = av_pix_fmt_count_planes ( pix_fmt ) ;
// A pixel or a group of pixels on each plane, with a value that represents black.
// Consider e.g. AV_PIX_FMT_UYVY422 for non-trivial cases.
uint8_t clear_block [ 4 ] [ MAX_BLOCK_SIZE ] = { 0 } ; // clear padding with 0
int clear_block_size [ 4 ] = { 0 } ;
ptrdiff_t plane_line_bytes [ 4 ] = { 0 } ;
int rgb , limited ;
int plane , c ;
if ( ! desc | | nb_planes < 1 | | nb_planes > 4 | | desc - > flags & AV_PIX_FMT_FLAG_HWACCEL )
return AVERROR ( EINVAL ) ;
rgb = ! ! ( desc - > flags & AV_PIX_FMT_FLAG_RGB ) ;
limited = ! rgb & & range ! = AVCOL_RANGE_JPEG ;
if ( desc - > flags & AV_PIX_FMT_FLAG_BITSTREAM ) {
ptrdiff_t bytewidth = av_image_get_linesize ( pix_fmt , width , 0 ) ;
uint8_t * data ;
int mono = pix_fmt = = AV_PIX_FMT_MONOWHITE | | pix_fmt = = AV_PIX_FMT_MONOBLACK ;
int fill = pix_fmt = = AV_PIX_FMT_MONOWHITE ? 0xFF : 0 ;
if ( nb_planes ! = 1 | | ! ( rgb | | mono ) | | bytewidth < 1 )
return AVERROR ( EINVAL ) ;
if ( ! dst_data )
return 0 ;
data = dst_data [ 0 ] ;
// (Bitstream + alpha will be handled incorrectly - it'll remain transparent.)
for ( ; height > 0 ; height - - ) {
memset ( data , fill , bytewidth ) ;
data + = dst_linesize [ 0 ] ;
}
return 0 ;
}
for ( c = 0 ; c < desc - > nb_components ; c + + ) {
const AVComponentDescriptor comp = desc - > comp [ c ] ;
// We try to operate on entire non-subsampled pixel groups (for
// AV_PIX_FMT_UYVY422 this would mean two consecutive pixels).
clear_block_size [ comp . plane ] = FFMAX ( clear_block_size [ comp . plane ] , comp . step ) ;
if ( clear_block_size [ comp . plane ] > MAX_BLOCK_SIZE )
return AVERROR ( EINVAL ) ;
}
// Create a byte array for clearing 1 pixel (sometimes several pixels).
for ( c = 0 ; c < desc - > nb_components ; c + + ) {
const AVComponentDescriptor comp = desc - > comp [ c ] ;
// (Multiple pixels happen e.g. with AV_PIX_FMT_UYVY422.)
int w = clear_block_size [ comp . plane ] / comp . step ;
uint8_t * c_data [ 4 ] ;
const int c_linesize [ 4 ] = { 0 } ;
uint16_t src_array [ MAX_BLOCK_SIZE ] ;
uint16_t src = 0 ;
int x ;
if ( comp . depth > 16 )
return AVERROR ( EINVAL ) ;
if ( ! rgb & & comp . depth < 8 )
return AVERROR ( EINVAL ) ;
if ( w < 1 )
return AVERROR ( EINVAL ) ;
if ( c = = 0 & & limited ) {
src = 16 < < ( comp . depth - 8 ) ;
} else if ( ( c = = 1 | | c = = 2 ) & & ! rgb ) {
src = 128 < < ( comp . depth - 8 ) ;
} else if ( c = = 3 ) {
// (Assume even limited YUV uses full range alpha.)
src = ( 1 < < comp . depth ) - 1 ;
}
for ( x = 0 ; x < w ; x + + )
src_array [ x ] = src ;
for ( x = 0 ; x < 4 ; x + + )
c_data [ x ] = & clear_block [ x ] [ 0 ] ;
av_write_image_line ( src_array , c_data , c_linesize , desc , 0 , 0 , c , w ) ;
}
for ( plane = 0 ; plane < nb_planes ; plane + + ) {
plane_line_bytes [ plane ] = av_image_get_linesize ( pix_fmt , width , plane ) ;
if ( plane_line_bytes [ plane ] < 0 )
return AVERROR ( EINVAL ) ;
}
if ( ! dst_data )
return 0 ;
for ( plane = 0 ; plane < nb_planes ; plane + + ) {
size_t bytewidth = plane_line_bytes [ plane ] ;
uint8_t * data = dst_data [ plane ] ;
int chroma_div = plane = = 1 | | plane = = 2 ? desc - > log2_chroma_h : 0 ;
int plane_h = ( ( height + ( 1 < < chroma_div ) - 1 ) ) > > chroma_div ;
for ( ; plane_h > 0 ; plane_h - - ) {
memset_bytes ( data , bytewidth , & clear_block [ plane ] [ 0 ] , clear_block_size [ plane ] ) ;
data + = dst_linesize [ plane ] ;
}
}
return 0 ;
}