mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2024-11-23 10:11:08 +00:00
mpegtsdemux: add MPEG TS demuxer rewrite from Edward Hervey
with contributions from Miquel Angel Farre Guiu and Zaheer Abbas Merali
This commit is contained in:
parent
2bb835f69a
commit
22ecf13e42
16 changed files with 7644 additions and 0 deletions
|
@ -320,6 +320,7 @@ AG_GST_CHECK_PLUGIN(legacyresample)
|
||||||
AG_GST_CHECK_PLUGIN(librfb)
|
AG_GST_CHECK_PLUGIN(librfb)
|
||||||
AG_GST_CHECK_PLUGIN(liveadder)
|
AG_GST_CHECK_PLUGIN(liveadder)
|
||||||
AG_GST_CHECK_PLUGIN(mpegdemux)
|
AG_GST_CHECK_PLUGIN(mpegdemux)
|
||||||
|
AG_GST_CHECK_PLUGIN(mpegtsdemux)
|
||||||
AG_GST_CHECK_PLUGIN(mpegtsmux)
|
AG_GST_CHECK_PLUGIN(mpegtsmux)
|
||||||
AG_GST_CHECK_PLUGIN(mpegpsmux)
|
AG_GST_CHECK_PLUGIN(mpegpsmux)
|
||||||
AG_GST_CHECK_PLUGIN(mpeg4videoparse)
|
AG_GST_CHECK_PLUGIN(mpeg4videoparse)
|
||||||
|
@ -1748,6 +1749,7 @@ gst/legacyresample/Makefile
|
||||||
gst/librfb/Makefile
|
gst/librfb/Makefile
|
||||||
gst/liveadder/Makefile
|
gst/liveadder/Makefile
|
||||||
gst/mpegdemux/Makefile
|
gst/mpegdemux/Makefile
|
||||||
|
gst/mpegtsdemux/Makefile
|
||||||
gst/mpegtsmux/Makefile
|
gst/mpegtsmux/Makefile
|
||||||
gst/mpegtsmux/tsmux/Makefile
|
gst/mpegtsmux/tsmux/Makefile
|
||||||
gst/mpegpsmux/Makefile
|
gst/mpegpsmux/Makefile
|
||||||
|
|
|
@ -103,6 +103,7 @@ rm -rf $RPM_BUILD_ROOT
|
||||||
%{_libdir}/gstreamer-%{majorminor}/libgstmpegtsmux.so
|
%{_libdir}/gstreamer-%{majorminor}/libgstmpegtsmux.so
|
||||||
%{_libdir}/gstreamer-%{majorminor}/libgstscaletempoplugin.so
|
%{_libdir}/gstreamer-%{majorminor}/libgstscaletempoplugin.so
|
||||||
%{_libdir}/gstreamer-%{majorminor}/libgstmpegdemux.so
|
%{_libdir}/gstreamer-%{majorminor}/libgstmpegdemux.so
|
||||||
|
%{_libdir}/gstreamer-%{majorminor}/libgstmpegtsdemux.so
|
||||||
%{_libdir}/gstreamer-%{majorminor}/libgstjp2k.so
|
%{_libdir}/gstreamer-%{majorminor}/libgstjp2k.so
|
||||||
%{_libdir}/gstreamer-%{majorminor}/libgstapexsink.so
|
%{_libdir}/gstreamer-%{majorminor}/libgstapexsink.so
|
||||||
%{_libdir}/gstreamer-%{majorminor}/libgstqtmux.so
|
%{_libdir}/gstreamer-%{majorminor}/libgstqtmux.so
|
||||||
|
|
27
gst/mpegtsdemux/Makefile.am
Normal file
27
gst/mpegtsdemux/Makefile.am
Normal file
|
@ -0,0 +1,27 @@
|
||||||
|
plugin_LTLIBRARIES = libgstmpegtsdemux.la
|
||||||
|
|
||||||
|
libgstmpegtsdemux_la_SOURCES = \
|
||||||
|
gsttsdemux.c \
|
||||||
|
gstmpegdesc.c \
|
||||||
|
mpegtsbase.c \
|
||||||
|
mpegtspacketizer.c \
|
||||||
|
mpegtsparse.c \
|
||||||
|
tsdemux.c
|
||||||
|
|
||||||
|
libgstmpegtsdemux_la_CFLAGS = \
|
||||||
|
$(GST_PLUGINS_BAD_CFLAGS) $(GST_PLUGINS_BASE_CFLAGS) \
|
||||||
|
$(GST_BASE_CFLAGS) $(GST_CFLAGS)
|
||||||
|
libgstmpegtsdemux_la_LIBADD = \
|
||||||
|
$(GST_PLUGINS_BASE_LIBS) -lgsttag-$(GST_MAJORMINOR) \
|
||||||
|
$(GST_BASE_LIBS) $(GST_LIBS)
|
||||||
|
libgstmpegtsdemux_la_LDFLAGS = $(GST_PLUGIN_LDFLAGS)
|
||||||
|
libgstmpegtsdemux_la_LIBTOOLFLAGS = --tag=disable-static
|
||||||
|
|
||||||
|
noinst_HEADERS = \
|
||||||
|
gstmpegdefs.h \
|
||||||
|
gstmpegdesc.h \
|
||||||
|
mpegtsbase.h \
|
||||||
|
mpegtspacketizer.h \
|
||||||
|
mpegtsparse.c \
|
||||||
|
tsdemux.h
|
||||||
|
|
117
gst/mpegtsdemux/TODO
Normal file
117
gst/mpegtsdemux/TODO
Normal file
|
@ -0,0 +1,117 @@
|
||||||
|
mpegtsparse rebasing
|
||||||
|
--------------------
|
||||||
|
|
||||||
|
Rationale :
|
||||||
|
-----------
|
||||||
|
|
||||||
|
mpegtsparse code is more sane to handle and work with.
|
||||||
|
|
||||||
|
We need a modular demuxer
|
||||||
|
|
||||||
|
We need to avoid duplicating code regarding mpeg-ts in a gazillion
|
||||||
|
elements and allow easy creatiof new elements.
|
||||||
|
|
||||||
|
|
||||||
|
Battleplan :
|
||||||
|
------------
|
||||||
|
* Figure out code from mpegtsparse which would be also needed for a
|
||||||
|
mpeg-ts demuxer (ex: packet/psi/pcr parsing).
|
||||||
|
* Extract common code into a base mpegtsbase class.
|
||||||
|
* Refactor mpegtsparse to subclass that base class.
|
||||||
|
* Create a minimalistic demuxer that creates pads (based on PSI info)
|
||||||
|
and outputs ES packets (let's say mpeg audio and video to start with)
|
||||||
|
|
||||||
|
Potential subclasses :
|
||||||
|
----------------------
|
||||||
|
* MpegTSParse : Program splitter. Given an incoming multi-program
|
||||||
|
mpeg-ts stream, it can provide request pads for each program. Each
|
||||||
|
of those pads will contain the ts packets specific to that program.
|
||||||
|
|
||||||
|
* TSDemux : Program demuxer. Given an incoming single or multi-program
|
||||||
|
mpeg-ts stream, it will reconstruct the original Program Streams of
|
||||||
|
the selected program and output them on dynamically created pads.
|
||||||
|
|
||||||
|
* HDVSplitter : Given an incoming HDV mpeg-ts stream, it will locate
|
||||||
|
the beginning of new scenes and output a mpeg-ts stream with the
|
||||||
|
PAT/PMT/AUX packets properly ordered and marked with DISCONT, so
|
||||||
|
that the following pipeline will automatically cut up a tape dump
|
||||||
|
into individual scenes:
|
||||||
|
filesrc ! hdvsplit ! multifilesink next-file=discont
|
||||||
|
|
||||||
|
Code/Design common to a program-spliter and a demuxer :
|
||||||
|
-------------------------------------------------------
|
||||||
|
* Parsing TS packets
|
||||||
|
* Establishing PAT/PMT mapping
|
||||||
|
* Handling the notions of Programs/Streams
|
||||||
|
* Seeking ?
|
||||||
|
|
||||||
|
One proposal... would be to have the base class automatically create
|
||||||
|
all the structures (and relationships) for the following objects:
|
||||||
|
|
||||||
|
* Programs (from PAT/PMT, dunno if it could come from something
|
||||||
|
else)
|
||||||
|
* Program id
|
||||||
|
* Streams contained in that program (with links to them)
|
||||||
|
* Which stream contains the PCR
|
||||||
|
* Metadata ?
|
||||||
|
* Streams (ideally... in a table for fast access)
|
||||||
|
* We want to be able to have stream-type specific information
|
||||||
|
easily accessible also (like mpeg video specific data)
|
||||||
|
* Maybe some other info ???
|
||||||
|
|
||||||
|
The subclasses would then be able to make their own decision based
|
||||||
|
on those objects.
|
||||||
|
Maybe we could have some virtual methods that will be called when a
|
||||||
|
new program is detected, a new stream is added, etc...
|
||||||
|
|
||||||
|
It is the subclass who decides what's to do with a given packet once
|
||||||
|
it's been parsed.
|
||||||
|
tsparse : forward it as-is to the pad corresponding to the program
|
||||||
|
tsdemux : forward it to the proper PS parser
|
||||||
|
hdvsplit : ?
|
||||||
|
|
||||||
|
|
||||||
|
Ideas to be taken into account for a proper demuxer :
|
||||||
|
-----------------------------------------------------
|
||||||
|
* Push-based (with inacurrate seeking)
|
||||||
|
* Pull-based (with fast *AND* accurate seeking)
|
||||||
|
* Modular system to add stream-type specific helper parsing
|
||||||
|
* Doesn't have to be fully fledged, just enough to help any kind of
|
||||||
|
seeking and scanning code.
|
||||||
|
* ...
|
||||||
|
|
||||||
|
Problems to figure out :
|
||||||
|
------------------------
|
||||||
|
* clock
|
||||||
|
Needed for proper dvb playback. mpegtsdemux currently does internal
|
||||||
|
clock estimation... to provide a clock with PCR estimations.
|
||||||
|
A proper way to solve that would be to timestamp the buffers at the
|
||||||
|
source element using the system clock, and then adjusting the PCR
|
||||||
|
against those values. (i.e. doing the opposite of what's done in
|
||||||
|
mpegtsdemux, but it will be more accurate since the timestamping is
|
||||||
|
done at the source).
|
||||||
|
|
||||||
|
|
||||||
|
Bugs that need fixing :
|
||||||
|
-----------------------
|
||||||
|
* Perfomance : Creation/Destruction of buffers is slow
|
||||||
|
* => This is due to g_type_instance_create using a dogslow rwlock
|
||||||
|
which take up to 50% of gst_adapter_take_buffer()
|
||||||
|
=> Bugzilla #585375 (performance and contention problems)
|
||||||
|
|
||||||
|
Code structure:
|
||||||
|
|
||||||
|
MpegTSBase
|
||||||
|
+--- MpegTSParse
|
||||||
|
+--- TSDemux
|
||||||
|
|
||||||
|
|
||||||
|
Known limitations and problems :
|
||||||
|
--------------------------------
|
||||||
|
* mpegtspacketizer
|
||||||
|
* Assumes 188 bytes packets. It should support all modes.
|
||||||
|
* offset/timestamp of incoming buffers need to be carried on to the
|
||||||
|
sub-buffers in order for several demuxer features to work correctly.
|
||||||
|
* mpegtsparser
|
||||||
|
* SERIOUS room for improvement performance-wise (see callgrind)
|
||||||
|
|
234
gst/mpegtsdemux/gstmpegdefs.h
Normal file
234
gst/mpegtsdemux/gstmpegdefs.h
Normal file
|
@ -0,0 +1,234 @@
|
||||||
|
/*
|
||||||
|
* This library is licensed under 2 different licenses and you
|
||||||
|
* can choose to use it under the terms of either one of them. The
|
||||||
|
* two licenses are the MPL 1.1 and the LGPL.
|
||||||
|
*
|
||||||
|
* MPL:
|
||||||
|
*
|
||||||
|
* The contents of this file are subject to the Mozilla Public License
|
||||||
|
* Version 1.1 (the "License"); you may not use this file except in
|
||||||
|
* compliance with the License. You may obtain a copy of the License at
|
||||||
|
* http://www.mozilla.org/MPL/.
|
||||||
|
*
|
||||||
|
* Software distributed under the License is distributed on an "AS IS"
|
||||||
|
* basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See the
|
||||||
|
* License for the specific language governing rights and limitations
|
||||||
|
* under the License.
|
||||||
|
*
|
||||||
|
* LGPL:
|
||||||
|
*
|
||||||
|
* This library is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU Library General Public
|
||||||
|
* License as published by the Free Software Foundation; either
|
||||||
|
* version 2 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Library General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Library General Public
|
||||||
|
* License along with this library; if not, write to the
|
||||||
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||||
|
* Boston, MA 02111-1307, USA.
|
||||||
|
*
|
||||||
|
* The Original Code is Fluendo MPEG Demuxer plugin.
|
||||||
|
*
|
||||||
|
* The Initial Developer of the Original Code is Fluendo, S.L.
|
||||||
|
* Portions created by Fluendo, S.L. are Copyright (C) 2005
|
||||||
|
* Fluendo, S.L. All Rights Reserved.
|
||||||
|
*
|
||||||
|
* Contributor(s): Wim Taymans <wim@fluendo.com>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __GST_MPEG_DEFS_H__
|
||||||
|
#define __GST_MPEG_DEFS_H__
|
||||||
|
|
||||||
|
/*
|
||||||
|
* 1011 1100 program_stream_map
|
||||||
|
* 1011 1101 private_stream_1
|
||||||
|
* 1011 1110 padding_stream
|
||||||
|
* 1011 1111 private_stream_2
|
||||||
|
* 110x xxxx ISO/IEC 13818-3 or ISO/IEC 11172-3 audio stream number x xxxx
|
||||||
|
* 1110 xxxx ITU-T Rec. H.262 | ISO/IEC 13818-2 or ISO/IEC 11172-2 video stream number xxxx
|
||||||
|
* 1111 0000 ECM_stream
|
||||||
|
* 1111 0001 EMM_stream
|
||||||
|
* 1111 0010 ITU-T Rec. H.222.0 | ISO/IEC 13818-1 Annex A or ISO/IEC 13818-6_DSMCC_stream
|
||||||
|
* 1111 0011 ISO/IEC_13522_stream
|
||||||
|
* 1111 0100 ITU-T Rec. H.222.1 type A
|
||||||
|
* 1111 0101 ITU-T Rec. H.222.1 type B
|
||||||
|
* 1111 0110 ITU-T Rec. H.222.1 type C
|
||||||
|
* 1111 0111 ITU-T Rec. H.222.1 type D
|
||||||
|
* 1111 1000 ITU-T Rec. H.222.1 type E
|
||||||
|
* 1111 1001 ancillary_stream
|
||||||
|
* 1111 1010 E 1111 1110 reserved data stream
|
||||||
|
* 1111 1111 program_stream_directory
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define ID_PS_END_CODE 0x000001B9
|
||||||
|
#define ID_PS_PACK_START_CODE 0x000001BA
|
||||||
|
#define ID_PS_SYSTEM_HEADER_START_CODE 0x000001BB
|
||||||
|
#define ID_PS_PROGRAM_STREAM_MAP 0x000001BC
|
||||||
|
#define ID_PRIVATE_STREAM_1 0x000001BD
|
||||||
|
#define ID_PADDING_STREAM 0x000001BE
|
||||||
|
#define ID_PRIVATE_STREAM_2 0x000001BF
|
||||||
|
#define ID_ISO_IEC_MPEG12_AUDIO_STREAM_0 0x000001C0
|
||||||
|
#define ID_ISO_IEC_MPEG12_AUDIO_STREAM_32 0x000001DF
|
||||||
|
#define ID_ISO_IEC_MPEG12_VIDEO_STREAM_0 0x000001E0
|
||||||
|
#define ID_ISO_IEC_MPEG12_VIDEO_STREAM_16 0x000001EF
|
||||||
|
#define ID_ECM_STREAM 0x000001F0
|
||||||
|
#define ID_EMM_STREAM 0x000001F1
|
||||||
|
#define ID_DSMCC_STREAM 0x000001F2
|
||||||
|
#define ID_ISO_IEC_13522_STREAM 0x000001F3
|
||||||
|
#define ID_ITU_TREC_H222_TYPE_A_STREAM 0x000001F4
|
||||||
|
#define ID_ITU_TREC_H222_TYPE_B_STREAM 0x000001F5
|
||||||
|
#define ID_ITU_TREC_H222_TYPE_C_STREAM 0x000001F6
|
||||||
|
#define ID_ITU_TREC_H222_TYPE_D_STREAM 0x000001F7
|
||||||
|
#define ID_ITU_TREC_H222_TYPE_E_STREAM 0x000001F8
|
||||||
|
#define ID_ANCILLARY_STREAM 0x000001F9
|
||||||
|
#define ID_RESERVED_STREAM_1 0x000001FA
|
||||||
|
#define ID_RESERVED_STREAM_2 0x000001FB
|
||||||
|
#define ID_EXTENDED_METADATA 0x000001FC
|
||||||
|
#define ID_EXTENDED_STREAM_ID 0x000001FD
|
||||||
|
#define ID_RESERVED_STREAM_3 0x000001FE
|
||||||
|
#define ID_PROGRAM_STREAM_DIRECTORY 0x000001FF
|
||||||
|
|
||||||
|
#define PACKET_VIDEO_START_CODE 0x000001E0
|
||||||
|
#define PACKET_AUDIO_START_CODE 0x000001C0
|
||||||
|
#define PICTURE_START_CODE 0x00000100
|
||||||
|
#define USER_DATA_START_CODE 0x000001B2
|
||||||
|
#define SEQUENCE_HEADER_CODE 0x000001B3
|
||||||
|
#define SEQUENCE_ERROR_CODE 0x000001B4
|
||||||
|
#define EXTENSION_START_CODE 0x000001B5
|
||||||
|
#define SEQUENCE_END_CODE 0x000001B7
|
||||||
|
#define GROUP_START_CODE 0x000001B8
|
||||||
|
|
||||||
|
#define AC3_SYNC_WORD 0x0b770000
|
||||||
|
|
||||||
|
#define MPEG_TS_SYNC_BYTE 0x00000047
|
||||||
|
|
||||||
|
#define PID_PROGRAM_ASSOCIATION_TABLE 0x0000
|
||||||
|
#define PID_CONDITIONAL_ACCESS_TABLE 0x0001
|
||||||
|
#define PID_RESERVED_FIRST 0x0002
|
||||||
|
#define PID_RESERVED_LAST 0x0010
|
||||||
|
#define PID_NULL_PACKET 0x1FFF
|
||||||
|
|
||||||
|
#define PID_TYPE_UNKNOWN 0
|
||||||
|
#define PID_TYPE_RESERVED 1
|
||||||
|
#define PID_TYPE_PROGRAM_ASSOCIATION 2
|
||||||
|
#define PID_TYPE_CONDITIONAL_ACCESS 3
|
||||||
|
#define PID_TYPE_PROGRAM_MAP 4
|
||||||
|
#define PID_TYPE_ELEMENTARY 5
|
||||||
|
#define PID_TYPE_NULL_PACKET 6
|
||||||
|
#define PID_TYPE_PRIVATE_SECTION 7
|
||||||
|
|
||||||
|
/* Stream type assignments
|
||||||
|
*
|
||||||
|
* 0x00 ITU-T | ISO/IEC Reserved
|
||||||
|
* 0x01 ISO/IEC 11172 Video
|
||||||
|
* 0x02 ITU-T Rec. H.262 | ISO/IEC 13818-2 Video or
|
||||||
|
* ISO/IEC 11172-2 constrained parameter video
|
||||||
|
* stream
|
||||||
|
* 0x03 ISO/IEC 11172 Audio
|
||||||
|
* 0x04 ISO/IEC 13818-3 Audio
|
||||||
|
* 0x05 ITU-T Rec. H.222.0 | ISO/IEC 13818-1
|
||||||
|
* private_sections
|
||||||
|
* 0x06 ITU-T Rec. H.222.0 | ISO/IEC 13818-1 PES
|
||||||
|
* packets containing private data
|
||||||
|
* 0x07 ISO/IEC 13522 MHEG
|
||||||
|
* 0x08 ITU-T Rec. H.222.0 | ISO/IEC 13818-1 Annex A
|
||||||
|
* DSM CC
|
||||||
|
* 0x09 ITU-T Rec. H.222.1
|
||||||
|
* 0x0A ISO/IEC 13818-6 type A
|
||||||
|
* 0x0B ISO/IEC 13818-6 type B
|
||||||
|
* 0x0C ISO/IEC 13818-6 type C
|
||||||
|
* 0x0D ISO/IEC 13818-6 type D
|
||||||
|
* 0x0E ISO/IEC 13818-1 auxiliary
|
||||||
|
* 0x0F-0x7F ITU-T Rec. H.222.0 | ISO/IEC 13818-1 Reserved
|
||||||
|
* 0x80-0xFF User Private
|
||||||
|
*/
|
||||||
|
#define ST_RESERVED 0x00
|
||||||
|
#define ST_VIDEO_MPEG1 0x01
|
||||||
|
#define ST_VIDEO_MPEG2 0x02
|
||||||
|
#define ST_AUDIO_MPEG1 0x03
|
||||||
|
#define ST_AUDIO_MPEG2 0x04
|
||||||
|
#define ST_PRIVATE_SECTIONS 0x05
|
||||||
|
#define ST_PRIVATE_DATA 0x06
|
||||||
|
#define ST_MHEG 0x07
|
||||||
|
#define ST_DSMCC 0x08
|
||||||
|
#define ST_H222_1 0x09
|
||||||
|
|
||||||
|
/* later extensions */
|
||||||
|
#define ST_AUDIO_AAC 0x0f
|
||||||
|
#define ST_VIDEO_MPEG4 0x10
|
||||||
|
#define ST_VIDEO_H264 0x1b
|
||||||
|
|
||||||
|
/* Un-official Dirac extension */
|
||||||
|
#define ST_VIDEO_DIRAC 0xd1
|
||||||
|
|
||||||
|
/* private stream types */
|
||||||
|
#define ST_PS_AUDIO_AC3 0x81
|
||||||
|
#define ST_PS_AUDIO_DTS 0x8a
|
||||||
|
#define ST_PS_AUDIO_LPCM 0x8b
|
||||||
|
#define ST_PS_DVD_SUBPICTURE 0xff
|
||||||
|
/* Blu-ray related */
|
||||||
|
#define ST_BD_AUDIO_LPCM 0x80
|
||||||
|
#define ST_BD_AUDIO_AC3 0x81
|
||||||
|
#define ST_BD_AUDIO_DTS 0x82
|
||||||
|
#define ST_BD_AUDIO_AC3_TRUE_HD 0x83
|
||||||
|
#define ST_BD_AUDIO_AC3_PLUS 0x84
|
||||||
|
#define ST_BD_AUDIO_DTS_HD 0x85
|
||||||
|
#define ST_BD_AUDIO_DTS_HD_MASTER_AUDIO 0x86
|
||||||
|
#define ST_BD_AUDIO_EAC3 0x87
|
||||||
|
#define ST_BD_PGS_SUBPICTURE 0x90
|
||||||
|
#define ST_BD_IGS 0x91
|
||||||
|
#define ST_BD_SUBTITLE 0x92
|
||||||
|
#define ST_BD_SECONDARY_AC3_PLUS 0xa1
|
||||||
|
#define ST_BD_SECONDARY_DTS_HD 0xa2
|
||||||
|
|
||||||
|
/* defined for VC1 extension in RP227 */
|
||||||
|
#define ST_PRIVATE_EA 0xea
|
||||||
|
|
||||||
|
/* HDV AUX stream mapping
|
||||||
|
* 0xA0 ISO/IEC 61834-11
|
||||||
|
* 0xA1 ISO/IEC 61834-11
|
||||||
|
*/
|
||||||
|
#define ST_HDV_AUX_A 0xa0
|
||||||
|
#define ST_HDV_AUX_V 0xa1
|
||||||
|
|
||||||
|
/* Un-official time-code stream */
|
||||||
|
#define ST_PS_TIMECODE 0xd2
|
||||||
|
|
||||||
|
/* Internal stream types >= 0x100 */
|
||||||
|
#define ST_GST_AUDIO_RAWA52 0x181
|
||||||
|
/* Used when we don't yet know which stream type it will be in a PS stream */
|
||||||
|
#define ST_GST_VIDEO_MPEG1_OR_2 0x102
|
||||||
|
|
||||||
|
#define CLOCK_BASE 9LL
|
||||||
|
#define CLOCK_FREQ (CLOCK_BASE * 10000)
|
||||||
|
|
||||||
|
#define PCRTIME_TO_GSTTIME(time) (gst_util_uint64_scale ((time), \
|
||||||
|
GST_MSECOND/10, 300 * CLOCK_BASE))
|
||||||
|
#define MPEGTIME_TO_GSTTIME(time) (gst_util_uint64_scale ((time), \
|
||||||
|
GST_MSECOND/10, CLOCK_BASE))
|
||||||
|
#define GSTTIME_TO_MPEGTIME(time) (gst_util_uint64_scale ((time), \
|
||||||
|
CLOCK_BASE, GST_MSECOND/10))
|
||||||
|
|
||||||
|
#define MPEG_MUX_RATE_MULT 50
|
||||||
|
|
||||||
|
/* sync:4 == 00xx ! pts:3 ! 1 ! pts:15 ! 1 | pts:15 ! 1 */
|
||||||
|
#define READ_TS(data, target, lost_sync_label) \
|
||||||
|
if ((*data & 0x01) != 0x01) goto lost_sync_label; \
|
||||||
|
target = ((guint64) (*data++ & 0x0E)) << 29; \
|
||||||
|
target |= ((guint64) (*data++ )) << 22; \
|
||||||
|
if ((*data & 0x01) != 0x01) goto lost_sync_label; \
|
||||||
|
target |= ((guint64) (*data++ & 0xFE)) << 14; \
|
||||||
|
target |= ((guint64) (*data++ )) << 7; \
|
||||||
|
if ((*data & 0x01) != 0x01) goto lost_sync_label; \
|
||||||
|
target |= ((guint64) (*data++ & 0xFE)) >> 1;
|
||||||
|
|
||||||
|
/* some extra GstFlowReturn values used internally */
|
||||||
|
#define GST_FLOW_NEED_MORE_DATA GST_FLOW_CUSTOM_SUCCESS
|
||||||
|
#define GST_FLOW_LOST_SYNC GST_FLOW_CUSTOM_SUCCESS_1
|
||||||
|
|
||||||
|
#endif /* __GST_MPEG_DEFS_H__ */
|
209
gst/mpegtsdemux/gstmpegdesc.c
Normal file
209
gst/mpegtsdemux/gstmpegdesc.c
Normal file
|
@ -0,0 +1,209 @@
|
||||||
|
/*
|
||||||
|
* The contents of this file are subject to the Mozilla Public License
|
||||||
|
* Version 1.1 (the "License"); you may not use this file except in
|
||||||
|
* compliance with the License. You may obtain a copy of the License at
|
||||||
|
* http://www.mozilla.org/MPL/.
|
||||||
|
*
|
||||||
|
* Software distributed under the License is distributed on an "AS IS"
|
||||||
|
* basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See the
|
||||||
|
* License for the specific language governing rights and limitations
|
||||||
|
* under the License.
|
||||||
|
*
|
||||||
|
* The Original Code is Fluendo MPEG Demuxer plugin.
|
||||||
|
*
|
||||||
|
* The Initial Developer of the Original Code is Fluendo, S.L.
|
||||||
|
* Portions created by Fluendo, S.L. are Copyright (C) 2005
|
||||||
|
* Fluendo, S.L. All Rights Reserved.
|
||||||
|
*
|
||||||
|
* Contributor(s): Wim Taymans <wim@fluendo.com>
|
||||||
|
*
|
||||||
|
* Alternatively, the contents of this file may be used under the terms of
|
||||||
|
* the GNU Lesser General Public License Version 2 or later (the "LGPL"),
|
||||||
|
* in which case the provisions of the LGPL are applicable instead
|
||||||
|
* of those above. If you wish to allow use of your version of this file only
|
||||||
|
* under the terms of the LGPL, and not to allow others to
|
||||||
|
* use your version of this file under the terms of the MPL, indicate your
|
||||||
|
* decision by deleting the provisions above and replace them with the notice
|
||||||
|
* and other provisions required by the LGPL. If you do not delete
|
||||||
|
* the provisions above, a recipient may use your version of this file under
|
||||||
|
* the terms of the MPL or the LGPL.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
#include <gst/gst.h>
|
||||||
|
|
||||||
|
#include "gstmpegdesc.h"
|
||||||
|
|
||||||
|
GST_DEBUG_CATEGORY (gstmpegtsdesc_debug);
|
||||||
|
#define GST_CAT_DEFAULT (gstmpegtsdesc_debug)
|
||||||
|
|
||||||
|
void
|
||||||
|
gst_mpeg_descriptor_free (GstMPEGDescriptor * desc)
|
||||||
|
{
|
||||||
|
g_return_if_fail (desc != NULL);
|
||||||
|
|
||||||
|
g_free (desc);
|
||||||
|
}
|
||||||
|
|
||||||
|
static guint
|
||||||
|
gst_mpeg_descriptor_parse_1 (guint8 * data, guint size)
|
||||||
|
{
|
||||||
|
guint8 tag;
|
||||||
|
guint8 length;
|
||||||
|
|
||||||
|
/* need at least 2 bytes for tag and length */
|
||||||
|
if (size < 2)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
tag = *data++;
|
||||||
|
length = *data++;
|
||||||
|
size -= 2;
|
||||||
|
|
||||||
|
GST_DEBUG ("tag: 0x%02x, length: %d", tag, length);
|
||||||
|
|
||||||
|
if (length > size)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
GST_MEMDUMP ("tag contents:", data, length);
|
||||||
|
|
||||||
|
return length + 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
GstMPEGDescriptor *
|
||||||
|
gst_mpeg_descriptor_parse (guint8 * data, guint size)
|
||||||
|
{
|
||||||
|
guint8 *current;
|
||||||
|
guint consumed, total, n_desc;
|
||||||
|
GstMPEGDescriptor *result;
|
||||||
|
|
||||||
|
g_return_val_if_fail (data != NULL, NULL);
|
||||||
|
|
||||||
|
current = data;
|
||||||
|
total = 0;
|
||||||
|
n_desc = 0;
|
||||||
|
|
||||||
|
do {
|
||||||
|
consumed = gst_mpeg_descriptor_parse_1 (current, size);
|
||||||
|
|
||||||
|
if (consumed > 0) {
|
||||||
|
current += consumed;
|
||||||
|
total += consumed;
|
||||||
|
size -= consumed;
|
||||||
|
n_desc++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
while (consumed > 0);
|
||||||
|
|
||||||
|
GST_DEBUG ("parsed %d descriptors", n_desc);
|
||||||
|
|
||||||
|
if (total == 0)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
result = g_malloc (sizeof (GstMPEGDescriptor) + total);
|
||||||
|
result->n_desc = n_desc;
|
||||||
|
result->data_length = total;
|
||||||
|
result->data = ((guint8 *) result) + sizeof (GstMPEGDescriptor);
|
||||||
|
|
||||||
|
memcpy (result->data, data, total);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
guint
|
||||||
|
gst_mpeg_descriptor_n_desc (GstMPEGDescriptor * desc)
|
||||||
|
{
|
||||||
|
g_return_val_if_fail (desc != NULL, 0);
|
||||||
|
|
||||||
|
return desc->n_desc;
|
||||||
|
}
|
||||||
|
|
||||||
|
guint8 *
|
||||||
|
gst_mpeg_descriptor_find (GstMPEGDescriptor * desc, gint tag)
|
||||||
|
{
|
||||||
|
guint8 length;
|
||||||
|
guint8 *current;
|
||||||
|
guint size;
|
||||||
|
|
||||||
|
g_return_val_if_fail (desc != NULL, NULL);
|
||||||
|
|
||||||
|
current = desc->data;
|
||||||
|
length = desc->data_length;
|
||||||
|
|
||||||
|
while (length > 0) {
|
||||||
|
if (DESC_TAG (current) == tag)
|
||||||
|
return current;
|
||||||
|
|
||||||
|
size = DESC_LENGTH (current) + 2;
|
||||||
|
|
||||||
|
current += size;
|
||||||
|
length -= size;
|
||||||
|
}
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* array needs freeing afterwards */
|
||||||
|
GArray *
|
||||||
|
gst_mpeg_descriptor_find_all (GstMPEGDescriptor * desc, gint tag)
|
||||||
|
{
|
||||||
|
GArray *all;
|
||||||
|
|
||||||
|
guint8 length;
|
||||||
|
guint8 *current;
|
||||||
|
guint size;
|
||||||
|
|
||||||
|
g_return_val_if_fail (desc != NULL, NULL);
|
||||||
|
all = g_array_new (TRUE, TRUE, sizeof (guint8 *));
|
||||||
|
|
||||||
|
current = desc->data;
|
||||||
|
length = desc->data_length;
|
||||||
|
|
||||||
|
while (length > 0) {
|
||||||
|
if (DESC_TAG (current) == tag)
|
||||||
|
g_array_append_val (all, current);
|
||||||
|
size = DESC_LENGTH (current) + 2;
|
||||||
|
|
||||||
|
current += size;
|
||||||
|
length -= size;
|
||||||
|
}
|
||||||
|
|
||||||
|
GST_DEBUG ("found tag 0x%02x %d times", tag, all->len);
|
||||||
|
|
||||||
|
return all;
|
||||||
|
}
|
||||||
|
|
||||||
|
guint8 *
|
||||||
|
gst_mpeg_descriptor_nth (GstMPEGDescriptor * desc, guint i)
|
||||||
|
{
|
||||||
|
guint8 length;
|
||||||
|
guint8 *current;
|
||||||
|
guint size;
|
||||||
|
|
||||||
|
g_return_val_if_fail (desc != NULL, NULL);
|
||||||
|
|
||||||
|
if (i > desc->n_desc)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
current = desc->data;
|
||||||
|
length = desc->data_length;
|
||||||
|
|
||||||
|
while (length > 0) {
|
||||||
|
if (i == 0)
|
||||||
|
return current;
|
||||||
|
|
||||||
|
size = DESC_LENGTH (current) + 2;
|
||||||
|
|
||||||
|
current += size;
|
||||||
|
length -= size;
|
||||||
|
i--;
|
||||||
|
|
||||||
|
}
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
gst_mpegtsdesc_init_debug (void)
|
||||||
|
{
|
||||||
|
GST_DEBUG_CATEGORY_INIT (gstmpegtsdesc_debug, "mpegtsdesc", 0,
|
||||||
|
"MPEG transport stream parser (descriptor)");
|
||||||
|
}
|
342
gst/mpegtsdemux/gstmpegdesc.h
Normal file
342
gst/mpegtsdemux/gstmpegdesc.h
Normal file
|
@ -0,0 +1,342 @@
|
||||||
|
/*
|
||||||
|
* The contents of this file are subject to the Mozilla Public License
|
||||||
|
* Version 1.1 (the "License"); you may not use this file except in
|
||||||
|
* compliance with the License. You may obtain a copy of the License at
|
||||||
|
* http://www.mozilla.org/MPL/.
|
||||||
|
*
|
||||||
|
* Software distributed under the License is distributed on an "AS IS"
|
||||||
|
* basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See the
|
||||||
|
* License for the specific language governing rights and limitations
|
||||||
|
* under the License.
|
||||||
|
*
|
||||||
|
* The Original Code is Fluendo MPEG Demuxer plugin.
|
||||||
|
*
|
||||||
|
* The Initial Developer of the Original Code is Fluendo, S.L.
|
||||||
|
* Portions created by Fluendo, S.L. are Copyright (C) 2005
|
||||||
|
* Fluendo, S.L. All Rights Reserved.
|
||||||
|
*
|
||||||
|
* Contributor(s): Wim Taymans <wim@fluendo.com>
|
||||||
|
*
|
||||||
|
* Alternatively, the contents of this file may be used under the terms of
|
||||||
|
* the GNU Lesser General Public License Version 2 or later (the "LGPL"),
|
||||||
|
* in which case the provisions of the LGPL are applicable instead
|
||||||
|
* of those above. If you wish to allow use of your version of this file only
|
||||||
|
* under the terms of the LGPL, and not to allow others to
|
||||||
|
* use your version of this file under the terms of the MPL, indicate your
|
||||||
|
* decision by deleting the provisions above and replace them with the notice
|
||||||
|
* and other provisions required by the LGPL. If you do not delete
|
||||||
|
* the provisions above, a recipient may use your version of this file under
|
||||||
|
* the terms of the MPL or the LGPL.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __GST_MPEG_DESC_H__
|
||||||
|
#define __GST_MPEG_DESC_H__
|
||||||
|
|
||||||
|
#include <glib.h>
|
||||||
|
/*
|
||||||
|
* descriptor_tag TS PS Identification
|
||||||
|
* 0 n/a n/a Reserved
|
||||||
|
* 1 n/a n/a Reserved
|
||||||
|
* 2 X X video_stream_descriptor
|
||||||
|
* 3 X X audio_stream_descriptor
|
||||||
|
* 4 X X hierarchy_descriptor
|
||||||
|
* 5 X X registration_descriptor
|
||||||
|
* 6 X X data_stream_alignment_descriptor
|
||||||
|
* 7 X X target_background_grid_descriptor
|
||||||
|
* 8 X X video_window_descriptor
|
||||||
|
* 9 X X CA_descriptor
|
||||||
|
* 10 X X ISO_639_language_descriptor
|
||||||
|
* 11 X X system_clock_descriptor
|
||||||
|
* 12 X X multiplex_buffer_utilization_descriptor
|
||||||
|
* 13 X X copyright_descriptor
|
||||||
|
* 14 X maximum bitrate descriptor
|
||||||
|
* 15 X X private data indicator descriptor
|
||||||
|
* 16 X X smoothing buffer descriptor
|
||||||
|
* 17 X STD_descriptor
|
||||||
|
* 18 X X IBP descriptor
|
||||||
|
* 19-63 n/a n/a ITU-T Rec. H.222.0 | ISO/IEC 13818-1 Reserved
|
||||||
|
* 64-255 n/a n/a User Private
|
||||||
|
*/
|
||||||
|
#define DESC_VIDEO_STREAM 2
|
||||||
|
#define DESC_AUDIO_STREAM 3
|
||||||
|
#define DESC_HIERARCHY 4
|
||||||
|
#define DESC_REGISTRATION 5
|
||||||
|
#define DESC_DATA_STREAM_ALIGNMENT 6
|
||||||
|
#define DESC_TARGET_BACKGROUND_GRID 7
|
||||||
|
#define DESC_VIDEO_WINDOW 8
|
||||||
|
#define DESC_CA 9
|
||||||
|
#define DESC_ISO_639_LANGUAGE 10
|
||||||
|
#define DESC_SYSTEM_CLOCK 11
|
||||||
|
#define DESC_MULTIPLEX_BUFFER_UTILISATION 12
|
||||||
|
#define DESC_COPYRIGHT 13
|
||||||
|
#define DESC_MAXIMUM_BITRATE 14
|
||||||
|
#define DESC_PRIVATE_DATA_INDICATOR 15
|
||||||
|
#define DESC_SMOOTHING_BUFFER 16
|
||||||
|
#define DESC_STD 17
|
||||||
|
#define DESC_IBP 18
|
||||||
|
|
||||||
|
#define DESC_DIRAC_TC_PRIVATE 0xAC
|
||||||
|
|
||||||
|
/* DVB tags */
|
||||||
|
#define DESC_DVB_CAROUSEL_IDENTIFIER 0x13
|
||||||
|
#define DESC_DVB_NETWORK_NAME 0x40
|
||||||
|
#define DESC_DVB_SERVICE_LIST 0x41
|
||||||
|
#define DESC_DVB_STUFFING 0x42
|
||||||
|
#define DESC_DVB_SATELLITE_DELIVERY_SYSTEM 0x43
|
||||||
|
#define DESC_DVB_CABLE_DELIVERY_SYSTEM 0x44
|
||||||
|
#define DESC_DVB_VBI_DATA 0x45
|
||||||
|
#define DESC_DVB_VBI_TELETEXT 0x46
|
||||||
|
#define DESC_DVB_BOUQUET_NAME 0x47
|
||||||
|
#define DESC_DVB_SERVICE 0x48
|
||||||
|
#define DESC_DVB_COUNTRY_AVAILABILITY 0x49
|
||||||
|
#define DESC_DVB_LINKAGE 0x4A
|
||||||
|
#define DESC_DVB_NVOD_REFERENCE 0x4B
|
||||||
|
#define DESC_DVB_TIME_SHIFTED_SERVICE 0x4C
|
||||||
|
#define DESC_DVB_SHORT_EVENT 0x4D
|
||||||
|
#define DESC_DVB_EXTENDED_EVENT 0x4E
|
||||||
|
#define DESC_DVB_TIME_SHIFTED_EVENT 0x4F
|
||||||
|
#define DESC_DVB_COMPONENT 0x50
|
||||||
|
#define DESC_DVB_MOSAIC 0x51
|
||||||
|
#define DESC_DVB_STREAM_IDENTIFIER 0x52
|
||||||
|
#define DESC_DVB_CA_IDENTIFIER 0x53
|
||||||
|
#define DESC_DVB_CONTENT 0x54
|
||||||
|
#define DESC_DVB_PARENTAL_RATING 0x55
|
||||||
|
#define DESC_DVB_TELETEXT 0x56
|
||||||
|
#define DESC_DVB_TELEPHONE 0x57
|
||||||
|
#define DESC_DVB_LOCAL_TIME_OFFSET 0x58
|
||||||
|
#define DESC_DVB_SUBTITLING 0x59
|
||||||
|
#define DESC_DVB_TERRESTRIAL_DELIVERY_SYSTEM 0x5A
|
||||||
|
#define DESC_DVB_MULTILINGUAL_NETWORK_NAME 0x5B
|
||||||
|
#define DESC_DVB_MULTILINGUAL_BOUQUET_NAME 0x5C
|
||||||
|
#define DESC_DVB_MULTILINGUAL_SERVICE_NAME 0x5D
|
||||||
|
#define DESC_DVB_MULTILINGUAL_COMPONENT 0x5E
|
||||||
|
#define DESC_DVB_PRIVATE_DATA 0x5F
|
||||||
|
#define DESC_DVB_SERVICE_MOVE 0x60
|
||||||
|
#define DESC_DVB_SHORT_SMOOTHING_BUFFER 0x61
|
||||||
|
#define DESC_DVB_FREQUENCY_LIST 0x62
|
||||||
|
#define DESC_DVB_PARTIAL_TRANSPORT_STREAM 0x63
|
||||||
|
#define DESC_DVB_DATA_BROADCAST 0x64
|
||||||
|
#define DESC_DVB_SCRAMBLING 0x65
|
||||||
|
#define DESC_DVB_DATA_BROADCAST_ID 0x66
|
||||||
|
#define DESC_DVB_TRANSPORT_STREAM 0x67
|
||||||
|
#define DESC_DVB_DSNG 0x68
|
||||||
|
#define DESC_DVB_PDC 0x69
|
||||||
|
#define DESC_DVB_AC3 0x6A
|
||||||
|
#define DESC_DVB_ANCILLARY_DATA 0x6B
|
||||||
|
#define DESC_DVB_CELL_LIST 0x6C
|
||||||
|
#define DESC_DVB_CELL_FREQUENCY_LINK 0x6D
|
||||||
|
#define DESC_DVB_ANNOUNCEMENT_SUPPORT 0x6E
|
||||||
|
#define DESC_DVB_APPLICATION_SIGNALLING 0x6F
|
||||||
|
#define DESC_DVB_ADAPTATION_FIELD_DATA 0x70
|
||||||
|
#define DESC_DVB_SERVICE_IDENTIFIER 0x71
|
||||||
|
#define DESC_DVB_SERVICE_AVAILABILITY 0x72
|
||||||
|
#define DESC_DVB_DEFAULT_AUTHORITY 0x73
|
||||||
|
#define DESC_DVB_RELATED_CONTENT 0x74
|
||||||
|
#define DESC_DVB_TVA_ID 0x75
|
||||||
|
#define DESC_DVB_CONTENT_IDENTIFIER 0x76
|
||||||
|
#define DESC_DVB_TIMESLICE_FEC_IDENTIFIER 0x77
|
||||||
|
#define DESC_DVB_ECM_REPETITION_RATE 0x78
|
||||||
|
#define DESC_DVB_S2_SATELLITE_DELIVERY_SYSTEM 0x79
|
||||||
|
#define DESC_DVB_ENHANCED_AC3 0x7A
|
||||||
|
#define DESC_DVB_DTS 0x7B
|
||||||
|
#define DESC_DVB_AAC 0x7C
|
||||||
|
/* 0x7D and 0x7E are reserved for future use */
|
||||||
|
#define DESC_DVB_EXTENSION 0x7F
|
||||||
|
/* 0x80 - 0xFE are user defined */
|
||||||
|
#define DESC_DTG_LOGICAL_CHANNEL 0x83 /* from DTG D-Book */
|
||||||
|
/* 0xFF is forbidden */
|
||||||
|
|
||||||
|
/* common for all descriptors */
|
||||||
|
#define DESC_TAG(desc) (desc[0])
|
||||||
|
#define DESC_LENGTH(desc) (desc[1])
|
||||||
|
|
||||||
|
/* video_stream_descriptor */
|
||||||
|
#define DESC_VIDEO_STREAM_multiple_framerate_flag(desc) (((desc)[2] & 0x80) == 0x80)
|
||||||
|
#define DESC_VIDEO_STREAM_frame_rate_code(desc) (((desc)[2] & 0x38) >> 3)
|
||||||
|
#define DESC_VIDEO_STREAM_MPEG_1_only_flag(desc) (((desc)[2] & 0x04) == 0x04)
|
||||||
|
#define DESC_VIDEO_STREAM_constrained_parameter_flag(desc) (((desc)[2] & 0x02) == 0x02)
|
||||||
|
#define DESC_VIDEO_STREAM_still_picture_flag(desc) (((desc)[2] & 0x01) == 0x01)
|
||||||
|
/* if (MPEG_1_only_flag == 1) */
|
||||||
|
#define DESC_VIDEO_STREAM_profile_and_level_indication(desc) ((desc)[3])
|
||||||
|
#define DESC_VIDEO_STREAM_chroma_format(desc) (((desc)[4] & 0xc0) >> 6)
|
||||||
|
#define DESC_VIDEO_STREAM_frame_rate_extension_flag(desc) (((desc)[4] & 0x20) == 0x20)
|
||||||
|
|
||||||
|
/* audio_stream_descriptor */
|
||||||
|
#define DESC_AUDIO_STREAM_free_format_flag(desc) (((desc)[2] & 0x80) == 0x80)
|
||||||
|
#define DESC_AUDIO_STREAM_ID(desc) (((desc)[2] & 0x40) == 0x40)
|
||||||
|
#define DESC_AUDIO_STREAM_layer(desc) (((desc)[2] & 0x30) >> 4)
|
||||||
|
#define DESC_AUDIO_STREAM_variable_rate_audio_indicator(desc) (((desc)[2] & 0x08) == 0x08)
|
||||||
|
|
||||||
|
/* hierarchy_descriptor */
|
||||||
|
#define DESC_HIERARCHY_hierarchy_type(desc) (((desc)[2] & 0x0f))
|
||||||
|
#define DESC_HIERARCHY_hierarchy_layer_index(desc) (((desc)[3] & 0x3f))
|
||||||
|
#define DESC_HIERARCHY_hierarchy_embedded_layer_index(desc) (((desc)[4] & 0x3f))
|
||||||
|
#define DESC_HIERARCHY_hierarchy_channel(desc) (((desc)[5] & 0x3f))
|
||||||
|
|
||||||
|
/* registration_descriptor */
|
||||||
|
#define DESC_REGISTRATION_format_identifier(desc) (GST_READ_UINT32_BE ((desc)+2))
|
||||||
|
#define DESC_REGISTRATION_additional_ident_info_len(desc) ((desc)[1] - 4)
|
||||||
|
#define DESC_REGISTRATION_additional_ident_info(desc) (&(desc)[6])
|
||||||
|
|
||||||
|
/* data_stream_alignment_descriptor */
|
||||||
|
#define DESC_DATA_STREAM_ALIGNMENT_alignment_type(desc) ((desc)[2])
|
||||||
|
|
||||||
|
/* target_background_grid_descriptor */
|
||||||
|
#define DESC_TARGET_BACKGROUND_GRID_horizontal_size(desc) (GST_READ_UINT16_BE ((desc)+2) >> 2)
|
||||||
|
#define DESC_TARGET_BACKGROUND_GRID_vertical_size(desc) ((GST_READ_UINT32_BE ((desc)+2) & 0x0003fff0) >> 4)
|
||||||
|
#define DESC_TARGET_BACKGROUND_GRID_aspect_ratio_information(desc) ((desc)[5] & 0x0f)
|
||||||
|
|
||||||
|
/* video_window_descriptor */
|
||||||
|
#define DESC_VIDEO_WINDOW_horizontal_offset(desc) (GST_READ_UINT16_BE ((desc)+2) >> 2)
|
||||||
|
#define DESC_VIDEO_WINDOW_vertical_offset(desc) ((GST_READ_UINT32_BE ((desc)+2) & 0x0003fff0) >> 4)
|
||||||
|
#define DESC_VIDEO_WINDOW_window_priority(desc) ((desc)[5] & 0x0f)
|
||||||
|
|
||||||
|
/* CA_descriptor */
|
||||||
|
#define DESC_CA_system_ID(desc) (GST_READ_UINT16_BE ((desc)+2))
|
||||||
|
#define DESC_CA_PID(desc) (GST_READ_UINT16_BE ((desc)+2) & 0x1fff)
|
||||||
|
|
||||||
|
/* ISO_639_language_descriptor */
|
||||||
|
#define DESC_ISO_639_LANGUAGE_codes_n(desc) ((desc[1]) >> 2)
|
||||||
|
#define DESC_ISO_639_LANGUAGE_language_code_nth(desc,i) (&(desc[2 + (4*i)]))
|
||||||
|
#define DESC_ISO_639_LANGUAGE_audio_type_nth(desc,i) ((desc)[5 + (4*i)])
|
||||||
|
|
||||||
|
/* system_clock_descriptor */
|
||||||
|
#define DESC_SYSTEM_CLOCK_external_clock_reference_indicator(desc) (((desc)[2] & 0x80) == 0x80)
|
||||||
|
#define DESC_SYSTEM_CLOCK_clock_accuracy_integer(desc) ((desc)[2] & 0x3f)
|
||||||
|
#define DESC_SYSTEM_CLOCK_clock_accuracy_exponent(desc) (((desc)[3] & 0xe0) >> 5)
|
||||||
|
|
||||||
|
/* multiplex_buffer_utilization_descriptor */
|
||||||
|
#define DESC_MULTIPLEX_BUFFER_UTILISATION_bound_valid_flag(desc) (((desc)[2] & 0x80) == 0x80)
|
||||||
|
#define DESC_MULTIPLEX_BUFFER_UTILISATION_LTW_offset_lower_bound(desc) (GST_READ_UINT16_BE ((desc)+2) & 0x7fff)
|
||||||
|
#define DESC_MULTIPLEX_BUFFER_UTILISATION_LTW_offset_upper_bound(desc) (GST_READ_UINT16_BE ((desc)+4) & 0x7fff)
|
||||||
|
|
||||||
|
/* copyright_descriptor */
|
||||||
|
#define DESC_COPYRIGHT_copyright_identifier(desc) (GST_READ_UINT32_BE ((desc)+2))
|
||||||
|
#define DESC_COPYRIGHT_additional_copyright_info_len(desc) ((desc)[1] - 4)
|
||||||
|
#define DESC_COPYRIGHT_additional_copyright_info(desc) (&(desc)[6])
|
||||||
|
|
||||||
|
/* maximum_bitrate_descriptor */
|
||||||
|
#define DESC_MAXIMUM_BITRAT_maximum_bitrate(desc) (((((guint32)desc[2]) & 0x3f) << 16) | \
|
||||||
|
GST_READ_UINT16_BE ((desc)+3))
|
||||||
|
|
||||||
|
/* private_data_indicator_descriptor */
|
||||||
|
#define DESC_PRIVATE_DATA_INDICATOR_indicator(desc) (GST_READ_UINT32_BE(&desc[2]))
|
||||||
|
|
||||||
|
/* smoothing_buffer_descriptor */
|
||||||
|
#define DESC_SMOOTHING_BUFFER_sb_leak_rate(desc) (((((guint32)desc[2]) & 0x3f) << 16) | \
|
||||||
|
GST_READ_UINT16_BE ((desc)+3))
|
||||||
|
#define DESC_SMOOTHING_BUFFER_sb_size(desc) (((((guint32)desc[5]) & 0x3f) << 16) | \
|
||||||
|
GST_READ_UINT16_BE ((desc)+6))
|
||||||
|
/* STD_descriptor */
|
||||||
|
#define DESC_STD_leak_valid_flag(desc) (((desc)[2] & 0x01) == 0x01)
|
||||||
|
|
||||||
|
/* ibp_descriptor */
|
||||||
|
#define DESC_IBP_closed_gop_flag(desc) (((desc)[2] & 0x80) == 0x80)
|
||||||
|
#define DESC_IBP_identical_gop_flag(desc) (((desc)[2] & 0x40) == 0x40)
|
||||||
|
#define DESC_IBP_max_gop_length(desc) (GST_READ_UINT16_BE ((desc)+6) & 0x3fff)
|
||||||
|
|
||||||
|
/* time_code descriptor */
|
||||||
|
#define DESC_TIMECODE_video_pid(desc) (GST_READ_UINT16_BE ((desc) + 2) & 0x1fff)
|
||||||
|
|
||||||
|
/* Stream identifier descriptor */
|
||||||
|
#define DESC_DVB_STREAM_IDENTIFIER_component_tag(desc) (desc[2])
|
||||||
|
|
||||||
|
/* DVB Network Name descriptor */
|
||||||
|
#define DESC_DVB_NETWORK_NAME_length(desc) (GST_READ_UINT8((desc)+1))
|
||||||
|
#define DESC_DVB_NETWORK_NAME_text(desc) (desc+2)
|
||||||
|
|
||||||
|
/* DVB Service Descriptor */
|
||||||
|
#define DESC_DVB_SERVICE_type(desc) (desc[2])
|
||||||
|
#define DESC_DVB_SERVICE_provider_name_length(desc) (desc[3])
|
||||||
|
#define DESC_DVB_SERVICE_provider_name_text(desc) (desc+4)
|
||||||
|
#define DESC_DVB_SERVICE_name_length(desc) (desc[4 + DESC_DVB_SERVICE_provider_name_length(desc)])
|
||||||
|
#define DESC_DVB_SERVICE_name_text(desc) (desc + 5 + DESC_DVB_SERVICE_provider_name_length(desc))
|
||||||
|
|
||||||
|
/* DVB Component Descriptor */
|
||||||
|
#define DESC_DVB_COMPONENT_stream_content(desc) (desc[2] & 0x0F)
|
||||||
|
#define DESC_DVB_COMPONENT_type(desc) (desc[3])
|
||||||
|
#define DESC_DVB_COMPONENT_tag(desc) (desc[4])
|
||||||
|
#define DESC_DVB_COMPONENT_language(desc) (desc + 5)
|
||||||
|
|
||||||
|
/* DVB Bouquet Name Descriptor */
|
||||||
|
#define DESC_DVB_BOUQUET_NAME_text(desc) (desc + 2)
|
||||||
|
|
||||||
|
/* DVB Short Event Descriptor */
|
||||||
|
#define DESC_DVB_SHORT_EVENT_name_text(desc) (desc + 6)
|
||||||
|
#define DESC_DVB_SHORT_EVENT_name_length(desc) (desc[5])
|
||||||
|
#define DESC_DVB_SHORT_EVENT_description_text(desc) (desc + 6 + DESC_DVB_SHORT_EVENT_name_length(desc) + 1)
|
||||||
|
#define DESC_DVB_SHORT_EVENT_description_length(desc) (desc[6 + DESC_DVB_SHORT_EVENT_name_length(desc)])
|
||||||
|
|
||||||
|
/* DVB Extended Event Descriptor */
|
||||||
|
#define DESC_DVB_EXTENDED_EVENT_descriptor_number(desc) ((desc[2] & 0xF0) >> 4)
|
||||||
|
#define DESC_DVB_EXTENDED_EVENT_last_descriptor_number(desc) (desc[2] & 0x0F)
|
||||||
|
#define DESC_DVB_EXTENDED_EVENT_iso639_language_code(desc) (desc + 3)
|
||||||
|
#define DESC_DVB_EXTENDED_EVENT_items_length(desc) (desc[6])
|
||||||
|
#define DESC_DVB_EXTENDED_EVENT_items(desc) (desc + 7)
|
||||||
|
#define DESC_DVB_EXTENDED_EVENT_text_length(desc) (desc[7 + DESC_DVB_EXTENDED_EVENT_items_length(desc)])
|
||||||
|
#define DESC_DVB_EXTENDED_EVENT_text(desc) (desc + 7 + DESC_DVB_EXTENDED_EVENT_items_length(desc) + 1)
|
||||||
|
|
||||||
|
/* DVB Satellite Delivery System Descriptor */
|
||||||
|
#define DESC_DVB_SATELLITE_DELIVERY_SYSTEM_frequency(desc) (desc + 2)
|
||||||
|
#define DESC_DVB_SATELLITE_DELIVERY_SYSTEM_orbital_position(desc) (desc + 6)
|
||||||
|
#define DESC_DVB_SATELLITE_DELIVERY_SYSTEM_west_east_flag(desc) ((desc[8] & 0x80) == 0x80)
|
||||||
|
#define DESC_DVB_SATELLITE_DELIVERY_SYSTEM_polarization(desc) ((desc[8] >> 5) & 0x3)
|
||||||
|
#define DESC_DVB_SATELLITE_DELIVERY_SYSTEM_modulation(desc) (desc[8] & 0x1F)
|
||||||
|
#define DESC_DVB_SATELLITE_DELIVERY_SYSTEM_symbol_rate(desc) (desc + 9)
|
||||||
|
#define DESC_DVB_SATELLITE_DELIVERY_SYSTEM_fec_inner(desc) (desc[12] & 0x0F)
|
||||||
|
|
||||||
|
/* DVB Terrestrial Delivery System Descriptor */
|
||||||
|
#define DESC_DVB_TERRESTRIAL_DELIVERY_SYSTEM_frequency(desc) (GST_READ_UINT32_BE((desc) + 2))
|
||||||
|
#define DESC_DVB_TERRESTRIAL_DELIVERY_SYSTEM_bandwidth(desc) ((desc[6] >> 5) & 0x7)
|
||||||
|
#define DESC_DVB_TERRESTRIAL_DELIVERY_SYSTEM_constellation(desc) ((desc[7] >> 6) & 0x3)
|
||||||
|
#define DESC_DVB_TERRESTRIAL_DELIVERY_SYSTEM_hierarchy(desc) ((desc[7] >> 3) & 0x7)
|
||||||
|
#define DESC_DVB_TERRESTRIAL_DELIVERY_SYSTEM_code_rate_hp(desc) (desc[7] & 0x7)
|
||||||
|
#define DESC_DVB_TERRESTRIAL_DELIVERY_SYSTEM_code_rate_lp(desc) ((desc[8] >> 5) & 0x7)
|
||||||
|
#define DESC_DVB_TERRESTRIAL_DELIVERY_SYSTEM_guard_interval(desc) ((desc[8] >> 3) & 0x3)
|
||||||
|
#define DESC_DVB_TERRESTRIAL_DELIVERY_SYSTEM_transmission_mode(desc) ((desc[8] >> 1) & 0x3)
|
||||||
|
#define DESC_DVB_TERRESTRIAL_DELIVERY_SYSTEM_other_frequency(desc) ((desc[8] & 0x01) == 0x01)
|
||||||
|
|
||||||
|
/* DVB Cable Delivery System Descriptor */
|
||||||
|
#define DESC_DVB_CABLE_DELIVERY_SYSTEM_frequency(desc) (desc + 2)
|
||||||
|
#define DESC_DVB_CABLE_DELIVERY_SYSTEM_fec_outer(desc) (desc[7] & 0x0F)
|
||||||
|
#define DESC_DVB_CABLE_DELIVERY_SYSTEM_modulation(desc) (desc[8])
|
||||||
|
#define DESC_DVB_CABLE_DELIVERY_SYSTEM_symbol_rate(desc) (desc + 9)
|
||||||
|
#define DESC_DVB_CABLE_DELIVERY_SYSTEM_fec_inner(desc) (desc[12] & 0x0F)
|
||||||
|
|
||||||
|
/* DVB Data Broadcast Descriptor */
|
||||||
|
#define DESC_DVB_DATA_BROADCAST_data_broadcast_id(desc) (GST_READ_UINT16_BE((desc) + 2))
|
||||||
|
#define DESC_DVB_DATA_BROADCAST_component_tag(desc) (desc[4])
|
||||||
|
#define DESC_DVB_DATA_BROADCAST_selector_length(desc) (desc[5])
|
||||||
|
#define DESC_DVB_DATA_BROADCAST_selector(desc) (desc + 6)
|
||||||
|
#define DESC_DVB_DATA_BROADCAST_iso639_language_code(desc) (desc + 6 + DESC_DVB_DATA_BROADCAST_selector_length(desc))
|
||||||
|
#define DESC_DVB_DATA_BROADCAST_text_length(desc) (desc + 9 + DESC_DVB_DATA_BROADCAST_selector_length(desc))
|
||||||
|
#define DESC_DVB_DATA_BROADCAST_text(desc) (desc + 10 + DESC_DVB_DATA_BROADCAST_selector_length(desc))
|
||||||
|
|
||||||
|
/* DVB Data Broadcast Id Descriptor */
|
||||||
|
#define DESC_DVB_DATA_BROADCAST_ID_data_broadcast_id(desc) (GST_READ_UINT16_BE((desc) + 2))
|
||||||
|
#define DESC_DVB_DATA_BROADCAST_ID_id_selector_byte(desc) (desc + 4)
|
||||||
|
|
||||||
|
/* DVB Carousel Identifier Descriptor */
|
||||||
|
#define DESC_DVB_CAROUSEL_IDENTIFIER_carousel_id(desc) (GST_READ_UINT32_BE((desc) + 2))
|
||||||
|
|
||||||
|
/* registration_descriptor format IDs */
|
||||||
|
#define DRF_ID_HDMV 0x48444d56
|
||||||
|
#define DRF_ID_VC1 0x56432D31 /* defined in RP227 */
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
guint n_desc;
|
||||||
|
guint8 data_length;
|
||||||
|
guint8 *data;
|
||||||
|
} GstMPEGDescriptor;
|
||||||
|
|
||||||
|
void gst_mpegtsdesc_init_debug (void);
|
||||||
|
GstMPEGDescriptor* gst_mpeg_descriptor_parse (guint8 *data, guint size);
|
||||||
|
void gst_mpeg_descriptor_free (GstMPEGDescriptor *desc);
|
||||||
|
|
||||||
|
guint gst_mpeg_descriptor_n_desc (GstMPEGDescriptor *desc);
|
||||||
|
guint8* gst_mpeg_descriptor_find (GstMPEGDescriptor *desc, gint tag);
|
||||||
|
GArray* gst_mpeg_descriptor_find_all (GstMPEGDescriptor * desc, gint tag);
|
||||||
|
|
||||||
|
guint8* gst_mpeg_descriptor_nth (GstMPEGDescriptor *desc, guint i);
|
||||||
|
|
||||||
|
#endif /* __GST_MPEG_DESC_H__ */
|
47
gst/mpegtsdemux/gsttsdemux.c
Normal file
47
gst/mpegtsdemux/gsttsdemux.c
Normal file
|
@ -0,0 +1,47 @@
|
||||||
|
/*
|
||||||
|
*
|
||||||
|
* This library is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU Library General Public
|
||||||
|
* License as published by the Free Software Foundation; either
|
||||||
|
* version 2 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Library General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Library General Public
|
||||||
|
* License along with this library; if not, write to the
|
||||||
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||||
|
* Boston, MA 02111-1307, USA.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifdef HAVE_CONFIG_H
|
||||||
|
#include "config.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "mpegtsbase.h"
|
||||||
|
#include "mpegtspacketizer.h"
|
||||||
|
#include "mpegtsparse.h"
|
||||||
|
#include "tsdemux.h"
|
||||||
|
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
plugin_init (GstPlugin * plugin)
|
||||||
|
{
|
||||||
|
if (!gst_mpegtsbase_plugin_init (plugin))
|
||||||
|
return FALSE;
|
||||||
|
if (!gst_mpegtsparse_plugin_init (plugin))
|
||||||
|
return FALSE;
|
||||||
|
if (!gst_ts_demux_plugin_init (plugin))
|
||||||
|
return FALSE;
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
GST_PLUGIN_DEFINE (GST_VERSION_MAJOR,
|
||||||
|
GST_VERSION_MINOR,
|
||||||
|
"mpegtsdemux",
|
||||||
|
"MPEG TS demuxer",
|
||||||
|
plugin_init, VERSION,
|
||||||
|
GST_LICENSE_UNKNOWN, GST_PACKAGE_NAME, GST_PACKAGE_ORIGIN);
|
1257
gst/mpegtsdemux/mpegtsbase.c
Normal file
1257
gst/mpegtsdemux/mpegtsbase.c
Normal file
File diff suppressed because it is too large
Load diff
169
gst/mpegtsdemux/mpegtsbase.h
Normal file
169
gst/mpegtsdemux/mpegtsbase.h
Normal file
|
@ -0,0 +1,169 @@
|
||||||
|
/*
|
||||||
|
* mpegtsbase.h - GStreamer MPEG transport stream base class
|
||||||
|
* Copyright (C) 2009 Edward Hervey <edward.hervey@collabora.co.uk>
|
||||||
|
* 2007 Alessandro Decina
|
||||||
|
*
|
||||||
|
* Authors:
|
||||||
|
* Alessandro Decina <alessandro@nnva.org>
|
||||||
|
* Edward Hervey <edward.hervey@collabora.co.uk>
|
||||||
|
*
|
||||||
|
* This library is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU Library General Public
|
||||||
|
* License as published by the Free Software Foundation; either
|
||||||
|
* version 2 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Library General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Library General Public
|
||||||
|
* License along with this library; if not, write to the
|
||||||
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||||
|
* Boston, MA 02111-1307, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#ifndef GST_MPEG_TS_BASE_H
|
||||||
|
#define GST_MPEG_TS_BASE_H
|
||||||
|
|
||||||
|
#include <gst/gst.h>
|
||||||
|
#include "mpegtspacketizer.h"
|
||||||
|
|
||||||
|
G_BEGIN_DECLS
|
||||||
|
|
||||||
|
#define GST_TYPE_MPEGTS_BASE \
|
||||||
|
(mpegts_base_get_type())
|
||||||
|
#define GST_MPEGTS_BASE(obj) \
|
||||||
|
(G_TYPE_CHECK_INSTANCE_CAST((obj),GST_TYPE_MPEGTS_BASE,MpegTSBase))
|
||||||
|
#define GST_MPEGTS_BASE_CLASS(klass) \
|
||||||
|
(G_TYPE_CHECK_CLASS_CAST((klass),GST_TYPE_MPEGTS_BASE,MpegTSBaseClass))
|
||||||
|
#define GST_IS_MPEGTS_BASE(obj) \
|
||||||
|
(G_TYPE_CHECK_INSTANCE_TYPE((obj),GST_TYPE_MPEGTS_BASE))
|
||||||
|
#define GST_IS_MPEGTS_BASE_CLASS(klass) \
|
||||||
|
(G_TYPE_CHECK_CLASS_TYPE((klass),GST_TYPE_MPEGTS_BASE))
|
||||||
|
#define GST_MPEGTS_BASE_GET_CLASS(obj) \
|
||||||
|
(G_TYPE_INSTANCE_GET_CLASS ((obj), GST_TYPE_MPEGTS_BASE, MpegTSBaseClass))
|
||||||
|
|
||||||
|
typedef struct _MpegTSBase MpegTSBase;
|
||||||
|
typedef struct _MpegTSBaseClass MpegTSBaseClass;
|
||||||
|
typedef struct _MpegTSBaseStream MpegTSBaseStream;
|
||||||
|
typedef struct _MpegTSBaseProgram MpegTSBaseProgram;
|
||||||
|
|
||||||
|
struct _MpegTSBaseStream
|
||||||
|
{
|
||||||
|
guint16 pid;
|
||||||
|
guint8 stream_type;
|
||||||
|
GstStructure* stream_info;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct _MpegTSBaseProgram
|
||||||
|
{
|
||||||
|
gint program_number;
|
||||||
|
guint16 pmt_pid;
|
||||||
|
guint16 pcr_pid;
|
||||||
|
GstStructure *pmt_info;
|
||||||
|
MpegTSBaseStream **streams;
|
||||||
|
gint patcount;
|
||||||
|
|
||||||
|
/* Pending Tags for the program */
|
||||||
|
GstTagList *tags;
|
||||||
|
guint event_id;
|
||||||
|
};
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
BASE_MODE_SCANNING,
|
||||||
|
BASE_MODE_SEEKING,
|
||||||
|
BASE_MODE_STREAMING
|
||||||
|
} MpegTSBaseMode;
|
||||||
|
|
||||||
|
struct _MpegTSBase {
|
||||||
|
GstElement element;
|
||||||
|
|
||||||
|
GstPad *sinkpad;
|
||||||
|
|
||||||
|
/* pull-based behaviour */
|
||||||
|
MpegTSBaseMode mode;
|
||||||
|
|
||||||
|
/* location of first sync point */
|
||||||
|
guint64 initial_sync_point;
|
||||||
|
|
||||||
|
/* Current pull offset (also set by seek handler) */
|
||||||
|
guint64 seek_offset;
|
||||||
|
|
||||||
|
/* Cached packetsize */
|
||||||
|
guint16 packetsize;
|
||||||
|
|
||||||
|
/* the following vars must be protected with the OBJECT_LOCK as they can be
|
||||||
|
* accessed from the application thread and the streaming thread */
|
||||||
|
GHashTable *programs;
|
||||||
|
|
||||||
|
GstStructure *pat;
|
||||||
|
MpegTSPacketizer2 *packetizer;
|
||||||
|
|
||||||
|
/* arrays that say whether a pid is a known psi pid or a pes pid
|
||||||
|
* FIXME: Make these bit arrays so we can make them 8 times smaller */
|
||||||
|
gboolean *known_psi;
|
||||||
|
gboolean *is_pes;
|
||||||
|
|
||||||
|
gboolean disposed;
|
||||||
|
|
||||||
|
/* size of the MpegTSBaseProgram structure, can be overridden
|
||||||
|
* by subclasses if they have their own MpegTSBaseProgram subclasses. */
|
||||||
|
gsize program_size;
|
||||||
|
|
||||||
|
/* size of the MpegTSBaseStream structure, can be overridden
|
||||||
|
* by subclasses if they have their own MpegTSBaseStream subclasses */
|
||||||
|
gsize stream_size;
|
||||||
|
|
||||||
|
/*Offset from the origin to the first PAT (pullmode) */
|
||||||
|
guint64 first_pat_offset;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct _MpegTSBaseClass {
|
||||||
|
GstElementClass parent_class;
|
||||||
|
|
||||||
|
/* Virtual methods */
|
||||||
|
GstFlowReturn (*push) (MpegTSBase *base, MpegTSPacketizerPacket *packet, MpegTSPacketizerSection * section);
|
||||||
|
gboolean (*push_event) (MpegTSBase *base, GstEvent * event);
|
||||||
|
/* program_started gets called when program's pmt arrives for first time */
|
||||||
|
void (*program_started) (MpegTSBase *base, MpegTSBaseProgram *program);
|
||||||
|
/* program_stopped gets called when pat no longer has program's pmt */
|
||||||
|
void (*program_stopped) (MpegTSBase *base, MpegTSBaseProgram *program);
|
||||||
|
|
||||||
|
/* stream_added is called whenever a new stream has been identified */
|
||||||
|
void (*stream_added) (MpegTSBase *base, MpegTSBaseStream *stream, MpegTSBaseProgram *program);
|
||||||
|
/* stream_removed is called whenever a stream is no longer referenced */
|
||||||
|
void (*stream_removed) (MpegTSBase *base, MpegTSBaseStream *stream);
|
||||||
|
|
||||||
|
/* find_timestamps is called to find PCR */
|
||||||
|
GstFlowReturn (*find_timestamps) (MpegTSBase * base, guint64 initoff, guint64 *offset);
|
||||||
|
|
||||||
|
/* signals */
|
||||||
|
void (*pat_info) (GstStructure *pat);
|
||||||
|
void (*pmt_info) (GstStructure *pmt);
|
||||||
|
void (*nit_info) (GstStructure *nit);
|
||||||
|
void (*sdt_info) (GstStructure *sdt);
|
||||||
|
void (*eit_info) (GstStructure *eit);
|
||||||
|
};
|
||||||
|
|
||||||
|
GType mpegts_base_get_type(void);
|
||||||
|
|
||||||
|
MpegTSBaseProgram *mpegts_base_get_program (MpegTSBase * base, gint program_number);
|
||||||
|
MpegTSBaseProgram *mpegts_base_add_program (MpegTSBase * base, gint program_number, guint16 pmt_pid);
|
||||||
|
|
||||||
|
guint8 *mpegts_get_descriptor_from_stream (MpegTSBaseStream * stream, guint8 tag);
|
||||||
|
guint8 *mpegts_get_descriptor_from_program (MpegTSBaseProgram * program, guint8 tag);
|
||||||
|
|
||||||
|
|
||||||
|
gboolean gst_mpegtsbase_plugin_init (GstPlugin * plugin);
|
||||||
|
|
||||||
|
gboolean mpegts_base_is_psi (MpegTSBase * base, MpegTSPacketizerPacket * packet);
|
||||||
|
gboolean mpegts_base_handle_psi (MpegTSBase * base, MpegTSPacketizerSection * section);
|
||||||
|
|
||||||
|
void mpegts_base_program_remove_stream (MpegTSBase * base, MpegTSBaseProgram * program, guint16 pid);
|
||||||
|
|
||||||
|
void mpegts_base_remove_program(MpegTSBase *base, gint program_number);
|
||||||
|
G_END_DECLS
|
||||||
|
|
||||||
|
#endif /* GST_MPEG_TS_BASE_H */
|
2714
gst/mpegtsdemux/mpegtspacketizer.c
Normal file
2714
gst/mpegtsdemux/mpegtspacketizer.c
Normal file
File diff suppressed because it is too large
Load diff
167
gst/mpegtsdemux/mpegtspacketizer.h
Normal file
167
gst/mpegtsdemux/mpegtspacketizer.h
Normal file
|
@ -0,0 +1,167 @@
|
||||||
|
/*
|
||||||
|
* mpegtspacketizer.h -
|
||||||
|
* Copyright (C) 2007 Alessandro Decina
|
||||||
|
*
|
||||||
|
* Authors:
|
||||||
|
* Alessandro Decina <alessandro@nnva.org>
|
||||||
|
*
|
||||||
|
* This library is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU Library General Public
|
||||||
|
* License as published by the Free Software Foundation; either
|
||||||
|
* version 2 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Library General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Library General Public
|
||||||
|
* License along with this library; if not, write to the
|
||||||
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||||
|
* Boston, MA 02111-1307, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef GST_MPEGTS_PACKETIZER_H
|
||||||
|
#define GST_MPEGTS_PACKETIZER_H
|
||||||
|
|
||||||
|
#include <gst/gst.h>
|
||||||
|
#include <gst/base/gstadapter.h>
|
||||||
|
#include <glib.h>
|
||||||
|
|
||||||
|
#define MPEGTS_NORMAL_PACKETSIZE 188
|
||||||
|
#define MPEGTS_M2TS_PACKETSIZE 192
|
||||||
|
#define MPEGTS_DVB_ASI_PACKETSIZE 204
|
||||||
|
#define MPEGTS_ATSC_PACKETSIZE 208
|
||||||
|
|
||||||
|
#define MPEGTS_MIN_PACKETSIZE MPEGTS_NORMAL_PACKETSIZE
|
||||||
|
#define MPEGTS_MAX_PACKETSIZE MPEGTS_ATSC_PACKETSIZE
|
||||||
|
|
||||||
|
#define MPEGTS_AFC_PCR_FLAG 0x10
|
||||||
|
#define MPEGTS_AFC_OPCR_FLAG 0x08
|
||||||
|
|
||||||
|
G_BEGIN_DECLS
|
||||||
|
|
||||||
|
#define GST_TYPE_MPEGTS_PACKETIZER \
|
||||||
|
(mpegts_packetizer_get_type())
|
||||||
|
#define GST_MPEGTS_PACKETIZER(obj) \
|
||||||
|
(G_TYPE_CHECK_INSTANCE_CAST((obj),GST_TYPE_MPEGTS_PACKETIZER,MpegTSPacketizer2))
|
||||||
|
#define GST_MPEGTS_PACKETIZER_CLASS(klass) \
|
||||||
|
(G_TYPE_CHECK_CLASS_CAST((klass),GST_TYPE_MPEGTS_PACKETIZER,MpegTSPacketizer2Class))
|
||||||
|
#define GST_IS_MPEGTS_PACKETIZER(obj) \
|
||||||
|
(G_TYPE_CHECK_INSTANCE_TYPE((obj),GST_TYPE_MPEGTS_PACKETIZER))
|
||||||
|
#define GST_IS_MPEGTS_PACKETIZER_CLASS(klass) \
|
||||||
|
(G_TYPE_CHECK_CLASS_TYPE((klass),GST_TYPE_MPEGTS_PACKETIZER))
|
||||||
|
|
||||||
|
typedef struct _MpegTSPacketizer2 MpegTSPacketizer2;
|
||||||
|
typedef struct _MpegTSPacketizer2Class MpegTSPacketizer2Class;
|
||||||
|
|
||||||
|
typedef struct
|
||||||
|
{
|
||||||
|
guint continuity_counter;
|
||||||
|
GstAdapter *section_adapter;
|
||||||
|
guint8 section_table_id;
|
||||||
|
guint section_length;
|
||||||
|
GSList *subtables;
|
||||||
|
guint64 offset;
|
||||||
|
} MpegTSPacketizerStream;
|
||||||
|
|
||||||
|
struct _MpegTSPacketizer2 {
|
||||||
|
GObject object;
|
||||||
|
|
||||||
|
GstAdapter *adapter;
|
||||||
|
/* streams hashed by pid */
|
||||||
|
MpegTSPacketizerStream **streams;
|
||||||
|
gboolean disposed;
|
||||||
|
gboolean know_packet_size;
|
||||||
|
guint16 packet_size;
|
||||||
|
GstCaps *caps;
|
||||||
|
|
||||||
|
/* current offset of the tip of the adapter */
|
||||||
|
guint64 offset;
|
||||||
|
gboolean empty;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct _MpegTSPacketizer2Class {
|
||||||
|
GObjectClass object_class;
|
||||||
|
};
|
||||||
|
|
||||||
|
typedef struct
|
||||||
|
{
|
||||||
|
GstBuffer *buffer;
|
||||||
|
gint16 pid;
|
||||||
|
guint8 payload_unit_start_indicator;
|
||||||
|
guint8 adaptation_field_control;
|
||||||
|
guint8 continuity_counter;
|
||||||
|
guint8 *payload;
|
||||||
|
|
||||||
|
guint8 *data_start;
|
||||||
|
guint8 *data_end;
|
||||||
|
guint8 *data;
|
||||||
|
|
||||||
|
guint8 afc_flags;
|
||||||
|
guint64 pcr;
|
||||||
|
guint64 opcr;
|
||||||
|
guint64 offset;
|
||||||
|
} MpegTSPacketizerPacket;
|
||||||
|
|
||||||
|
typedef struct
|
||||||
|
{
|
||||||
|
gboolean complete;
|
||||||
|
GstBuffer *buffer;
|
||||||
|
gint16 pid;
|
||||||
|
guint8 table_id;
|
||||||
|
guint16 subtable_extension;
|
||||||
|
guint section_length;
|
||||||
|
guint8 version_number;
|
||||||
|
guint8 current_next_indicator;
|
||||||
|
guint32 crc;
|
||||||
|
} MpegTSPacketizerSection;
|
||||||
|
|
||||||
|
typedef struct
|
||||||
|
{
|
||||||
|
guint8 table_id;
|
||||||
|
/* the spec says sub_table_extension is the fourth and fifth byte of a
|
||||||
|
* section when the section_syntax_indicator is set to a value of "1". If
|
||||||
|
* section_syntax_indicator is 0, sub_table_extension will be set to 0 */
|
||||||
|
guint16 subtable_extension;
|
||||||
|
guint8 version_number;
|
||||||
|
guint32 crc;
|
||||||
|
} MpegTSPacketizerStreamSubtable;
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
PACKET_BAD = FALSE,
|
||||||
|
PACKET_OK = TRUE,
|
||||||
|
PACKET_NEED_MORE
|
||||||
|
} MpegTSPacketizerPacketReturn;
|
||||||
|
|
||||||
|
GType mpegts_packetizer_get_type(void);
|
||||||
|
|
||||||
|
MpegTSPacketizer2 *mpegts_packetizer_new (void);
|
||||||
|
void mpegts_packetizer_clear (MpegTSPacketizer2 *packetizer);
|
||||||
|
void mpegts_packetizer_push (MpegTSPacketizer2 *packetizer, GstBuffer *buffer);
|
||||||
|
gboolean mpegts_packetizer_has_packets (MpegTSPacketizer2 *packetizer);
|
||||||
|
MpegTSPacketizerPacketReturn mpegts_packetizer_next_packet (MpegTSPacketizer2 *packetizer,
|
||||||
|
MpegTSPacketizerPacket *packet);
|
||||||
|
void mpegts_packetizer_clear_packet (MpegTSPacketizer2 *packetizer,
|
||||||
|
MpegTSPacketizerPacket *packet);
|
||||||
|
void mpegts_packetizer_remove_stream(MpegTSPacketizer2 *packetizer,
|
||||||
|
gint16 pid);
|
||||||
|
|
||||||
|
gboolean mpegts_packetizer_push_section (MpegTSPacketizer2 *packetzer,
|
||||||
|
MpegTSPacketizerPacket *packet, MpegTSPacketizerSection *section);
|
||||||
|
GstStructure *mpegts_packetizer_parse_pat (MpegTSPacketizer2 *packetizer,
|
||||||
|
MpegTSPacketizerSection *section);
|
||||||
|
GstStructure *mpegts_packetizer_parse_pmt (MpegTSPacketizer2 *packetizer,
|
||||||
|
MpegTSPacketizerSection *section);
|
||||||
|
GstStructure *mpegts_packetizer_parse_nit (MpegTSPacketizer2 *packetizer,
|
||||||
|
MpegTSPacketizerSection *section);
|
||||||
|
GstStructure *mpegts_packetizer_parse_sdt (MpegTSPacketizer2 *packetizer,
|
||||||
|
MpegTSPacketizerSection *section);
|
||||||
|
GstStructure *mpegts_packetizer_parse_eit (MpegTSPacketizer2 *packetizer,
|
||||||
|
MpegTSPacketizerSection *section);
|
||||||
|
GstStructure *mpegts_packetizer_parse_tdt (MpegTSPacketizer2 *packetizer,
|
||||||
|
MpegTSPacketizerSection *section);
|
||||||
|
|
||||||
|
G_END_DECLS
|
||||||
|
|
||||||
|
#endif /* GST_MPEGTS_PACKETIZER_H */
|
718
gst/mpegtsdemux/mpegtsparse.c
Normal file
718
gst/mpegtsdemux/mpegtsparse.c
Normal file
|
@ -0,0 +1,718 @@
|
||||||
|
/*
|
||||||
|
* mpegtsparse.c -
|
||||||
|
* Copyright (C) 2007 Alessandro Decina
|
||||||
|
*
|
||||||
|
* Authors:
|
||||||
|
* Alessandro Decina <alessandro@nnva.org>
|
||||||
|
* Zaheer Abbas Merali <zaheerabbas at merali dot org>
|
||||||
|
*
|
||||||
|
* This library is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU Library General Public
|
||||||
|
* License as published by the Free Software Foundation; either
|
||||||
|
* version 2 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Library General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Library General Public
|
||||||
|
* License along with this library; if not, write to the
|
||||||
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||||
|
* Boston, MA 02111-1307, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifdef HAVE_CONFIG_H
|
||||||
|
#include "config.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
#include "mpegtsbase.h"
|
||||||
|
#include "mpegtsparse.h"
|
||||||
|
#include "gstmpegdesc.h"
|
||||||
|
|
||||||
|
/* latency in mseconds */
|
||||||
|
#define TS_LATENCY 700
|
||||||
|
|
||||||
|
#define TABLE_ID_UNSET 0xFF
|
||||||
|
#define RUNNING_STATUS_RUNNING 4
|
||||||
|
|
||||||
|
GST_DEBUG_CATEGORY_STATIC (mpegts_parse_debug);
|
||||||
|
#define GST_CAT_DEFAULT mpegts_parse_debug
|
||||||
|
|
||||||
|
typedef struct _MpegTSParsePad MpegTSParsePad;
|
||||||
|
|
||||||
|
typedef struct
|
||||||
|
{
|
||||||
|
MpegTSBaseProgram program;
|
||||||
|
gint selected;
|
||||||
|
gboolean active;
|
||||||
|
MpegTSParsePad *tspad;
|
||||||
|
} MpegTSParseProgram;
|
||||||
|
|
||||||
|
struct _MpegTSParsePad
|
||||||
|
{
|
||||||
|
GstPad *pad;
|
||||||
|
|
||||||
|
/* the program number that the peer wants on this pad */
|
||||||
|
gint program_number;
|
||||||
|
MpegTSParseProgram *program;
|
||||||
|
|
||||||
|
/* set to FALSE before a push and TRUE after */
|
||||||
|
gboolean pushed;
|
||||||
|
|
||||||
|
/* the return of the latest push */
|
||||||
|
GstFlowReturn flow_return;
|
||||||
|
|
||||||
|
GstTagList *tags;
|
||||||
|
guint event_id;
|
||||||
|
};
|
||||||
|
|
||||||
|
static GstStaticPadTemplate src_template =
|
||||||
|
GST_STATIC_PAD_TEMPLATE ("src%d", GST_PAD_SRC,
|
||||||
|
GST_PAD_REQUEST,
|
||||||
|
GST_STATIC_CAPS ("video/mpegts, " "systemstream = (boolean) true ")
|
||||||
|
);
|
||||||
|
|
||||||
|
static GstStaticPadTemplate program_template =
|
||||||
|
GST_STATIC_PAD_TEMPLATE ("program_%d", GST_PAD_SRC,
|
||||||
|
GST_PAD_SOMETIMES,
|
||||||
|
GST_STATIC_CAPS ("video/mpegts, " "systemstream = (boolean) true ")
|
||||||
|
);
|
||||||
|
|
||||||
|
enum
|
||||||
|
{
|
||||||
|
ARG_0,
|
||||||
|
PROP_PROGRAM_NUMBERS,
|
||||||
|
/* FILL ME */
|
||||||
|
};
|
||||||
|
|
||||||
|
static void
|
||||||
|
mpegts_parse_program_started (MpegTSBase * base, MpegTSBaseProgram * program);
|
||||||
|
static void
|
||||||
|
mpegts_parse_program_stopped (MpegTSBase * base, MpegTSBaseProgram * program);
|
||||||
|
|
||||||
|
static GstFlowReturn
|
||||||
|
mpegts_parse_push (MpegTSBase * base, MpegTSPacketizerPacket * packet,
|
||||||
|
MpegTSPacketizerSection * section);
|
||||||
|
static void mpegts_parse_set_property (GObject * object, guint prop_id,
|
||||||
|
const GValue * value, GParamSpec * pspec);
|
||||||
|
static void mpegts_parse_get_property (GObject * object, guint prop_id,
|
||||||
|
GValue * value, GParamSpec * pspec);
|
||||||
|
static void mpegts_parse_finalize (GObject * object);
|
||||||
|
|
||||||
|
static MpegTSParsePad *mpegts_parse_create_tspad (MpegTSParse2 * parse,
|
||||||
|
const gchar * name);
|
||||||
|
static void mpegts_parse_destroy_tspad (MpegTSParse2 * parse,
|
||||||
|
MpegTSParsePad * tspad);
|
||||||
|
static GstPad *mpegts_parse_activate_program (MpegTSParse2 * parse,
|
||||||
|
MpegTSParseProgram * program);
|
||||||
|
static void mpegts_parse_reset_selected_programs (MpegTSParse2 * parse,
|
||||||
|
gchar * programs);
|
||||||
|
|
||||||
|
static void mpegts_parse_pad_removed (GstElement * element, GstPad * pad);
|
||||||
|
static GstPad *mpegts_parse_request_new_pad (GstElement * element,
|
||||||
|
GstPadTemplate * templ, const gchar * name);
|
||||||
|
static void mpegts_parse_release_pad (GstElement * element, GstPad * pad);
|
||||||
|
static gboolean mpegts_parse_src_pad_query (GstPad * pad, GstQuery * query);
|
||||||
|
static gboolean push_event (MpegTSBase * base, GstEvent * event);
|
||||||
|
|
||||||
|
GST_BOILERPLATE (MpegTSParse2, mpegts_parse, MpegTSBase, GST_TYPE_MPEGTS_BASE);
|
||||||
|
|
||||||
|
static void
|
||||||
|
mpegts_parse_base_init (gpointer klass)
|
||||||
|
{
|
||||||
|
GstElementClass *element_class = GST_ELEMENT_CLASS (klass);
|
||||||
|
|
||||||
|
gst_element_class_add_pad_template (element_class,
|
||||||
|
gst_static_pad_template_get (&src_template));
|
||||||
|
gst_element_class_add_pad_template (element_class,
|
||||||
|
gst_static_pad_template_get (&program_template));
|
||||||
|
|
||||||
|
gst_element_class_set_details_simple (element_class,
|
||||||
|
"MPEG transport stream parser", "Codec/Parser",
|
||||||
|
"Parses MPEG2 transport streams",
|
||||||
|
"Alessandro Decina <alessandro@nnva.org>, "
|
||||||
|
"Zaheer Abbas Merali <zaheerabbas at merali dot org>");
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
mpegts_parse_class_init (MpegTSParse2Class * klass)
|
||||||
|
{
|
||||||
|
GObjectClass *gobject_class;
|
||||||
|
GstElementClass *element_class;
|
||||||
|
MpegTSBaseClass *ts_class;
|
||||||
|
|
||||||
|
element_class = GST_ELEMENT_CLASS (klass);
|
||||||
|
element_class->pad_removed = mpegts_parse_pad_removed;
|
||||||
|
element_class->request_new_pad = mpegts_parse_request_new_pad;
|
||||||
|
element_class->release_pad = mpegts_parse_release_pad;
|
||||||
|
|
||||||
|
gobject_class = G_OBJECT_CLASS (klass);
|
||||||
|
gobject_class->set_property = mpegts_parse_set_property;
|
||||||
|
gobject_class->get_property = mpegts_parse_get_property;
|
||||||
|
gobject_class->finalize = mpegts_parse_finalize;
|
||||||
|
|
||||||
|
g_object_class_install_property (gobject_class, PROP_PROGRAM_NUMBERS,
|
||||||
|
g_param_spec_string ("program-numbers",
|
||||||
|
"Program Numbers",
|
||||||
|
"Colon separated list of programs", "",
|
||||||
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
||||||
|
|
||||||
|
ts_class = GST_MPEGTS_BASE_CLASS (klass);
|
||||||
|
ts_class->push = GST_DEBUG_FUNCPTR (mpegts_parse_push);
|
||||||
|
ts_class->push_event = GST_DEBUG_FUNCPTR (push_event);
|
||||||
|
ts_class->program_started = GST_DEBUG_FUNCPTR (mpegts_parse_program_started);
|
||||||
|
ts_class->program_stopped = GST_DEBUG_FUNCPTR (mpegts_parse_program_stopped);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
mpegts_parse_init (MpegTSParse2 * parse, MpegTSParse2Class * klass)
|
||||||
|
{
|
||||||
|
parse->need_sync_program_pads = FALSE;
|
||||||
|
parse->program_numbers = g_strdup ("");
|
||||||
|
parse->pads_to_add = NULL;
|
||||||
|
parse->pads_to_remove = NULL;
|
||||||
|
GST_MPEGTS_BASE (parse)->program_size = sizeof (MpegTSParseProgram);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
mpegts_parse_finalize (GObject * object)
|
||||||
|
{
|
||||||
|
MpegTSParse2 *parse = GST_MPEGTS_PARSE (object);
|
||||||
|
|
||||||
|
g_free (parse->program_numbers);
|
||||||
|
|
||||||
|
if (G_OBJECT_CLASS (parent_class)->finalize)
|
||||||
|
G_OBJECT_CLASS (parent_class)->finalize (object);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
mpegts_parse_set_property (GObject * object, guint prop_id,
|
||||||
|
const GValue * value, GParamSpec * pspec)
|
||||||
|
{
|
||||||
|
MpegTSParse2 *parse = GST_MPEGTS_PARSE (object);
|
||||||
|
|
||||||
|
switch (prop_id) {
|
||||||
|
case PROP_PROGRAM_NUMBERS:
|
||||||
|
mpegts_parse_reset_selected_programs (parse, g_value_dup_string (value));
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
mpegts_parse_get_property (GObject * object, guint prop_id,
|
||||||
|
GValue * value, GParamSpec * pspec)
|
||||||
|
{
|
||||||
|
MpegTSParse2 *parse = GST_MPEGTS_PARSE (object);
|
||||||
|
|
||||||
|
switch (prop_id) {
|
||||||
|
case PROP_PROGRAM_NUMBERS:
|
||||||
|
g_value_set_string (value, parse->program_numbers);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static GstPad *
|
||||||
|
mpegts_parse_activate_program (MpegTSParse2 * parse,
|
||||||
|
MpegTSParseProgram * program)
|
||||||
|
{
|
||||||
|
MpegTSParsePad *tspad;
|
||||||
|
gchar *pad_name;
|
||||||
|
|
||||||
|
pad_name =
|
||||||
|
g_strdup_printf ("program_%d",
|
||||||
|
((MpegTSBaseProgram *) program)->program_number);
|
||||||
|
|
||||||
|
tspad = mpegts_parse_create_tspad (parse, pad_name);
|
||||||
|
tspad->program_number = ((MpegTSBaseProgram *) program)->program_number;
|
||||||
|
tspad->program = program;
|
||||||
|
program->tspad = tspad;
|
||||||
|
g_free (pad_name);
|
||||||
|
gst_pad_set_active (tspad->pad, TRUE);
|
||||||
|
program->active = TRUE;
|
||||||
|
|
||||||
|
return tspad->pad;
|
||||||
|
}
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
push_event (MpegTSBase * base, GstEvent * event)
|
||||||
|
{
|
||||||
|
MpegTSParse2 *parse = (MpegTSParse2 *) base;
|
||||||
|
GList *tmp;
|
||||||
|
|
||||||
|
for (tmp = GST_ELEMENT_CAST (parse)->srcpads; tmp; tmp = tmp->next) {
|
||||||
|
GstPad *pad = (GstPad *) tmp->data;
|
||||||
|
if (pad) {
|
||||||
|
gst_event_ref (event);
|
||||||
|
gst_pad_push_event (pad, event);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static GstPad *
|
||||||
|
mpegts_parse_deactivate_program (MpegTSParse2 * parse,
|
||||||
|
MpegTSParseProgram * program)
|
||||||
|
{
|
||||||
|
MpegTSParsePad *tspad;
|
||||||
|
|
||||||
|
tspad = program->tspad;
|
||||||
|
gst_pad_set_active (tspad->pad, FALSE);
|
||||||
|
program->active = FALSE;
|
||||||
|
|
||||||
|
/* tspad will be destroyed in GstElementClass::pad_removed */
|
||||||
|
|
||||||
|
return tspad->pad;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
mpegts_parse_sync_program_pads (MpegTSParse2 * parse)
|
||||||
|
{
|
||||||
|
GList *walk;
|
||||||
|
|
||||||
|
GST_INFO_OBJECT (parse, "begin sync pads");
|
||||||
|
for (walk = parse->pads_to_remove; walk; walk = walk->next)
|
||||||
|
gst_element_remove_pad (GST_ELEMENT (parse), GST_PAD (walk->data));
|
||||||
|
|
||||||
|
for (walk = parse->pads_to_add; walk; walk = walk->next)
|
||||||
|
gst_element_add_pad (GST_ELEMENT (parse), GST_PAD (walk->data));
|
||||||
|
|
||||||
|
if (parse->pads_to_add)
|
||||||
|
g_list_free (parse->pads_to_add);
|
||||||
|
|
||||||
|
if (parse->pads_to_remove)
|
||||||
|
g_list_free (parse->pads_to_remove);
|
||||||
|
|
||||||
|
GST_OBJECT_LOCK (parse);
|
||||||
|
parse->pads_to_remove = NULL;
|
||||||
|
parse->pads_to_add = NULL;
|
||||||
|
parse->need_sync_program_pads = FALSE;
|
||||||
|
GST_OBJECT_UNLOCK (parse);
|
||||||
|
|
||||||
|
GST_INFO_OBJECT (parse, "end sync pads");
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
foreach_program_activate_or_deactivate (gpointer key, gpointer value,
|
||||||
|
gpointer data)
|
||||||
|
{
|
||||||
|
MpegTSParse2 *parse = GST_MPEGTS_PARSE (data);
|
||||||
|
MpegTSParseProgram *program = (MpegTSParseProgram *) value;
|
||||||
|
|
||||||
|
/* at this point selected programs have program->selected == 2,
|
||||||
|
* unselected programs thay may have to be deactivated have selected == 1 and
|
||||||
|
* unselected inactive programs have selected == 0 */
|
||||||
|
|
||||||
|
switch (--program->selected) {
|
||||||
|
case 1:
|
||||||
|
/* selected */
|
||||||
|
if (!program->active
|
||||||
|
&& ((MpegTSBaseProgram *) program)->pmt_pid != G_MAXUINT16)
|
||||||
|
parse->pads_to_add =
|
||||||
|
g_list_append (parse->pads_to_add,
|
||||||
|
mpegts_parse_activate_program (parse, program));
|
||||||
|
else {
|
||||||
|
program->selected = 2;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 0:
|
||||||
|
/* unselected */
|
||||||
|
if (program->active)
|
||||||
|
parse->pads_to_remove = g_list_append (parse->pads_to_remove,
|
||||||
|
mpegts_parse_deactivate_program (parse, program));
|
||||||
|
break;
|
||||||
|
case -1:
|
||||||
|
/* was already unselected */
|
||||||
|
program->selected = 0;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
g_return_if_reached ();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
mpegts_parse_reset_selected_programs (MpegTSParse2 * parse,
|
||||||
|
gchar * program_numbers)
|
||||||
|
{
|
||||||
|
GST_OBJECT_LOCK (parse);
|
||||||
|
if (parse->program_numbers)
|
||||||
|
g_free (parse->program_numbers);
|
||||||
|
|
||||||
|
parse->program_numbers = program_numbers;
|
||||||
|
|
||||||
|
if (*parse->program_numbers != '\0') {
|
||||||
|
gint program_number;
|
||||||
|
MpegTSParseProgram *program;
|
||||||
|
gchar **progs, **walk;
|
||||||
|
|
||||||
|
progs = g_strsplit (parse->program_numbers, ":", 0);
|
||||||
|
|
||||||
|
walk = progs;
|
||||||
|
while (*walk != NULL) {
|
||||||
|
program_number = strtol (*walk, NULL, 0);
|
||||||
|
program =
|
||||||
|
(MpegTSParseProgram *) mpegts_base_get_program ((MpegTSBase *) parse,
|
||||||
|
program_number);
|
||||||
|
if (program == NULL)
|
||||||
|
/* create the program, it will get activated once we get a PMT for it */
|
||||||
|
program = (MpegTSParseProgram *) mpegts_base_add_program ((MpegTSBase *)
|
||||||
|
parse, program_number, G_MAXUINT16);
|
||||||
|
program->selected = 2;
|
||||||
|
++walk;
|
||||||
|
}
|
||||||
|
g_strfreev (progs);
|
||||||
|
}
|
||||||
|
|
||||||
|
g_hash_table_foreach (((MpegTSBase *) parse)->programs,
|
||||||
|
foreach_program_activate_or_deactivate, parse);
|
||||||
|
|
||||||
|
if (parse->pads_to_remove || parse->pads_to_add)
|
||||||
|
parse->need_sync_program_pads = TRUE;
|
||||||
|
GST_OBJECT_UNLOCK (parse);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static MpegTSParsePad *
|
||||||
|
mpegts_parse_create_tspad (MpegTSParse2 * parse, const gchar * pad_name)
|
||||||
|
{
|
||||||
|
GstPad *pad;
|
||||||
|
MpegTSParsePad *tspad;
|
||||||
|
|
||||||
|
pad = gst_pad_new_from_static_template (&program_template, pad_name);
|
||||||
|
gst_pad_set_query_function (pad,
|
||||||
|
GST_DEBUG_FUNCPTR (mpegts_parse_src_pad_query));
|
||||||
|
|
||||||
|
/* create our wrapper */
|
||||||
|
tspad = g_new0 (MpegTSParsePad, 1);
|
||||||
|
tspad->pad = pad;
|
||||||
|
tspad->program_number = -1;
|
||||||
|
tspad->program = NULL;
|
||||||
|
tspad->pushed = FALSE;
|
||||||
|
tspad->flow_return = GST_FLOW_NOT_LINKED;
|
||||||
|
gst_pad_set_element_private (pad, tspad);
|
||||||
|
|
||||||
|
return tspad;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
mpegts_parse_destroy_tspad (MpegTSParse2 * parse, MpegTSParsePad * tspad)
|
||||||
|
{
|
||||||
|
if (tspad->tags) {
|
||||||
|
gst_tag_list_free (tspad->tags);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* free the wrapper */
|
||||||
|
g_free (tspad);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
mpegts_parse_pad_removed (GstElement * element, GstPad * pad)
|
||||||
|
{
|
||||||
|
MpegTSParsePad *tspad;
|
||||||
|
MpegTSParse2 *parse = GST_MPEGTS_PARSE (element);
|
||||||
|
|
||||||
|
if (gst_pad_get_direction (pad) == GST_PAD_SINK)
|
||||||
|
return;
|
||||||
|
|
||||||
|
tspad = (MpegTSParsePad *) gst_pad_get_element_private (pad);
|
||||||
|
mpegts_parse_destroy_tspad (parse, tspad);
|
||||||
|
|
||||||
|
if (GST_ELEMENT_CLASS (parent_class)->pad_removed)
|
||||||
|
GST_ELEMENT_CLASS (parent_class)->pad_removed (element, pad);
|
||||||
|
}
|
||||||
|
|
||||||
|
static GstPad *
|
||||||
|
mpegts_parse_request_new_pad (GstElement * element, GstPadTemplate * template,
|
||||||
|
const gchar * unused)
|
||||||
|
{
|
||||||
|
MpegTSParse2 *parse;
|
||||||
|
gchar *name;
|
||||||
|
GstPad *pad;
|
||||||
|
|
||||||
|
g_return_val_if_fail (template != NULL, NULL);
|
||||||
|
g_return_val_if_fail (GST_IS_MPEGTS_PARSE (element), NULL);
|
||||||
|
|
||||||
|
parse = GST_MPEGTS_PARSE (element);
|
||||||
|
|
||||||
|
GST_OBJECT_LOCK (element);
|
||||||
|
name = g_strdup_printf ("src%d", parse->req_pads++);
|
||||||
|
GST_OBJECT_UNLOCK (element);
|
||||||
|
|
||||||
|
pad = mpegts_parse_create_tspad (parse, name)->pad;
|
||||||
|
gst_pad_set_active (pad, TRUE);
|
||||||
|
gst_element_add_pad (element, pad);
|
||||||
|
g_free (name);
|
||||||
|
|
||||||
|
return pad;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
mpegts_parse_release_pad (GstElement * element, GstPad * pad)
|
||||||
|
{
|
||||||
|
g_return_if_fail (GST_IS_MPEGTS_PARSE (element));
|
||||||
|
|
||||||
|
gst_pad_set_active (pad, FALSE);
|
||||||
|
/* we do the cleanup in GstElement::pad-removed */
|
||||||
|
gst_element_remove_pad (element, pad);
|
||||||
|
}
|
||||||
|
|
||||||
|
static GstFlowReturn
|
||||||
|
mpegts_parse_tspad_push_section (MpegTSParse2 * parse, MpegTSParsePad * tspad,
|
||||||
|
MpegTSPacketizerSection * section, GstBuffer * buffer)
|
||||||
|
{
|
||||||
|
GstFlowReturn ret = GST_FLOW_NOT_LINKED;
|
||||||
|
gboolean to_push = TRUE;
|
||||||
|
|
||||||
|
if (tspad->program_number != -1) {
|
||||||
|
if (tspad->program) {
|
||||||
|
/* we push all sections to all pads except PMTs which we
|
||||||
|
* only push to pads meant to receive that program number */
|
||||||
|
if (section->table_id == 0x02) {
|
||||||
|
/* PMT */
|
||||||
|
if (section->subtable_extension != tspad->program_number)
|
||||||
|
to_push = FALSE;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
/* there's a program filter on the pad but the PMT for the program has not
|
||||||
|
* been parsed yet, ignore the pad until we get a PMT */
|
||||||
|
to_push = FALSE;
|
||||||
|
ret = GST_FLOW_OK;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
GST_DEBUG_OBJECT (parse,
|
||||||
|
"pushing section: %d program number: %d table_id: %d", to_push,
|
||||||
|
tspad->program_number, section->table_id);
|
||||||
|
if (to_push) {
|
||||||
|
ret = gst_pad_push (tspad->pad, buffer);
|
||||||
|
} else {
|
||||||
|
gst_buffer_unref (buffer);
|
||||||
|
if (gst_pad_is_linked (tspad->pad))
|
||||||
|
ret = GST_FLOW_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static GstFlowReturn
|
||||||
|
mpegts_parse_tspad_push (MpegTSParse2 * parse, MpegTSParsePad * tspad,
|
||||||
|
guint16 pid, GstBuffer * buffer)
|
||||||
|
{
|
||||||
|
GstFlowReturn ret = GST_FLOW_NOT_LINKED;
|
||||||
|
MpegTSBaseStream **pad_pids = NULL;
|
||||||
|
|
||||||
|
if (tspad->program_number != -1) {
|
||||||
|
if (tspad->program) {
|
||||||
|
MpegTSBaseProgram *bp = (MpegTSBaseProgram *) tspad->program;
|
||||||
|
pad_pids = bp->streams;
|
||||||
|
if (bp->tags) {
|
||||||
|
gst_element_found_tags_for_pad (GST_ELEMENT_CAST (parse), tspad->pad,
|
||||||
|
bp->tags);
|
||||||
|
bp->tags = NULL;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
/* there's a program filter on the pad but the PMT for the program has not
|
||||||
|
* been parsed yet, ignore the pad until we get a PMT */
|
||||||
|
gst_buffer_unref (buffer);
|
||||||
|
ret = GST_FLOW_OK;
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (pad_pids == NULL || pad_pids[pid]) {
|
||||||
|
/* push if there's no filter or if the pid is in the filter */
|
||||||
|
ret = gst_pad_push (tspad->pad, buffer);
|
||||||
|
} else {
|
||||||
|
gst_buffer_unref (buffer);
|
||||||
|
if (gst_pad_is_linked (tspad->pad))
|
||||||
|
ret = GST_FLOW_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
out:
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
pad_clear_for_push (GstPad * pad, MpegTSParse2 * parse)
|
||||||
|
{
|
||||||
|
MpegTSParsePad *tspad = (MpegTSParsePad *) gst_pad_get_element_private (pad);
|
||||||
|
|
||||||
|
tspad->flow_return = GST_FLOW_NOT_LINKED;
|
||||||
|
tspad->pushed = FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static GstFlowReturn
|
||||||
|
mpegts_parse_push (MpegTSBase * base, MpegTSPacketizerPacket * packet,
|
||||||
|
MpegTSPacketizerSection * section)
|
||||||
|
{
|
||||||
|
MpegTSParse2 *parse = (MpegTSParse2 *) base;
|
||||||
|
guint32 pads_cookie;
|
||||||
|
gboolean done = FALSE;
|
||||||
|
GstPad *pad = NULL;
|
||||||
|
MpegTSParsePad *tspad;
|
||||||
|
guint16 pid;
|
||||||
|
GstBuffer *buffer;
|
||||||
|
GstFlowReturn ret;
|
||||||
|
GList *srcpads;
|
||||||
|
|
||||||
|
if (G_UNLIKELY (parse->need_sync_program_pads))
|
||||||
|
mpegts_parse_sync_program_pads (parse);
|
||||||
|
|
||||||
|
pid = packet->pid;
|
||||||
|
buffer = gst_buffer_make_metadata_writable (packet->buffer);
|
||||||
|
/* we have the same caps on all the src pads */
|
||||||
|
gst_buffer_set_caps (buffer, base->packetizer->caps);
|
||||||
|
|
||||||
|
GST_OBJECT_LOCK (parse);
|
||||||
|
/* clear tspad->pushed on pads */
|
||||||
|
g_list_foreach (GST_ELEMENT_CAST (parse)->srcpads,
|
||||||
|
(GFunc) pad_clear_for_push, parse);
|
||||||
|
if (GST_ELEMENT_CAST (parse)->srcpads)
|
||||||
|
ret = GST_FLOW_NOT_LINKED;
|
||||||
|
else
|
||||||
|
ret = GST_FLOW_OK;
|
||||||
|
|
||||||
|
/* Get cookie and source pads list */
|
||||||
|
pads_cookie = GST_ELEMENT_CAST (parse)->pads_cookie;
|
||||||
|
srcpads = GST_ELEMENT_CAST (parse)->srcpads;
|
||||||
|
if (G_LIKELY (srcpads)) {
|
||||||
|
pad = GST_PAD_CAST (srcpads->data);
|
||||||
|
g_object_ref (pad);
|
||||||
|
}
|
||||||
|
GST_OBJECT_UNLOCK (parse);
|
||||||
|
|
||||||
|
while (pad && !done) {
|
||||||
|
tspad = gst_pad_get_element_private (pad);
|
||||||
|
|
||||||
|
if (G_LIKELY (!tspad->pushed)) {
|
||||||
|
/* ref the buffer as gst_pad_push takes a ref but we want to reuse the
|
||||||
|
* same buffer for next pushes */
|
||||||
|
gst_buffer_ref (buffer);
|
||||||
|
if (section) {
|
||||||
|
tspad->flow_return =
|
||||||
|
mpegts_parse_tspad_push_section (parse, tspad, section, buffer);
|
||||||
|
} else {
|
||||||
|
tspad->flow_return =
|
||||||
|
mpegts_parse_tspad_push (parse, tspad, pid, buffer);
|
||||||
|
}
|
||||||
|
tspad->pushed = TRUE;
|
||||||
|
|
||||||
|
if (G_UNLIKELY (tspad->flow_return != GST_FLOW_OK
|
||||||
|
&& tspad->flow_return != GST_FLOW_NOT_LINKED)) {
|
||||||
|
/* return the error upstream */
|
||||||
|
ret = tspad->flow_return;
|
||||||
|
done = TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ret == GST_FLOW_NOT_LINKED)
|
||||||
|
ret = tspad->flow_return;
|
||||||
|
|
||||||
|
g_object_unref (pad);
|
||||||
|
|
||||||
|
if (G_UNLIKELY (!done)) {
|
||||||
|
GST_OBJECT_LOCK (parse);
|
||||||
|
if (G_UNLIKELY (pads_cookie != GST_ELEMENT_CAST (parse)->pads_cookie)) {
|
||||||
|
/* resync */
|
||||||
|
GST_DEBUG ("resync");
|
||||||
|
pads_cookie = GST_ELEMENT_CAST (parse)->pads_cookie;
|
||||||
|
srcpads = GST_ELEMENT_CAST (parse)->srcpads;
|
||||||
|
} else {
|
||||||
|
GST_DEBUG ("getting next pad");
|
||||||
|
/* Get next pad */
|
||||||
|
srcpads = g_list_next (srcpads);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (srcpads) {
|
||||||
|
pad = GST_PAD_CAST (srcpads->data);
|
||||||
|
g_object_ref (pad);
|
||||||
|
} else
|
||||||
|
done = TRUE;
|
||||||
|
GST_OBJECT_UNLOCK (parse);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
gst_buffer_unref (buffer);
|
||||||
|
packet->buffer = NULL;
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
mpegts_parse_program_started (MpegTSBase * base, MpegTSBaseProgram * program)
|
||||||
|
{
|
||||||
|
MpegTSParse2 *parse = GST_MPEGTS_PARSE (base);
|
||||||
|
MpegTSParseProgram *parseprogram = (MpegTSParseProgram *) program;
|
||||||
|
if (parseprogram->selected == 2) {
|
||||||
|
parse->pads_to_add =
|
||||||
|
g_list_append (parse->pads_to_add,
|
||||||
|
mpegts_parse_activate_program (parse, parseprogram));
|
||||||
|
parseprogram->selected = 1;
|
||||||
|
parse->need_sync_program_pads = TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
mpegts_parse_program_stopped (MpegTSBase * base, MpegTSBaseProgram * program)
|
||||||
|
{
|
||||||
|
MpegTSParse2 *parse = GST_MPEGTS_PARSE (base);
|
||||||
|
MpegTSParseProgram *parseprogram = (MpegTSParseProgram *) program;
|
||||||
|
|
||||||
|
if (parseprogram->active) {
|
||||||
|
parse->pads_to_remove =
|
||||||
|
g_list_append (parse->pads_to_remove,
|
||||||
|
mpegts_parse_deactivate_program (parse, parseprogram));
|
||||||
|
parse->need_sync_program_pads = TRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
mpegts_parse_src_pad_query (GstPad * pad, GstQuery * query)
|
||||||
|
{
|
||||||
|
MpegTSParse2 *parse = GST_MPEGTS_PARSE (gst_pad_get_parent (pad));
|
||||||
|
gboolean res;
|
||||||
|
|
||||||
|
switch (GST_QUERY_TYPE (query)) {
|
||||||
|
case GST_QUERY_LATENCY:
|
||||||
|
{
|
||||||
|
if ((res = gst_pad_peer_query (((MpegTSBase *) parse)->sinkpad, query))) {
|
||||||
|
gboolean is_live;
|
||||||
|
GstClockTime min_latency, max_latency;
|
||||||
|
|
||||||
|
gst_query_parse_latency (query, &is_live, &min_latency, &max_latency);
|
||||||
|
if (is_live) {
|
||||||
|
min_latency += TS_LATENCY * GST_MSECOND;
|
||||||
|
if (max_latency != GST_CLOCK_TIME_NONE)
|
||||||
|
max_latency += TS_LATENCY * GST_MSECOND;
|
||||||
|
}
|
||||||
|
|
||||||
|
gst_query_set_latency (query, is_live, min_latency, max_latency);
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
res = gst_pad_query_default (pad, query);
|
||||||
|
}
|
||||||
|
gst_object_unref (parse);
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
gboolean
|
||||||
|
gst_mpegtsparse_plugin_init (GstPlugin * plugin)
|
||||||
|
{
|
||||||
|
GST_DEBUG_CATEGORY_INIT (mpegts_parse_debug, "tsparse", 0,
|
||||||
|
"MPEG transport stream parser");
|
||||||
|
|
||||||
|
gst_mpegtsdesc_init_debug ();
|
||||||
|
|
||||||
|
return gst_element_register (plugin, "tsparse",
|
||||||
|
GST_RANK_NONE, GST_TYPE_MPEGTS_PARSE);
|
||||||
|
}
|
71
gst/mpegtsdemux/mpegtsparse.h
Normal file
71
gst/mpegtsdemux/mpegtsparse.h
Normal file
|
@ -0,0 +1,71 @@
|
||||||
|
/*
|
||||||
|
* mpegts_parse.h - GStreamer MPEG transport stream parser
|
||||||
|
* Copyright (C) 2007 Alessandro Decina
|
||||||
|
*
|
||||||
|
* Authors:
|
||||||
|
* Alessandro Decina <alessandro@nnva.org>
|
||||||
|
*
|
||||||
|
* This library is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU Library General Public
|
||||||
|
* License as published by the Free Software Foundation; either
|
||||||
|
* version 2 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Library General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Library General Public
|
||||||
|
* License along with this library; if not, write to the
|
||||||
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||||
|
* Boston, MA 02111-1307, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#ifndef GST_MPEG_TS_PARSE_H
|
||||||
|
#define GST_MPEG_TS_PARSE_H
|
||||||
|
|
||||||
|
#include <gst/gst.h>
|
||||||
|
#include "mpegtsbase.h"
|
||||||
|
#include "mpegtspacketizer.h"
|
||||||
|
|
||||||
|
G_BEGIN_DECLS
|
||||||
|
|
||||||
|
#define GST_TYPE_MPEGTS_PARSE \
|
||||||
|
(mpegts_parse_get_type())
|
||||||
|
#define GST_MPEGTS_PARSE(obj) \
|
||||||
|
(G_TYPE_CHECK_INSTANCE_CAST((obj),GST_TYPE_MPEGTS_PARSE,MpegTSParse2))
|
||||||
|
#define GST_MPEGTS_PARSE_CLASS(klass) \
|
||||||
|
(G_TYPE_CHECK_CLASS_CAST((klass),GST_TYPE_MPEGTS_PARSE,MpegTSParse2Class))
|
||||||
|
#define GST_IS_MPEGTS_PARSE(obj) \
|
||||||
|
(G_TYPE_CHECK_INSTANCE_TYPE((obj),GST_TYPE_MPEGTS_PARSE))
|
||||||
|
#define GST_IS_MPEGTS_PARSE_CLASS(klass) \
|
||||||
|
(G_TYPE_CHECK_CLASS_TYPE((klass),GST_TYPE_MPEGTS_PARSE))
|
||||||
|
|
||||||
|
typedef struct _MpegTSParse2 MpegTSParse2;
|
||||||
|
typedef struct _MpegTSParse2Class MpegTSParse2Class;
|
||||||
|
|
||||||
|
struct _MpegTSParse2 {
|
||||||
|
MpegTSBase parent;
|
||||||
|
|
||||||
|
/* the following vars must be protected with the OBJECT_LOCK as they can be
|
||||||
|
* accessed from the application thread and the streaming thread */
|
||||||
|
gchar *program_numbers;
|
||||||
|
GList *pads_to_add;
|
||||||
|
GList *pads_to_remove;
|
||||||
|
guint req_pads;
|
||||||
|
|
||||||
|
gboolean need_sync_program_pads;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct _MpegTSParse2Class {
|
||||||
|
MpegTSBaseClass parent_class;
|
||||||
|
};
|
||||||
|
|
||||||
|
GType mpegts_parse_get_type(void);
|
||||||
|
|
||||||
|
gboolean gst_mpegtsparse_plugin_init (GstPlugin * plugin);
|
||||||
|
|
||||||
|
G_END_DECLS
|
||||||
|
|
||||||
|
#endif /* GST_MPEG_TS_PARSE_H */
|
1491
gst/mpegtsdemux/tsdemux.c
Normal file
1491
gst/mpegtsdemux/tsdemux.c
Normal file
File diff suppressed because it is too large
Load diff
78
gst/mpegtsdemux/tsdemux.h
Normal file
78
gst/mpegtsdemux/tsdemux.h
Normal file
|
@ -0,0 +1,78 @@
|
||||||
|
/*
|
||||||
|
* tsdemux - GStreamer MPEG transport stream demuxer
|
||||||
|
* Copyright (C) 2009 Zaheer Abbas Merali
|
||||||
|
* 2010 Edward Hervey
|
||||||
|
*
|
||||||
|
* Authors:
|
||||||
|
* Zaheer Abbas Merali <zaheerabbas at merali dot org>
|
||||||
|
* Edward Hervey <edward.hervey@collabora.co.uk>
|
||||||
|
*
|
||||||
|
* This library is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU Library General Public
|
||||||
|
* License as published by the Free Software Foundation; either
|
||||||
|
* version 2 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Library General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Library General Public
|
||||||
|
* License along with this library; if not, write to the
|
||||||
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||||
|
* Boston, MA 02111-1307, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#ifndef GST_TS_DEMUX_H
|
||||||
|
#define GST_TS_DEMUX_H
|
||||||
|
|
||||||
|
#include <gst/gst.h>
|
||||||
|
#include <gst/base/gstbytereader.h>
|
||||||
|
#include "mpegtsbase.h"
|
||||||
|
#include "mpegtspacketizer.h"
|
||||||
|
|
||||||
|
G_BEGIN_DECLS
|
||||||
|
#define GST_TYPE_TS_DEMUX \
|
||||||
|
(gst_ts_demux_get_type())
|
||||||
|
#define GST_TS_DEMUX(obj) \
|
||||||
|
(G_TYPE_CHECK_INSTANCE_CAST((obj),GST_TYPE_TS_DEMUX,GstTSDemux))
|
||||||
|
#define GST_TS_DEMUX_CLASS(klass) \
|
||||||
|
(G_TYPE_CHECK_CLASS_CAST((klass),GST_TYPE_TS_DEMUX,GstTSDemuxClass))
|
||||||
|
#define GST_IS_TS_DEMUX(obj) \
|
||||||
|
(G_TYPE_CHECK_INSTANCE_TYPE((obj),GST_TYPE_TS_DEMUX))
|
||||||
|
#define GST_IS_TS_DEMUX_CLASS(klass) \
|
||||||
|
(G_TYPE_CHECK_CLASS_TYPE((klass),GST_TYPE_TS_DEMUX))
|
||||||
|
#define GST_TS_DEMUX_GET_CLASS(obj) \
|
||||||
|
(G_TYPE_INSTANCE_GET_CLASS ((obj), GST_TYPE_TS_DEMUX, GstTSDemuxClass))
|
||||||
|
#define GST_TS_DEMUX_CAST(obj) ((GstTSDemux*) obj)
|
||||||
|
typedef struct _GstTSDemux GstTSDemux;
|
||||||
|
typedef struct _GstTSDemuxClass GstTSDemuxClass;
|
||||||
|
|
||||||
|
struct _GstTSDemux
|
||||||
|
{
|
||||||
|
MpegTSBase parent;
|
||||||
|
|
||||||
|
/* the following vars must be protected with the OBJECT_LOCK as they can be
|
||||||
|
* accessed from the application thread and the streaming thread */
|
||||||
|
guint program_number; /* Required program number (ignore:-1) */
|
||||||
|
gboolean emit_statistics;
|
||||||
|
|
||||||
|
/*< private >*/
|
||||||
|
MpegTSBaseProgram *program; /* Current program */
|
||||||
|
guint current_program_number;
|
||||||
|
gboolean need_newsegment;
|
||||||
|
GstClockTime duration; /* Total duration */
|
||||||
|
};
|
||||||
|
|
||||||
|
struct _GstTSDemuxClass
|
||||||
|
{
|
||||||
|
MpegTSBaseClass parent_class;
|
||||||
|
};
|
||||||
|
|
||||||
|
GType gst_ts_demux_get_type (void);
|
||||||
|
|
||||||
|
gboolean gst_ts_demux_plugin_init (GstPlugin * plugin);
|
||||||
|
|
||||||
|
G_END_DECLS
|
||||||
|
#endif /* GST_TS_DEMUX_H */
|
Loading…
Reference in a new issue