]> granicus.if.org Git - handbrake/commitdiff
fix crash during 2 pass encoding TS or PS source
authorjstebbins <jstebbins.hb@gmail.com>
Sat, 15 Oct 2011 00:16:02 +0000 (00:16 +0000)
committerjstebbins <jstebbins.hb@gmail.com>
Sat, 15 Oct 2011 00:16:02 +0000 (00:16 +0000)
stream.c cached information that it probed during the first time it
opened any source.  Then later it would re-use that cached data.  I
was prematurely deleting the cached data during 2 pass encodes.

The problem is that there is no way to know when the cached data is no
longer needed.  You could have a thousand items in the queue all using
the same source, or you could have only 1.  So you either have to (a)
keep the cached data indefinitely, or (b) you have to be able to handle
the case where scanned cached data is flushed before you start an
encode. (a) is poor design.  And if you choose (b) you might as well
eliminate the cache all together.  It doesn't really save any time and
only complicates the code.

In summary, the cache is gone.

git-svn-id: svn://svn.handbrake.fr/HandBrake/trunk@4286 b64f7644-9d1e-0410-96f1-a4d463321fa5

libhb/stream.c

index 9f2775a4beeb38cfcc5be87ff91bcff786a692fe..90f6a895b5332b377269a4ec1adcdb896704a6f7 100644 (file)
@@ -282,39 +282,6 @@ static void hb_ps_resolve_stream_types(hb_stream_t *stream);
 void hb_ts_stream_reset(hb_stream_t *stream);
 void hb_ps_stream_reset(hb_stream_t *stream);
 
-/*
- * streams have a bunch of state that's learned during the scan. We don't
- * want to throw away the state when scan does a close then relearn
- * everything when reader does an open. So we save the stream state on
- * the close following a scan and reuse it when 'reader' does an open.
- */
-static hb_list_t *stream_state_list;
-
-static hb_stream_t *hb_stream_lookup( const char *path )
-{
-    if ( stream_state_list == NULL )
-        return NULL;
-
-    hb_stream_t *ss;
-    int i = 0;
-
-    while ( ( ss = hb_list_item( stream_state_list, i++ ) ) != NULL )
-    {
-        if ( strcmp( path, ss->path ) == 0 )
-        {
-            break;
-        }
-    }
-    return ss;
-}
-
-static void hb_stream_state_delete( hb_stream_t *ss )
-{
-    hb_list_rem( stream_state_list, ss );
-    free( ss->path );
-    free( ss );
-}
-
 /*
  * logging routines.
  * these frontend hb_log because transport streams can have a lot of errors
@@ -749,6 +716,71 @@ static void hb_stream_delete_ps_entry(hb_stream_t *stream, int indx)
     }
 }
 
+static void prune_streams(hb_stream_t *d)
+{
+    if ( d->hb_stream_type == transport )
+    {
+        int ii, jj;
+        for ( ii = 0; ii < d->ts.count; ii++)
+        {
+            // If probing didn't find audio or video, and the pid
+            // is not the PCR, remove the track
+            if ( ts_stream_kind ( d, ii ) == U &&
+                 !d->ts.list[ii].is_pcr )
+            {
+                hb_stream_delete_ts_entry(d, ii);
+                continue;
+            }
+
+            if ( ts_stream_kind ( d, ii ) == A )
+            {
+                for ( jj = d->ts.list[ii].pes_list; jj != -1;
+                      jj = d->pes.list[jj].next )
+                {
+                    if ( audio_inactive( d, d->pes.list[jj].stream_id,
+                                         d->pes.list[jj].stream_id_ext ) )
+                    {
+                        hb_stream_delete_ps_entry(d, jj);
+                    }
+                }
+                if ( !d->ts.list[ii].is_pcr &&
+                     hb_stream_try_delete_ts_entry(d, ii) )
+                {
+                    continue;
+                }
+            }
+        }
+        // reset to beginning of file and reset some stream 
+        // state information
+        hb_stream_seek( d, 0. );
+    }
+    else if ( d->hb_stream_type == program )
+    {
+        int ii;
+        for ( ii = 0; ii < d->pes.count; ii++)
+        {
+            // If probing didn't find audio or video, remove the track
+            if ( d->pes.list[ii].stream_kind == U )
+            {
+                hb_stream_delete_ps_entry(d, ii);
+            }
+
+            if ( d->pes.list[ii].stream_kind == A &&
+                 audio_inactive( d, d->pes.list[ii].stream_id,
+                                 d->pes.list[ii].stream_id_ext ) )
+            {
+                // this PID isn't wanted (we don't have a codec for it
+                // or scan didn't find audio parameters)
+                hb_stream_delete_ps_entry(d, ii);
+                continue;
+            }
+        }
+        // reset to beginning of file and reset some stream 
+        // state information
+        hb_stream_seek( d, 0. );
+    }
+}
+
 /***********************************************************************
  * hb_stream_open
  ***********************************************************************
@@ -771,108 +803,15 @@ hb_stream_t * hb_stream_open( char *path, hb_title_t *title, int scan )
         return NULL;
     }
 
-    /*
-     * if we're opening the stream to read & convert, we need
-     * the state we saved when we scanned the stream. if we're
-     * opening the stream to scan it we want to rebuild the state
-     * (even if we have saved state, the stream may have changed).
-     */
-    hb_stream_t *ss = hb_stream_lookup( path );
-    if ( !scan && ss && ss->hb_stream_type != ffmpeg )
-    {
-        /*
-         * copy the saved state since we might be encoding the same stream
-         * multiple times.
-         */
-        memcpy( d, ss, sizeof(*d) );
-        d->file_handle = f;
-        d->title = title;
-        d->scan = scan;
-        d->path = strdup( path );
-
-        if ( d->hb_stream_type == transport )
-        {
-            d->ts.packet = malloc( d->packetsize );
-
-            int ii, jj;
-            for ( ii = 0; ii < d->ts.count; ii++)
-            {
-                // If probing didn't find audio or video, and the pid
-                // is not the PCR, remove the track
-                if ( ts_stream_kind ( d, ii ) == U &&
-                     !d->ts.list[ii].is_pcr )
-                {
-                    hb_stream_delete_ts_entry(d, ii);
-                    continue;
-                }
-
-                if ( ts_stream_kind ( d, ii ) == A )
-                {
-                    for ( jj = d->ts.list[ii].pes_list; jj != -1;
-                          jj = d->pes.list[jj].next )
-                    {
-                        if ( audio_inactive( d, d->pes.list[jj].stream_id,
-                                             d->pes.list[jj].stream_id_ext ) )
-                        {
-                            hb_stream_delete_ps_entry(d, jj);
-                        }
-                    }
-                    if ( !d->ts.list[ii].is_pcr &&
-                         hb_stream_try_delete_ts_entry(d, ii) )
-                    {
-                        continue;
-                    }
-                }
-                d->ts.list[ii].buf = hb_buffer_init(d->packetsize);
-                d->ts.list[ii].extra_buf = hb_buffer_init(d->packetsize);
-                d->ts.list[ii].buf->size = 0;
-                d->ts.list[ii].extra_buf->size = 0;
-            }
-            // reset to beginning of file and reset some stream 
-            // state information
-            hb_stream_seek( d, 0. );
-        }
-        else if ( d->hb_stream_type == program )
-        {
-            int ii;
-            for ( ii = 0; ii < d->pes.count; ii++)
-            {
-                // If probing didn't find audio or video, remove the track
-                if ( d->pes.list[ii].stream_kind == U )
-                {
-                    hb_stream_delete_ps_entry(d, ii);
-                }
-
-                if ( d->pes.list[ii].stream_kind == A &&
-                     audio_inactive( d, d->pes.list[ii].stream_id,
-                                     d->pes.list[ii].stream_id_ext ) )
-                {
-                    // this PID isn't wanted (we don't have a codec for it
-                    // or scan didn't find audio parameters)
-                    hb_stream_delete_ps_entry(d, ii);
-                    continue;
-                }
-            }
-            // reset to beginning of file and reset some stream 
-            // state information
-            hb_stream_seek( d, 0. );
-        }
-        return d;
-    }
-    else if( title && !( title->flags & HBTF_NO_IDR ) )
+    if( title && !( title->flags & HBTF_NO_IDR ) )
     {
         d->has_IDRs = 1;
     }
 
     /*
-     * opening for scan - delete any saved state then (re)scan the stream.
      * If it's something we can deal with (MPEG2 PS or TS) return a stream
      * reference structure & null otherwise.
      */
-    if ( ss != NULL )
-    {
-        hb_stream_state_delete( ss );
-    }
     d->file_handle = f;
     d->title = title;
     d->scan = scan;
@@ -881,6 +820,13 @@ hb_stream_t * hb_stream_open( char *path, hb_title_t *title, int scan )
     {
         if ( hb_stream_get_type( d ) != 0 )
         {
+            if( !scan )
+            {
+                prune_streams( d );
+            }
+            // reset to beginning of file and reset some stream 
+            // state information
+            hb_stream_seek( d, 0. );
             return d;
         }
         fclose( d->file_handle );
@@ -1036,23 +982,7 @@ void hb_stream_close( hb_stream_t ** _d )
                 (double)stream->frames );
     }
 
-    /*
-     * if the stream was opened for a scan, cache the result, otherwise delete
-     * the state.
-     */
-    if ( stream->scan )
-    {
-        hb_stream_delete_dynamic( stream );
-        if ( stream_state_list == NULL )
-        {
-            stream_state_list = hb_list_init();
-        }
-        hb_list_add( stream_state_list, stream );
-    }
-    else
-    {
-        hb_stream_delete( stream );
-    }
+    hb_stream_delete( stream );
     *_d = NULL;
 }
 
@@ -2226,48 +2156,59 @@ static void hb_ts_stream_init(hb_stream_t *stream)
     }
     hb_ts_resolve_pid_types(stream);
 
-    hb_log("Found the following PIDS");
-    hb_log("    Video PIDS : ");
-    for (i=0; i < stream->ts.count; i++)
+    if( stream->scan )
     {
-        if ( ts_stream_kind( stream, i ) == V )
+        hb_log("Found the following PIDS");
+        hb_log("    Video PIDS : ");
+        for (i=0; i < stream->ts.count; i++)
         {
-            hb_log( "      0x%x type %s (0x%x)%s",
-                    stream->ts.list[i].pid,
-                    stream_type_name2(stream, 
-                            &stream->pes.list[stream->ts.list[i].pes_list]),
-                    ts_stream_type( stream, i ),
-                    stream->ts.list[i].is_pcr ? " (PCR)" : "");
+            if ( ts_stream_kind( stream, i ) == V )
+            {
+                hb_log( "      0x%x type %s (0x%x)%s",
+                        stream->ts.list[i].pid,
+                        stream_type_name2(stream, 
+                                &stream->pes.list[stream->ts.list[i].pes_list]),
+                        ts_stream_type( stream, i ),
+                        stream->ts.list[i].is_pcr ? " (PCR)" : "");
+            }
         }
-    }
-    hb_log("    Audio PIDS : ");
-    for (i = 0; i < stream->ts.count; i++)
-    {
-        if ( ts_stream_kind( stream, i ) == A )
+        hb_log("    Audio PIDS : ");
+        for (i = 0; i < stream->ts.count; i++)
+        {
+            if ( ts_stream_kind( stream, i ) == A )
+            {
+                hb_log( "      0x%x type %s (0x%x)%s",
+                        stream->ts.list[i].pid,
+                        stream_type_name2(stream,
+                                &stream->pes.list[stream->ts.list[i].pes_list]),
+                        ts_stream_type( stream, i ),
+                        stream->ts.list[i].is_pcr ? " (PCR)" : "");
+            }
+        }
+        hb_log("    Other PIDS : ");
+        for (i = 0; i < stream->ts.count; i++)
         {
-            hb_log( "      0x%x type %s (0x%x)%s",
-                    stream->ts.list[i].pid,
-                    stream_type_name2(stream,
-                            &stream->pes.list[stream->ts.list[i].pes_list]),
-                    ts_stream_type( stream, i ),
-                    stream->ts.list[i].is_pcr ? " (PCR)" : "");
+            if ( ts_stream_kind( stream, i ) == N ||
+                 ts_stream_kind( stream, i ) == P )
+            {
+                hb_log( "      0x%x type %s (0x%x)%s",
+                        stream->ts.list[i].pid,
+                        stream_type_name2(stream,
+                                &stream->pes.list[stream->ts.list[i].pes_list]),
+                        ts_stream_type( stream, i ),
+                        stream->ts.list[i].is_pcr ? " (PCR)" : "");
+            }
+            if ( ts_stream_kind( stream, i ) == N )
+                hb_stream_delete_ts_entry(stream, i);
         }
     }
-    hb_log("    Other PIDS : ");
-    for (i = 0; i < stream->ts.count; i++)
+    else
     {
-        if ( ts_stream_kind( stream, i ) == N ||
-             ts_stream_kind( stream, i ) == P )
+        for (i = 0; i < stream->ts.count; i++)
         {
-            hb_log( "      0x%x type %s (0x%x)%s",
-                    stream->ts.list[i].pid,
-                    stream_type_name2(stream,
-                            &stream->pes.list[stream->ts.list[i].pes_list]),
-                    ts_stream_type( stream, i ),
-                    stream->ts.list[i].is_pcr ? " (PCR)" : "");
+            if ( ts_stream_kind( stream, i ) == N )
+                hb_stream_delete_ts_entry(stream, i);
         }
-        if ( ts_stream_kind( stream, i ) == N )
-            hb_stream_delete_ts_entry(stream, i);
     }
 }
 
@@ -2289,45 +2230,56 @@ static void hb_ps_stream_init(hb_stream_t *stream)
     hb_ps_stream_find_streams(stream);
     hb_ps_resolve_stream_types(stream);
 
-    hb_log("Found the following streams");
-    hb_log("    Video Streams : ");
-    for (i=0; i < stream->pes.count; i++)
+    if( stream->scan )
     {
-        if ( stream->pes.list[i].stream_kind == V )
+        hb_log("Found the following streams");
+        hb_log("    Video Streams : ");
+        for (i=0; i < stream->pes.count; i++)
         {
-            hb_log( "      0x%x-0x%x type %s (0x%x)",
-                    stream->pes.list[i].stream_id,
-                    stream->pes.list[i].stream_id_ext,
-                    stream_type_name2(stream,
-                                     &stream->pes.list[i]),
-                    stream->pes.list[i].stream_type);
+            if ( stream->pes.list[i].stream_kind == V )
+            {
+                hb_log( "      0x%x-0x%x type %s (0x%x)",
+                        stream->pes.list[i].stream_id,
+                        stream->pes.list[i].stream_id_ext,
+                        stream_type_name2(stream,
+                                         &stream->pes.list[i]),
+                        stream->pes.list[i].stream_type);
+            }
         }
-    }
-    hb_log("    Audio Streams : ");
-    for (i = 0; i < stream->pes.count; i++)
-    {
-        if ( stream->pes.list[i].stream_kind == A )
+        hb_log("    Audio Streams : ");
+        for (i = 0; i < stream->pes.count; i++)
+        {
+            if ( stream->pes.list[i].stream_kind == A )
+            {
+                hb_log( "      0x%x-0x%x type %s (0x%x)",
+                        stream->pes.list[i].stream_id,
+                        stream->pes.list[i].stream_id_ext,
+                        stream_type_name2(stream,
+                                         &stream->pes.list[i]),
+                        stream->pes.list[i].stream_type );
+            }
+        }
+        hb_log("    Other Streams : ");
+        for (i = 0; i < stream->pes.count; i++)
         {
-            hb_log( "      0x%x-0x%x type %s (0x%x)",
-                    stream->pes.list[i].stream_id,
-                    stream->pes.list[i].stream_id_ext,
-                    stream_type_name2(stream,
-                                     &stream->pes.list[i]),
-                    stream->pes.list[i].stream_type );
+            if ( stream->pes.list[i].stream_kind == N )
+            {
+                hb_log( "      0x%x-0x%x type %s (0x%x)",
+                        stream->pes.list[i].stream_id,
+                        stream->pes.list[i].stream_id_ext,
+                        stream_type_name2(stream,
+                                         &stream->pes.list[i]),
+                        stream->pes.list[i].stream_type );
+                hb_stream_delete_ps_entry(stream, i);
+            }
         }
     }
-    hb_log("    Other Streams : ");
-    for (i = 0; i < stream->pes.count; i++)
+    else
     {
-        if ( stream->pes.list[i].stream_kind == N )
+        for (i = 0; i < stream->pes.count; i++)
         {
-            hb_log( "      0x%x-0x%x type %s (0x%x)",
-                    stream->pes.list[i].stream_id,
-                    stream->pes.list[i].stream_id_ext,
-                    stream_type_name2(stream,
-                                     &stream->pes.list[i]),
-                    stream->pes.list[i].stream_type );
-            hb_stream_delete_ps_entry(stream, i);
+            if ( stream->pes.list[i].stream_kind == N )
+                hb_stream_delete_ps_entry(stream, i);
         }
     }
 }