From 1c0ae631e0898549b16b029e55e4526040872d5d Mon Sep 17 00:00:00 2001 From: jstebbins Date: Mon, 17 Oct 2011 16:24:24 +0000 Subject: [PATCH] fix a problem with resolution changes in h.264 First, the scaling code in decavcodec.c was broken and didn't properly compensate for frames that had a different resolution than the rest of the stream. Second, libav can not handle resolution changes when doing frame based multi-threading. So disable threading when resolution changes are detected. git-svn-id: svn://svn.handbrake.fr/HandBrake/trunk@4295 b64f7644-9d1e-0410-96f1-a4d463321fa5 --- libhb/common.h | 1 + libhb/decavcodec.c | 32 +++++++++++++++++++++++++------- libhb/scan.c | 17 +++++++++++++++++ 3 files changed, 43 insertions(+), 7 deletions(-) diff --git a/libhb/common.h b/libhb/common.h index 65e022880..39a2c45cd 100644 --- a/libhb/common.h +++ b/libhb/common.h @@ -646,6 +646,7 @@ struct hb_title_s double aspect; // aspect ratio for the title's video double container_aspect; // aspect ratio from container (0 if none) + int has_resolution_change; int width; int height; int pixel_aspect_width; diff --git a/libhb/decavcodec.c b/libhb/decavcodec.c index 28498273c..6cb44b160 100644 --- a/libhb/decavcodec.c +++ b/libhb/decavcodec.c @@ -74,6 +74,7 @@ struct hb_work_private_s hb_title_t *title; AVCodecContext *context; AVCodecParserContext *parser; + int threads; int video_codec_opened; hb_list_t *list; double duration; // frame duration (for video) @@ -91,6 +92,9 @@ struct hb_work_private_s pts_heap_t pts_heap; void* buffer; struct SwsContext *sws_context; // if we have to rescale or convert color space + int sws_width; + int sws_height; + int sws_pix_fmt; hb_downmix_t *downmix; int cadence[12]; int wait_for_keyframe; @@ -548,14 +552,23 @@ static hb_buffer_t *copy_frame( hb_work_private_t *pv, AVFrame *frame ) AVPicture dstpic; avpicture_fill( &dstpic, dst, PIX_FMT_YUV420P, w, h ); - if ( ! pv->sws_context ) + if ( ! pv->sws_context || + pv->sws_width != context->width || + pv->sws_height != context->height || + pv->sws_pix_fmt != context->pix_fmt ) { - pv->sws_context = hb_sws_get_context( w, h, context->pix_fmt, - w, h, PIX_FMT_YUV420P, - SWS_LANCZOS|SWS_ACCURATE_RND); + if( pv->sws_context ) + sws_freeContext( pv->sws_context ); + pv->sws_context = hb_sws_get_context( + context->width, context->height, context->pix_fmt, + w, h, PIX_FMT_YUV420P, + SWS_LANCZOS|SWS_ACCURATE_RND); + pv->sws_width = context->width; + pv->sws_height = context->height; + pv->sws_pix_fmt = context->pix_fmt; } sws_scale( pv->sws_context, (const uint8_t* const *)frame->data, - frame->linesize, 0, h, + frame->linesize, 0, context->height, dstpic.data, dstpic.linesize ); } else @@ -979,6 +992,11 @@ static int decavcodecvInit( hb_work_object_t * w, hb_job_t * job ) pv->title = w->title; pv->list = hb_list_init(); + if( pv->job ) + { + if( !pv->job->title || !pv->job->title->has_resolution_change ) + pv->threads = HB_FFMPEG_THREADS_AUTO; + } if ( pv->title->opaque_priv ) { AVFormatContext *ic = (AVFormatContext*)pv->title->opaque_priv; @@ -994,7 +1012,7 @@ static int decavcodecvInit( hb_work_object_t * w, hb_job_t * job ) pv->context->error_recognition = 1; pv->context->error_concealment = FF_EC_GUESS_MVS|FF_EC_DEBLOCK; - if ( hb_avcodec_open( pv->context, codec, NULL, pv->job ? HB_FFMPEG_THREADS_AUTO : 0 ) ) + if ( hb_avcodec_open( pv->context, codec, NULL, pv->threads ) ) { hb_log( "decavcodecvInit: avcodec_open failed" ); return 1; @@ -1156,7 +1174,7 @@ static int decavcodecvWork( hb_work_object_t * w, hb_buffer_t ** buf_in, return HB_WORK_OK; } // disable threaded decoding for scan, can cause crashes - if ( hb_avcodec_open( pv->context, codec, NULL, pv->job ? HB_FFMPEG_THREADS_AUTO : 0 ) ) + if ( hb_avcodec_open( pv->context, codec, NULL, pv->threads ) ) { hb_log( "decavcodecvWork: avcodec_open failed" ); *buf_out = hb_buffer_init( 0 );; diff --git a/libhb/scan.c b/libhb/scan.c index 29547bc9b..df20d2069 100644 --- a/libhb/scan.c +++ b/libhb/scan.c @@ -468,6 +468,22 @@ static void most_common_info( info_list_t *info_list, hb_work_info_t *info ) *info = info_list[biggest].info; } +static int has_resolution_change( info_list_t *info_list ) +{ + int w, h, i; + + if( !info_list[0].count ) + return 0; + w = info_list[0].info.width; + h = info_list[0].info.height; + for ( i = 1; info_list[i].count; ++i ) + { + if ( w != info_list[i].info.width || h != info_list[i].info.height ) + return 1; + } + return 0; +} + static int is_close_to( int val, int target, int thresh ) { int diff = val - target; @@ -819,6 +835,7 @@ skip_preview: hb_work_info_t vid_info; most_common_info( info_list, &vid_info ); + title->has_resolution_change = has_resolution_change( info_list ); if ( title->video_codec_name == NULL ) { title->video_codec_name = strdup( vid_info.name ); -- 2.40.0