mpegtsmux: Fixup program array indices after stream removal
authorJan Alexander Steffens (heftig) <jan.steffens@ltnglobal.com>
Thu, 20 May 2021 08:09:57 +0000 (10:09 +0200)
committerGStreamer Marge Bot <gitlab-merge-bot@gstreamer-foundation.org>
Thu, 20 May 2021 13:35:06 +0000 (13:35 +0000)
Each stream stores the `program_array_index` of its position in its
program's `streams` array. When we remove a stream from this array, we
need to correct the `program_array_index` of all streams that were
backshifted by the removal.

Also extract the removal into a new function and add some more safety
checks.

Part-of: <https://gitlab.freedesktop.org/gstreamer/gst-plugins-bad/-/merge_requests/2266>

gst/mpegtsmux/tsmux/tsmux.c

index 566111bf8ec011c36684b30fbd744104d95921e6..632f9f8f446e1219c380a66fc8ac3ac9c34b09af 100644 (file)
@@ -757,6 +757,31 @@ tsmux_find_stream (TsMux * mux, guint16 pid)
   return found;
 }
 
+static gboolean
+tsmux_program_remove_stream (TsMuxProgram * program, TsMuxStream * stream)
+{
+  GArray *streams = program->streams;
+  TsMuxStream *s;
+  gint i;
+
+  i = stream->program_array_index;
+  g_return_val_if_fail (i >= 0, FALSE);
+
+  s = g_array_index (streams, TsMuxStream *, i);
+  g_return_val_if_fail (s == stream, FALSE);
+
+  g_array_remove_index (streams, i);
+
+  /* Correct indices of remaining streams, if any */
+  for (; i < streams->len; i++) {
+    s = g_array_index (streams, TsMuxStream *, i);
+    s->program_array_index -= 1;
+  }
+
+  return streams->len == 0;
+}
+
+
 gboolean
 tsmux_remove_stream (TsMux * mux, guint16 pid, TsMuxProgram * program)
 {
@@ -769,20 +794,16 @@ tsmux_remove_stream (TsMux * mux, guint16 pid, TsMuxProgram * program)
     TsMuxStream *stream = (TsMuxStream *) cur->data;
 
     if (tsmux_stream_get_pid (stream) == pid) {
-      if (program->streams->len == 1) {
-        tsmux_program_delete (mux, program);
-        ret = TRUE;
-      } else {
-        program->streams =
-            g_array_remove_index (program->streams,
-            stream->program_array_index);
-      }
-
+      ret = tsmux_program_remove_stream (program, stream);
       mux->streams = g_list_remove (mux->streams, stream);
       tsmux_stream_free (stream);
-      return ret;
+      break;
     }
   }
+
+  if (ret)
+    tsmux_program_delete (mux, program);
+
   return ret;
 }