@ -99,32 +99,42 @@ static int libspeex_decode_frame(AVCodecContext *avctx,
uint8_t * buf = avpkt - > data ;
int buf_size = avpkt - > size ;
LibSpeexContext * s = avctx - > priv_data ;
int16_t * output = data , * end ;
int i , num_samples ;
num_samples = s - > frame_size * avctx - > channels ;
end = output + * data_size / sizeof ( * output ) ;
int16_t * output = data ;
int out_size , ret , consumed = 0 ;
/* check output buffer size */
out_size = s - > frame_size * avctx - > channels *
av_get_bytes_per_sample ( avctx - > sample_fmt ) ;
if ( * data_size < out_size ) {
av_log ( avctx , AV_LOG_ERROR , " Output buffer is too small \n " ) ;
return AVERROR ( EINVAL ) ;
}
/* if there is not enough data left for the smallest possible frame,
reset the libspeex buffer using the current packet , otherwise ignore
the current packet and keep decoding frames from the libspeex buffer . */
if ( speex_bits_remaining ( & s - > bits ) < 43 ) {
/* check for flush packet */
if ( ! buf | | ! buf_size ) {
* data_size = 0 ;
return buf_size ;
}
/* set new buffer */
speex_bits_read_from ( & s - > bits , buf , buf_size ) ;
consumed = buf_size ;
}
for ( i = 0 ; speex_bits_remaining ( & s - > bits ) & & output + num_samples < end ; i + + ) {
int ret = speex_decode_int ( s - > dec_state , & s - > bits , output ) ;
/* decode a single frame */
ret = speex_decode_int ( s - > dec_state , & s - > bits , output ) ;
if ( ret < = - 2 ) {
av_log ( avctx , AV_LOG_ERROR , " Error decoding Speex frame. \n " ) ;
return - 1 ;
} else if ( ret = = - 1 )
// end of stream
break ;
}
if ( avctx - > channels = = 2 )
speex_decode_stereo_int ( output , s - > frame_size , & s - > stereo ) ;
output + = num_samples ;
}
avctx - > frame_size = s - > frame_size * i ;
* data_size = avctx - > channels * avctx - > frame_size * sizeof ( * output ) ;
return buf_size ;
* data_size = out_size ;
return consumed ;
}
static av_cold int libspeex_decode_close ( AVCodecContext * avctx )
@ -138,6 +148,12 @@ static av_cold int libspeex_decode_close(AVCodecContext *avctx)
return 0 ;
}
static av_cold void libspeex_decode_flush ( AVCodecContext * avctx )
{
LibSpeexContext * s = avctx - > priv_data ;
speex_bits_reset ( & s - > bits ) ;
}
AVCodec ff_libspeex_decoder = {
. name = " libspeex " ,
. type = AVMEDIA_TYPE_AUDIO ,
@ -146,5 +162,7 @@ AVCodec ff_libspeex_decoder = {
. init = libspeex_decode_init ,
. close = libspeex_decode_close ,
. decode = libspeex_decode_frame ,
. flush = libspeex_decode_flush ,
. capabilities = CODEC_CAP_SUBFRAMES | CODEC_CAP_DELAY ,
. long_name = NULL_IF_CONFIG_SMALL ( " libspeex Speex " ) ,
} ;