hb_work_object_t * w = _w;
hb_work_private_t * r = w->private_data;
hb_fifo_t ** fifos;
- hb_buffer_t * buf;
+ hb_buffer_t * buf = NULL;
hb_list_t * list;
int n;
int chapter = -1;
r->start_found = 2;
r->duration -= r->job->pts_to_start;
r->job->pts_to_start = pts_to_start;
+ hb_buffer_close(&buf);
}
+ // hb_stream_seek_ts does nothing for TS streams and will return
+ // an error. In this case, the current buf remains valid and
+ // gets processed below.
}
else if( r->stream )
{
break;
}
- if (r->bd)
+ if (buf == NULL)
{
- if( (buf = hb_bd_read( r->bd )) == NULL )
- {
- break;
- }
- }
- else if (r->dvd)
- {
- if( (buf = hb_dvd_read( r->dvd )) == NULL )
- {
- break;
- }
+ if (r->bd)
+ {
+ if( (buf = hb_bd_read( r->bd )) == NULL )
+ {
+ break;
+ }
+ }
+ else if (r->dvd)
+ {
+ if( (buf = hb_dvd_read( r->dvd )) == NULL )
+ {
+ break;
+ }
+ }
+ else if (r->stream)
+ {
+ if ( (buf = hb_stream_read( r->stream )) == NULL )
+ {
+ break;
+ }
+ }
}
- else if (r->stream)
+ if (r->stream && r->start_found == 2 )
{
- if ( (buf = hb_stream_read( r->stream )) == NULL )
- {
- break;
- }
- if ( r->start_found == 2 )
- {
// We will inspect the timestamps of each frame in sync
// to skip from this seek point to the timestamp we
// want to start at.
r->job->pts_to_start = 0;
}
r->start_found = 1;
- }
}
(hb_demux[r->title->demuxer])( buf, list, &r->demux );
break;
}
- if ( !r->start_found &&
- start >= r->pts_to_start )
+ if (!r->start_found && start >= r->pts_to_start)
{
// pts_to_start point found
r->start_found = 1;
+ if (r->stream)
+ {
+ // libav multi-threaded decoders can get into
+ // a bad state if the initial data is not
+ // decodable. So try to improve the chances of
+ // a good start by waiting for an initial iframe
+ hb_stream_set_need_keyframe(r->stream, 1);
+ hb_buffer_close( &buf );
+ continue;
+ }
}
// This log is handy when you need to debug timing problems
//hb_log("id %x scr_offset %"PRId64
push_buf( r, fifos[n], buf_copy );
}
push_buf( r, fifos[0], buf );
+ buf = NULL;
}
else
{
return NULL;
}
+void hb_stream_set_need_keyframe(hb_stream_t *stream, int need_keyframe)
+{
+ if ( stream->hb_stream_type == transport )
+ {
+ // Only wait for a keyframe if the stream is known to have IDRs
+ if ( stream->has_IDRs )
+ {
+ stream->need_keyframe = need_keyframe;
+ }
+ }
+ else if ( stream->hb_stream_type == program )
+ {
+ // Only wait for a keyframe if the stream is known to have IDRs
+ if ( stream->has_IDRs )
+ {
+ stream->need_keyframe = need_keyframe;
+ }
+ }
+ else
+ {
+ stream->need_keyframe = need_keyframe;
+ }
+}
+
void hb_ts_stream_reset(hb_stream_t *stream)
{
int i;