mpegts: Add provisional AV1 mapping

The main difference with the WIP av1-in-mpegts mapping is that the payload data
is not startcode-escaped. Most of the rest is sensible usage of it:

* Custom AV1G (AV1 Gstreamer) registration descriptor instead of AV01
* AV1CodecConfigurationRecord is stored in the same 0x80 custom descriptor and
  conforms fully to the isobmff spec (i.e. does not the HDR fields from the
  provisional mpegts specification which conflict with that one).
* Data is stored as OBU
* Access Unit is the frame level (same as provisional mpegts mapping)

Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/4442>
This commit is contained in:
Edward Hervey 2023-04-18 11:37:25 +02:00 committed by GStreamer Marge Bot
parent 5480894dee
commit 19cea52dbd
7 changed files with 55 additions and 2 deletions

View file

@ -223744,7 +223744,7 @@
"presence": "sometimes" "presence": "sometimes"
}, },
"video_%%01x_%%05x": { "video_%%01x_%%05x": {
"caps": "video/mpeg:\n mpegversion: { (int)1, (int)2, (int)4 }\n systemstream: false\nvideo/x-h264:\n stream-format: byte-stream\nvideo/x-h265:\n stream-format: byte-stream\nvideo/x-vp9:\nvideo/x-dirac:\nvideo/x-cavs:\nvideo/x-wmv:\n wmvversion: 3\n format: WVC1\nimage/x-jpc:\nimage/x-jxsc:\n", "caps": "video/mpeg:\n mpegversion: { (int)1, (int)2, (int)4 }\n systemstream: false\nvideo/x-h264:\n stream-format: byte-stream\nvideo/x-h265:\n stream-format: byte-stream\nvideo/x-vp9:\nvideo/x-av1:\n stream-format: obu-stream\n alignment: frame\nvideo/x-dirac:\nvideo/x-cavs:\nvideo/x-wmv:\n wmvversion: 3\n format: WVC1\nimage/x-jpc:\nimage/x-jxsc:\n",
"direction": "src", "direction": "src",
"presence": "sometimes" "presence": "sometimes"
} }
@ -224002,7 +224002,7 @@
"long-name": "MPEG Transport Stream Muxer", "long-name": "MPEG Transport Stream Muxer",
"pad-templates": { "pad-templates": {
"sink_%%d": { "sink_%%d": {
"caps": "video/mpeg:\n parsed: true\n mpegversion: { (int)1, (int)2, (int)4 }\n systemstream: false\nvideo/x-dirac:\nimage/x-jpc:\n alignment: frame\nvideo/x-h264:\n stream-format: byte-stream\n alignment: { (string)au, (string)nal }\nvideo/x-h265:\n stream-format: byte-stream\n alignment: { (string)au, (string)nal }\naudio/mpeg:\n parsed: true\n mpegversion: 1\naudio/mpeg:\n framed: true\n mpegversion: { (int)2, (int)4 }\n stream-format: { (string)adts, (string)raw }\naudio/x-lpcm:\n width: { (int)16, (int)20, (int)24 }\n rate: { (int)48000, (int)96000 }\n channels: [ 1, 8 ]\n dynamic_range: [ 0, 255 ]\n emphasis: { (boolean)false, (boolean)true }\n mute: { (boolean)false, (boolean)true }\naudio/x-ac3:\n framed: true\naudio/x-dts:\n framed: true\naudio/x-opus:\n channels: [ 1, 255 ]\naudio/x-smpte-302m:\nsubpicture/x-dvb:\napplication/x-teletext:\nmeta/x-klv:\n parsed: true\nmeta/x-id3:\n parsed: true\nmeta/x-st-2038:\n alignment: line\nvideo/x-vp9:\nimage/x-jpc:\n alignment: frame\n profile: [ 0, 49151 ]\nimage/x-jxsc:\n alignment: frame\n sampling: { (string)YCbCr-4:2:2, (string)YCbCr-4:4:4 }\n", "caps": "video/mpeg:\n parsed: true\n mpegversion: { (int)1, (int)2, (int)4 }\n systemstream: false\nvideo/x-dirac:\nimage/x-jpc:\n alignment: frame\nvideo/x-av1:\n stream-format: obu-stream\n alignment: frame\nvideo/x-h264:\n stream-format: byte-stream\n alignment: { (string)au, (string)nal }\nvideo/x-h265:\n stream-format: byte-stream\n alignment: { (string)au, (string)nal }\naudio/mpeg:\n parsed: true\n mpegversion: 1\naudio/mpeg:\n framed: true\n mpegversion: { (int)2, (int)4 }\n stream-format: { (string)adts, (string)raw }\naudio/x-lpcm:\n width: { (int)16, (int)20, (int)24 }\n rate: { (int)48000, (int)96000 }\n channels: [ 1, 8 ]\n dynamic_range: [ 0, 255 ]\n emphasis: { (boolean)false, (boolean)true }\n mute: { (boolean)false, (boolean)true }\naudio/x-ac3:\n framed: true\naudio/x-dts:\n framed: true\naudio/x-opus:\n channels: [ 1, 255 ]\naudio/x-smpte-302m:\nsubpicture/x-dvb:\napplication/x-teletext:\nmeta/x-klv:\n parsed: true\nmeta/x-id3:\n parsed: true\nmeta/x-st-2038:\n alignment: line\nvideo/x-vp9:\nimage/x-jpc:\n alignment: frame\n profile: [ 0, 49151 ]\nimage/x-jxsc:\n alignment: frame\n sampling: { (string)YCbCr-4:2:2, (string)YCbCr-4:4:4 }\n",
"direction": "sink", "direction": "sink",
"presence": "request", "presence": "request",
"type": "GstBaseTsMuxPad" "type": "GstBaseTsMuxPad"

View file

@ -242,5 +242,6 @@
#define DRF_ID_ID3 0x49443320 /* defined in SMPTE registration authority */ #define DRF_ID_ID3 0x49443320 /* defined in SMPTE registration authority */
#define DRF_ID_VANC 0x56414e43 /* defined in SMPTE ST-2038 */ #define DRF_ID_VANC 0x56414e43 /* defined in SMPTE ST-2038 */
#define DRF_ID_VP09 0x56503039 /* Custom GStreamer vp9 */ #define DRF_ID_VP09 0x56503039 /* Custom GStreamer vp9 */
#define DRF_ID_AV1G 0x41563147 /* Custom AV1 GStreamer mapping */
#endif /* __GST_MPEG_DESC_H__ */ #endif /* __GST_MPEG_DESC_H__ */

View file

@ -230,6 +230,7 @@ struct _TSDemuxStream
"video/x-h264,stream-format=(string)byte-stream;" \ "video/x-h264,stream-format=(string)byte-stream;" \
"video/x-h265,stream-format=(string)byte-stream;" \ "video/x-h265,stream-format=(string)byte-stream;" \
"video/x-vp9;" \ "video/x-vp9;" \
"video/x-av1,stream-format=(string)obu-stream,alignment=(string)frame;" \
"video/x-dirac;" \ "video/x-dirac;" \
"video/x-cavs;" \ "video/x-cavs;" \
"video/x-wmv," \ "video/x-wmv," \
@ -1498,6 +1499,24 @@ create_pad_for_stream (MpegTSBase * base, MpegTSBaseStream * bstream,
is_audio = TRUE; is_audio = TRUE;
caps = gst_caps_new_empty_simple ("audio/x-smpte-302m"); caps = gst_caps_new_empty_simple ("audio/x-smpte-302m");
break; break;
case DRF_ID_AV1G:
GST_DEBUG ("AV1");
is_video = TRUE;
caps =
gst_caps_new_simple ("video/x-av1", "stream-format",
G_TYPE_STRING, "obu-stream", "alignment", G_TYPE_STRING, "frame",
NULL);
desc = mpegts_get_descriptor_from_stream (bstream, 0x80);
if (desc != NULL) {
GstCaps *av1c_caps;
GstBuffer *buf = gst_buffer_new_wrapped (g_memdup2 (desc->data + 2,
desc->length), desc->length);
av1c_caps = gst_codec_utils_av1_create_caps_from_av1c (buf);
gst_caps_set_simple (av1c_caps, "codec_data", GST_TYPE_BUFFER, buf,
NULL);
caps = gst_caps_intersect (caps, av1c_caps);
}
break;
case DRF_ID_OPUS: case DRF_ID_OPUS:
desc = mpegts_get_descriptor_from_stream (bstream, desc = mpegts_get_descriptor_from_stream (bstream,
GST_MTS_DESC_DVB_EXTENSION); GST_MTS_DESC_DVB_EXTENSION);

View file

@ -633,6 +633,23 @@ gst_base_ts_mux_create_or_update_stream (GstBaseTsMux * mux,
GST_ERROR_OBJECT (mux, GST_ERROR_OBJECT (mux,
"VP9 requires enabling custom mapping which does not have a public specification"); "VP9 requires enabling custom mapping which does not have a public specification");
} }
} else if (strcmp (mt, "video/x-av1") == 0) {
if (mux->enable_custom_mappings) {
st = TSMUX_ST_PS_VIDEO_AV1;
if (!codec_data)
codec_data = gst_codec_utils_av1_create_av1c_from_caps (caps);
if (codec_data) {
GstMapInfo map;
if (gst_buffer_map (codec_data, &map, GST_MAP_READ)) {
pmt_descriptor =
gst_mpegts_descriptor_from_custom (0x80, map.data, map.size);
gst_buffer_unmap (codec_data, &map);
}
}
} else {
GST_ERROR_OBJECT (mux,
"AV1 requires enabling custom mapping which does not have a public specification yet");
}
} else if (strcmp (mt, "audio/mpeg") == 0) { } else if (strcmp (mt, "audio/mpeg") == 0) {
gint mpegversion; gint mpegversion;

View file

@ -103,6 +103,7 @@ static GstStaticPadTemplate gst_mpeg_ts_mux_sink_factory =
"systemstream = (boolean) false; " "systemstream = (boolean) false; "
"video/x-dirac;" "video/x-dirac;"
"image/x-jpc, alignment = (string) frame;" "image/x-jpc, alignment = (string) frame;"
"video/x-av1,stream-format=(string)obu-stream,alignment=(string)frame;"
"video/x-h264,stream-format=(string)byte-stream," "video/x-h264,stream-format=(string)byte-stream,"
"alignment=(string){au, nal}; " "alignment=(string){au, nal}; "
"video/x-h265,stream-format=(string)byte-stream," "video/x-h265,stream-format=(string)byte-stream,"

View file

@ -258,6 +258,13 @@ tsmux_stream_new (guint16 pid, guint stream_type, guint stream_number)
stream->stream_type = TSMUX_ST_PRIVATE_DATA; stream->stream_type = TSMUX_ST_PRIVATE_DATA;
stream->pi.flags |= TSMUX_PACKET_FLAG_PES_FULL_HEADER; stream->pi.flags |= TSMUX_PACKET_FLAG_PES_FULL_HEADER;
break; break;
case TSMUX_ST_PS_VIDEO_AV1:
/* FIXME: assign sequential extended IDs? */
stream->id = 0xBD;
stream->gst_stream_type = GST_STREAM_TYPE_VIDEO;
stream->stream_type = TSMUX_ST_PRIVATE_DATA;
stream->pi.flags |= TSMUX_PACKET_FLAG_PES_FULL_HEADER;
break;
default: default:
/* Might be a custom stream type implemented by a subclass */ /* Might be a custom stream type implemented by a subclass */
break; break;
@ -1045,6 +1052,13 @@ tsmux_stream_default_get_es_descrs (TsMuxStream * stream,
GST_DEBUG ("adding VP09 registration descriptor"); GST_DEBUG ("adding VP09 registration descriptor");
g_ptr_array_add (pmt_stream->descriptors, descriptor); g_ptr_array_add (pmt_stream->descriptors, descriptor);
} }
if (stream->internal_stream_type == TSMUX_ST_PS_VIDEO_AV1) {
/* AV1G is our custom non-bytestream format ! */
descriptor = gst_mpegts_descriptor_from_registration ("AV1G", NULL, 0);
g_ptr_array_add (pmt_stream->descriptors, descriptor);
if (stream->pmt_descriptor)
g_ptr_array_add (pmt_stream->descriptors, stream->pmt_descriptor);
}
default: default:
break; break;
} }

View file

@ -136,6 +136,7 @@ enum TsMuxStreamType {
TSMUX_ST_PS_ST_2038 = 0x91, /* only used internally */ TSMUX_ST_PS_ST_2038 = 0x91, /* only used internally */
TSMUX_ST_PS_S302M = 0x92, /* only used internally */ TSMUX_ST_PS_S302M = 0x92, /* only used internally */
TSMUX_ST_PS_VP9 = 0x93, /* only used internally */ TSMUX_ST_PS_VP9 = 0x93, /* only used internally */
TSMUX_ST_PS_VIDEO_AV1 = 0x94, /* only used internally */
TSMUX_ST_PS_DVD_SUBPICTURE = 0xff, TSMUX_ST_PS_DVD_SUBPICTURE = 0xff,
/* Non-standard definitions */ /* Non-standard definitions */