diff options
author | jstebbins <[email protected]> | 2010-04-03 21:14:43 +0000 |
---|---|---|
committer | jstebbins <[email protected]> | 2010-04-03 21:14:43 +0000 |
commit | 9de28e6ef25b02d4569a06293a941b1d874304d3 (patch) | |
tree | 0fda94962695504b7abff1826ded9a42715c6e38 /libhb/work.c | |
parent | b79e337bda086bc9721cb35754ae474c46e2e2bb (diff) |
fix silence insertion problem in audio sync and pipeline stall problem
the current audio buffer was being dropped when silence was inserted, causing
the time to fall even further behind and provoke more silence insertion
in some cases.
with pont-to-point, it is possible for one stream to complete before the
other(s). when the work_loop exits for that stream, the fifo is no longer
serviced and may fill. This can back up and cause a stall in reader
causing the streams that are not yet complete to stall. The Solution is
to continue servicing the fifo after work for a stream is complete. This was
complicated by the fact that the video sync work object was being used as the
indicator that all work was finished. When it exited everything was told
to stop. So now, the muxer work object (last in the chain) is the
indicator when work is done.
git-svn-id: svn://svn.handbrake.fr/HandBrake/trunk@3197 b64f7644-9d1e-0410-96f1-a4d463321fa5
Diffstat (limited to 'libhb/work.c')
-rw-r--r-- | libhb/work.c | 85 |
1 files changed, 63 insertions, 22 deletions
diff --git a/libhb/work.c b/libhb/work.c index af4665bbc..4448278a2 100644 --- a/libhb/work.c +++ b/libhb/work.c @@ -381,6 +381,7 @@ static void do_job( hb_job_t * job, int cpu_count ) hb_title_t * title; int i, j; hb_work_object_t * w; + hb_work_object_t * sync; hb_work_object_t * muxer; hb_interjob_t * interjob; @@ -641,12 +642,7 @@ static void do_job( hb_job_t * job, int cpu_count ) } /* Synchronization */ - if( hb_sync_init( job ) ) - { - hb_error( "Failure to initialise sync" ); - *job->die = 1; - goto cleanup; - } + sync = hb_sync_init( job ); /* Video decoder */ int vcodec = title->video_codec? title->video_codec : WORK_DECMPEG2; @@ -658,7 +654,10 @@ static void do_job( hb_job_t * job, int cpu_count ) /* Video renderer */ hb_list_add( job->list_work, ( w = hb_get_work( WORK_RENDER ) ) ); w->fifo_in = job->fifo_sync; - w->fifo_out = job->fifo_render; + if( !job->indepth_scan ) + w->fifo_out = job->fifo_render; + else + w->fifo_out = NULL; if( !job->indepth_scan ) { @@ -860,7 +859,7 @@ static void do_job( hb_job_t * job, int cpu_count ) job->done = 0; /* Launch processing threads */ - for( i = 1; i < hb_list_count( job->list_work ); i++ ) + for( i = 0; i < hb_list_count( job->list_work ); i++ ) { w = hb_list_item( job->list_work, i ); w->done = &job->done; @@ -875,12 +874,32 @@ static void do_job( hb_job_t * job, int cpu_count ) HB_LOW_PRIORITY ); } - // The muxer requires track information that's set up by the encoder - // init routines so we have to init the muxer last. - muxer = job->indepth_scan? NULL : hb_muxer_init( job ); + if ( job->indepth_scan ) + { + muxer = NULL; + w = sync; + sync->done = &job->done; + } + else + { + sync->done = &job->done; + sync->thread_sleep_interval = 10; + if( sync->init( w, job ) ) + { + hb_error( "Failure to initialise thread '%s'", w->name ); + *job->die = 1; + goto cleanup; + } + sync->thread = hb_thread_init( sync->name, work_loop, sync, + HB_LOW_PRIORITY ); - w = hb_list_item( job->list_work, 0 ); - while( !*job->die && w->status != HB_WORK_DONE ) + // The muxer requires track information that's set up by the encoder + // init routines so we have to init the muxer last. + muxer = hb_muxer_init( job ); + w = muxer; + } + + while ( !*job->die && !*w->done && w->status != HB_WORK_DONE ) { hb_buffer_t * buf_in, * buf_out; @@ -896,12 +915,17 @@ static void do_job( hb_job_t * job, int cpu_count ) break; } + buf_out = NULL; w->status = w->work( w, &buf_in, &buf_out ); if( buf_in ) { hb_buffer_close( &buf_in ); } + if ( buf_out && w->fifo_out == NULL ) + { + hb_buffer_close( &buf_out ); + } if( buf_out ) { while ( !*job->die ) @@ -914,9 +938,20 @@ static void do_job( hb_job_t * job, int cpu_count ) } } } - hb_list_rem( job->list_work, w ); - w->close( w ); - free( w ); + + job->done = 1; + if( muxer != NULL ) + { + muxer->close( muxer ); + free( muxer ); + + if( sync->thread != NULL ) + { + hb_thread_close( &sync->thread ); + sync->close( sync ); + } + free( sync ); + } hb_handle_t * h = job->h; hb_state_t state; @@ -926,12 +961,6 @@ static void do_job( hb_job_t * job, int cpu_count ) cleanup: /* Stop the write thread (thread_close will block until the muxer finishes) */ - if( muxer != NULL ) - { - hb_thread_close( &muxer->thread ); - muxer->close( muxer ); - } - job->done = 1; /* Close work objects */ @@ -1125,6 +1154,10 @@ static void work_loop( void * _w ) { hb_buffer_close( &buf_in ); } + if ( buf_out && w->fifo_out == NULL ) + { + hb_buffer_close( &buf_out ); + } if( buf_out ) { while ( !*w->done ) @@ -1137,4 +1170,12 @@ static void work_loop( void * _w ) } } } + // Consume data in incoming fifo till job complete so that + // residual data does not stall the pipeline + while( !*w->done ) + { + buf_in = hb_fifo_get_wait( w->fifo_in ); + if ( buf_in != NULL ) + hb_buffer_close( &buf_in ); + } } |