mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-10 17:35:59 +00:00
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:
parent
5480894dee
commit
19cea52dbd
7 changed files with 55 additions and 2 deletions
|
@ -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"
|
||||||
|
|
|
@ -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__ */
|
||||||
|
|
|
@ -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);
|
||||||
|
|
|
@ -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;
|
||||||
|
|
||||||
|
|
|
@ -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,"
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
|
@ -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 */
|
||||||
|
|
Loading…
Reference in a new issue