📄 ts.c
字号:
/***************************************************************************** * ts.c: MPEG-II TS Muxer ***************************************************************************** * Copyright (C) 2001, 2002 VideoLAN * $Id: ts.c,v 1.26 2003/08/10 14:23:15 gbazin Exp $ * * Authors: Laurent Aimar <fenrir@via.ecp.fr> * Eric Petit <titer@videolan.org> * * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111, USA. *****************************************************************************//***************************************************************************** * Preamble *****************************************************************************/#include <stdlib.h>#include <vlc/vlc.h>#include <vlc/input.h>#include <vlc/sout.h>#include "codecs.h"#include "bits.h"#include "pes.h"#if defined MODULE_NAME_IS_mux_ts_dvbpsi# ifdef HAVE_DVBPSI_DR_H# include <dvbpsi/dvbpsi.h># include <dvbpsi/descriptor.h># include <dvbpsi/pat.h># include <dvbpsi/pmt.h># include <dvbpsi/dr.h># include <dvbpsi/psi.h># else# include "dvbpsi.h"# include "descriptor.h"# include "tables/pat.h"# include "tables/pmt.h"# include "descriptors/dr.h"# include "psi.h"# endif#endif/* * TODO: * - check PCR frequency requirement * - check PAT/PMT " " * - check PCR/PCR "soft" * * - remove creation of PAT/PMT without dvbpsi * - ? * FIXME: * - subtitle support is far from perfect. I expect some subtitles drop * if they arrive a bit late * (We cannot rely on the fact that the fifo should be full) *//***************************************************************************** * Module descriptor *****************************************************************************/static int Open ( vlc_object_t * );static void Close ( vlc_object_t * );vlc_module_begin();#if defined MODULE_NAME_IS_mux_ts set_description( _("TS muxer") ); set_capability( "sout mux", 100 ); add_shortcut( "ts" ); add_shortcut( "ts_nodvbpsi" );#elif defined MODULE_NAME_IS_mux_ts_dvbpsi set_description( _("TS muxer (libdvbpsi)") ); set_capability( "sout mux", 120 ); add_shortcut( "ts" ); add_shortcut( "ts_dvbpsi" );#endif set_callbacks( Open, Close );vlc_module_end();/***************************************************************************** * Exported prototypes *****************************************************************************/static int Capability(sout_mux_t *, int, void *, void * );static int AddStream( sout_mux_t *, sout_input_t * );static int DelStream( sout_mux_t *, sout_input_t * );static int Mux ( sout_mux_t * );/***************************************************************************** * Local prototypes *****************************************************************************/#define SOUT_BUFFER_FLAGS_PRIVATE_PCR_SOFT ( 1 << SOUT_BUFFER_FLAGS_PRIVATE_SHIFT )typedef struct{ int i_depth; sout_buffer_t *p_first; sout_buffer_t **pp_last;} sout_buffer_chain_t;static inline void BufferChainInit ( sout_buffer_chain_t *c ){ c->i_depth = 0; c->p_first = NULL; c->pp_last = &c->p_first;}static inline void BufferChainAppend( sout_buffer_chain_t *c, sout_buffer_t *b ){ *c->pp_last = b; c->i_depth++; while( b->p_next ) { b = b->p_next; c->i_depth++; } c->pp_last = &b->p_next;}static inline sout_buffer_t *BufferChainGet( sout_buffer_chain_t *c ){ sout_buffer_t *b = c->p_first; if( b ) { c->i_depth--; c->p_first = b->p_next; if( c->p_first == NULL ) { c->pp_last = &c->p_first; } b->p_next = NULL; } return b;}typedef struct ts_stream_s{ int i_pid; int i_stream_type; int i_stream_id; int i_continuity_counter; /* to be used for carriege of DIV3 */ vlc_fourcc_t i_bih_codec; int i_bih_width, i_bih_height; /* Specific to mpeg4 in mpeg2ts */ int i_es_id; int i_decoder_specific_info; uint8_t *p_decoder_specific_info; /* for TS building */ sout_buffer_chain_t chain_ts;} ts_stream_t;struct sout_mux_sys_t{ int i_pcr_pid; sout_input_t *p_pcr_input; int i_stream_id_mpga; int i_stream_id_mpgv; int i_stream_id_a52; int i_audio_bound; int i_video_bound; int i_pid_free; // first usable pid int i_pat_version_number; ts_stream_t pat; int i_pmt_version_number; ts_stream_t pmt; // Up to now only one program int i_mpeg4_streams; int i_null_continuity_counter; /* Needed ? */ /* for TS building */ int64_t i_bitrate_min; int64_t i_bitrate_max; int64_t i_pcr_delay; int64_t i_pcr_soft_delay; mtime_t i_pcr; /* last PCR emited (for pcr-soft) */ mtime_t i_dts; mtime_t i_length; sout_buffer_chain_t chain_ts;};/* Reserve a pid and return it */static int AllocatePID( sout_mux_sys_t *p_sys ){ return( ++p_sys->i_pid_free );}static void GetPAT( sout_mux_t *p_mux, sout_buffer_chain_t *c );static void GetPMT( sout_mux_t *p_mux, sout_buffer_chain_t *c );static int TSFill ( sout_mux_t *, sout_input_t * );static void PEStoTS ( sout_instance_t *, sout_buffer_chain_t *, sout_buffer_t *, ts_stream_t *, vlc_bool_t );static void TSSetDate( sout_buffer_chain_t *, mtime_t, mtime_t );static void TSSetConstraints( sout_mux_t*, sout_buffer_chain_t *, mtime_t i_length, int i_bitrate_min, int i_bitrate_max );#if !defined( HAVE_ATOLL )/* Et oui y'a des systemes de MERDE (ex: OS X, Solaris) qui ne l'ont pas :((( */static long long atoll(const char *str){ long long i_value = 0; int sign = 1; if( *str == '-' ) { sign = -1; } while( *str >= '0' && *str <= '9' ) { i_value = i_value * 10 + ( *str - '0' ); } return i_value * sign;}#endif/***************************************************************************** * Open: *****************************************************************************/static int Open( vlc_object_t *p_this ){ sout_mux_t *p_mux =(sout_mux_t*)p_this; sout_mux_sys_t *p_sys; char *val; msg_Dbg( p_mux, "Open" ); p_sys = malloc( sizeof( sout_mux_sys_t ) ); p_mux->pf_capacity = Capability; p_mux->pf_addstream = AddStream; p_mux->pf_delstream = DelStream; p_mux->pf_mux = Mux; p_mux->p_sys = p_sys; p_mux->i_preheader = 30; // really enough for a pes header srand( (uint32_t)mdate() ); p_sys->i_stream_id_mpga = 0xc0; p_sys->i_stream_id_a52 = 0x80; p_sys->i_stream_id_mpgv = 0xe0; p_sys->i_audio_bound = 0; p_sys->i_video_bound = 0; p_sys->i_pat_version_number = rand() % 32; p_sys->pat.i_pid = 0; p_sys->pat.i_continuity_counter = 0; p_sys->i_pmt_version_number = rand() % 32; p_sys->pmt.i_pid = 0x42; p_sys->pmt.i_continuity_counter = 0; p_sys->i_pid_free = 0x43; p_sys->i_pcr_pid = 0x1fff; p_sys->p_pcr_input = NULL; p_sys->i_mpeg4_streams = 0; p_sys->i_null_continuity_counter = 0; /* Allow to create constrained stream */ p_sys->i_bitrate_min = 0; p_sys->i_bitrate_max = 0; if( ( val = sout_cfg_find_value( p_mux->p_cfg, "bmin" ) ) ) { p_sys->i_bitrate_min = atoll( val ); } if( ( val = sout_cfg_find_value( p_mux->p_cfg, "bmax" ) ) ) { p_sys->i_bitrate_max = atoll( val ); } if( p_sys->i_bitrate_min > 0 && p_sys->i_bitrate_max > 0 && p_sys->i_bitrate_min > p_sys->i_bitrate_max ) { msg_Err( p_mux, "incompatible minimum and maximum bitrate, " "disabling bitrate control" ); p_sys->i_bitrate_min = 0; p_sys->i_bitrate_max = 0; } p_sys->i_pcr_delay = 100000; if( ( val = sout_cfg_find_value( p_mux->p_cfg, "pcr" ) ) ) { p_sys->i_pcr_delay = (int64_t)atoi( val ) * 1000; if( p_sys->i_pcr_delay <= 0 ) { msg_Err( p_mux, "invalid pcr delay ("I64Fd"ms) reseting to 100ms", p_sys->i_pcr_delay / 1000 ); p_sys->i_pcr_delay = 100000; } } p_sys->i_pcr_soft_delay = 0; if( ( val = sout_cfg_find_value( p_mux->p_cfg, "pcr-soft" ) ) ) { p_sys->i_pcr_soft_delay = (int64_t)atoi( val ) * 1000; if( p_sys->i_pcr_soft_delay <= 0 || p_sys->i_pcr_soft_delay >= p_sys->i_pcr_delay ) { msg_Err( p_mux, "invalid pcr-soft delay ("I64Fd"ms) disabled", p_sys->i_pcr_soft_delay / 1000 ); p_sys->i_pcr_soft_delay = 0; } } msg_Dbg( p_mux, "pcr_delay="I64Fd" pcr_soft_delay="I64Fd, p_sys->i_pcr_delay, p_sys->i_pcr_soft_delay ); /* for TS g閚閞ation */ p_sys->i_pcr = 0; p_sys->i_dts = 0; p_sys->i_length = 0; BufferChainInit( &p_sys->chain_ts ); return VLC_SUCCESS;}/***************************************************************************** * Close: *****************************************************************************/static void Close( vlc_object_t * p_this ){ sout_mux_t *p_mux = (sout_mux_t*)p_this; sout_mux_sys_t *p_sys = p_mux->p_sys; sout_buffer_t *p_data; msg_Dbg( p_mux, "Close" ); /* Empty TS buffer */ while( ( p_data = BufferChainGet( &p_sys->chain_ts ) ) ) { sout_BufferDelete( p_mux->p_sout, p_data ); } free( p_sys );}/***************************************************************************** * Capability: *****************************************************************************/static int Capability( sout_mux_t *p_mux, int i_query, void *p_args, void *p_answer ){ switch( i_query ) { case SOUT_MUX_CAP_GET_ADD_STREAM_ANY_TIME: *(vlc_bool_t*)p_answer = VLC_TRUE; return( SOUT_MUX_CAP_ERR_OK );
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -