📄 ps.c
字号:
/***************************************************************************** * ps.c: MPEG PS (ISO/IEC 13818-1) / MPEG SYSTEM (ISO/IEC 1172-1) * multiplexer module for vlc ***************************************************************************** * Copyright (C) 2001, 2002 VideoLAN * $Id: ps.c 10731 2005-04-18 15:32:31Z gbazin $ * * Authors: Laurent Aimar <fenrir@via.ecp.fr> * Eric Petit <titer@videolan.org> * Gildas Bazin <gbazin@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"#include "iso_lang.h"/***************************************************************************** * Module descriptor *****************************************************************************/#define DTS_TEXT N_("DTS delay (ms)")#define DTS_LONGTEXT N_("This option will delay the DTS (decoding time " \ "stamps) and PTS (presentation timestamps) of the data in the " \ "stream, compared to the SCRs. This allows for some buffering inside " \ "the client decoder.")#define PES_SIZE_TEXT N_("PES maximum size")#define PES_SIZE_LONGTEXT N_("This option will set the maximum allowed PES "\ "size when producing the MPEG PS stream.")static int Open ( vlc_object_t * );static void Close ( vlc_object_t * );#define SOUT_CFG_PREFIX "sout-ps-"vlc_module_begin(); set_description( _("PS muxer") ); set_shortname( "MPEG-PS" ); set_category( CAT_SOUT ); set_subcategory( SUBCAT_SOUT_MUX ); set_capability( "sout mux", 50 ); add_shortcut( "ps" ); add_shortcut( "mpeg1" ); add_shortcut( "dvd" ); set_callbacks( Open, Close ); add_integer( SOUT_CFG_PREFIX "dts-delay", 200, NULL, DTS_TEXT, DTS_LONGTEXT, VLC_TRUE ); add_integer( SOUT_CFG_PREFIX "pes-max-size", PES_PAYLOAD_SIZE_MAX, NULL, PES_SIZE_TEXT, PES_SIZE_LONGTEXT, VLC_TRUE );vlc_module_end();/***************************************************************************** * Exported prototypes *****************************************************************************/static int Control ( sout_mux_t *, int, va_list );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 *****************************************************************************/static int MuxGetStream ( sout_mux_t *, int *, mtime_t * );static void MuxWritePackHeader ( sout_mux_t *, block_t **, mtime_t );static void MuxWriteSystemHeader( sout_mux_t *, block_t **, mtime_t );static void MuxWritePSM ( sout_mux_t *, block_t **, mtime_t );static void StreamIdInit ( vlc_bool_t *id, int i_range );static int StreamIdGet ( vlc_bool_t *id, int i_id_min, int i_id_max );static void StreamIdRelease ( vlc_bool_t *id, int i_id_min, int i_id );typedef struct ps_stream_s{ int i_stream_id; int i_stream_type; int i_max_buff_size; /* used in system header */ /* Language is iso639-2T */ uint8_t lang[3];} ps_stream_t;struct sout_mux_sys_t{ /* Which id are unused */ vlc_bool_t stream_id_mpga[16]; /* 0xc0 -> 0xcf */ vlc_bool_t stream_id_mpgv[16]; /* 0xe0 -> 0xef */ vlc_bool_t stream_id_a52[8]; /* 0x80 -> 0x87 <- FIXME I'm not sure */ vlc_bool_t stream_id_spu[32]; /* 0x20 -> 0x3f */ vlc_bool_t stream_id_dts[8]; /* 0x88 -> 0x8f */ vlc_bool_t stream_id_lpcm[16]; /* 0xa0 -> 0xaf */ int i_audio_bound; int i_video_bound; int i_pes_count; int i_system_header; int i_dts_delay; int i_rate_bound; /* units of 50 bytes/second */ int64_t i_instant_bitrate; int64_t i_instant_size; int64_t i_instant_dts; vlc_bool_t b_mpeg2; int i_pes_max_size; int i_psm_version; uint32_t crc32_table[256];};static const char *ppsz_sout_options[] = { "dts-delay", "pes-max-size", NULL};/***************************************************************************** * 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; vlc_value_t val; msg_Info( p_mux, "Open" ); sout_CfgParse( p_mux, SOUT_CFG_PREFIX, ppsz_sout_options, p_mux->p_cfg ); p_mux->pf_control = Control; p_mux->pf_addstream = AddStream; p_mux->pf_delstream = DelStream; p_mux->pf_mux = Mux; p_mux->p_sys = p_sys = malloc( sizeof( sout_mux_sys_t ) ); /* Init free stream id */ StreamIdInit( p_sys->stream_id_a52, 8 ); StreamIdInit( p_sys->stream_id_dts, 8 ); StreamIdInit( p_sys->stream_id_mpga, 16 ); StreamIdInit( p_sys->stream_id_mpgv, 16 ); StreamIdInit( p_sys->stream_id_lpcm, 16 ); StreamIdInit( p_sys->stream_id_spu, 32 ); p_sys->i_audio_bound = 0; p_sys->i_video_bound = 0; p_sys->i_system_header = 0; p_sys->i_pes_count = 0; p_sys->i_psm_version = 0; p_sys->i_instant_bitrate = 0; p_sys->i_instant_size = 0; p_sys->i_instant_dts = 0; p_sys->i_rate_bound = 0; p_sys->b_mpeg2 = !(p_mux->psz_mux && !strcmp( p_mux->psz_mux, "mpeg1" )); var_Get( p_mux, SOUT_CFG_PREFIX "dts-delay", &val ); p_sys->i_dts_delay = (int64_t)val.i_int * 1000; var_Get( p_mux, SOUT_CFG_PREFIX "pes-max-size", &val ); p_sys->i_pes_max_size = (int64_t)val.i_int; /* Initialise CRC32 table */ if( p_sys->b_mpeg2 ) { uint32_t i, j, k; for( i = 0; i < 256; i++ ) { k = 0; for( j = (i << 24) | 0x800000; j != 0x80000000; j <<= 1 ) k = (k << 1) ^ (((k ^ j) & 0x80000000) ? 0x04c11db7 : 0); p_sys->crc32_table[i] = k; } } 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; block_t *p_end; msg_Info( p_mux, "Close" ); p_end = block_New( p_mux, 4 ); p_end->p_buffer[0] = 0x00; p_end->p_buffer[1] = 0x00; p_end->p_buffer[2] = 0x01; p_end->p_buffer[3] = 0xb9; sout_AccessOutWrite( p_mux->p_access, p_end ); free( p_sys );}/***************************************************************************** * Control: *****************************************************************************/static int Control( sout_mux_t *p_mux, int i_query, va_list args ){ vlc_bool_t *pb_bool; char **ppsz; switch( i_query ) { case MUX_CAN_ADD_STREAM_WHILE_MUXING: pb_bool = (vlc_bool_t*)va_arg( args, vlc_bool_t * ); *pb_bool = VLC_TRUE; return VLC_SUCCESS; case MUX_GET_ADD_STREAM_WAIT: pb_bool = (vlc_bool_t*)va_arg( args, vlc_bool_t * ); *pb_bool = VLC_FALSE; return VLC_SUCCESS; case MUX_GET_MIME: ppsz = (char**)va_arg( args, char ** ); *ppsz = strdup( "video/mpeg" ); return VLC_SUCCESS; default: return VLC_EGENERIC; }}/***************************************************************************** * AddStream: *****************************************************************************/static int AddStream( sout_mux_t *p_mux, sout_input_t *p_input ){ sout_mux_sys_t *p_sys = p_mux->p_sys; ps_stream_t *p_stream; msg_Dbg( p_mux, "adding input codec=%4.4s", (char*)&p_input->p_fmt->i_codec ); p_input->p_sys = p_stream = malloc( sizeof( ps_stream_t ) ); p_stream->i_stream_type = 0x81; /* Init this new stream */ switch( p_input->p_fmt->i_codec ) { case VLC_FOURCC( 'm', 'p', '1', 'v' ): p_stream->i_stream_id = StreamIdGet( p_sys->stream_id_mpgv, 0xe0, 0xef ); p_stream->i_stream_type = 0x01; /* ISO/IEC 11172 Video */ break; case VLC_FOURCC( 'm', 'p', '2', 'v' ): case VLC_FOURCC( 'm', 'p', 'g', 'v' ): p_stream->i_stream_id = StreamIdGet( p_sys->stream_id_mpgv, 0xe0, 0xef ); p_stream->i_stream_type = 0x02; /* ISO/IEC 13818 Video */ break; case VLC_FOURCC( 'm', 'p', '4', 'v' ): p_stream->i_stream_id = StreamIdGet( p_sys->stream_id_mpgv, 0xe0, 0xef ); p_stream->i_stream_type = 0x10; break; case VLC_FOURCC( 'h', '2', '6', '4' ): p_stream->i_stream_id = StreamIdGet( p_sys->stream_id_mpgv, 0xe0, 0xef ); p_stream->i_stream_type = 0x1b; break; case VLC_FOURCC( 'l', 'p', 'c', 'm' ): p_stream->i_stream_id = 0xbd00 | StreamIdGet( p_sys->stream_id_lpcm, 0xa0, 0xaf ); break; case VLC_FOURCC( 'd', 't', 's', ' ' ): p_stream->i_stream_id = 0xbd00 | StreamIdGet( p_sys->stream_id_dts, 0x88, 0x8f ); break; case VLC_FOURCC( 'a', '5', '2', ' ' ): p_stream->i_stream_id = 0xbd00 | StreamIdGet( p_sys->stream_id_a52, 0x80, 0x87 ); break; case VLC_FOURCC( 'm', 'p', 'g', 'a' ): p_stream->i_stream_id = StreamIdGet( p_sys->stream_id_mpga, 0xc0, 0xcf ); p_stream->i_stream_type = 0x03; /* ISO/IEC 11172 Audio */ break; case VLC_FOURCC( 'm', 'p', '4', 'a' ): p_stream->i_stream_id = StreamIdGet( p_sys->stream_id_mpga, 0xc0, 0xcf ); p_stream->i_stream_type = 0x0f; break; case VLC_FOURCC( 's', 'p', 'u', ' ' ): p_stream->i_stream_id = 0xbd00 | StreamIdGet( p_sys->stream_id_spu, 0x20, 0x3f ); break; default: goto error; } if( p_stream->i_stream_id < 0 ) goto error; if( p_input->p_fmt->i_cat == AUDIO_ES ) { p_sys->i_audio_bound++; p_stream->i_max_buff_size = 4 * 1024; } else if( p_input->p_fmt->i_cat == VIDEO_ES ) { p_sys->i_video_bound++; p_stream->i_max_buff_size = 400 * 1024; /* FIXME -- VCD uses 46, SVCD uses 230, ffmpeg has 230 with a note that it is small */ } else { /* FIXME -- what's valid for not audio or video? */ p_stream->i_max_buff_size = 4 * 1024; } /* Try to set a sensible default value for the instant bitrate */ p_sys->i_instant_bitrate += p_input->p_fmt->i_bitrate + 1000/* overhead */; /* FIXME -- spec requires an upper limit rate boundary in the system header; our codecs are VBR; using 2x nominal rate, convert to 50 bytes/sec */ p_sys->i_rate_bound += p_input->p_fmt->i_bitrate * 2 / (8 * 50); p_sys->i_psm_version++; p_stream->lang[0] = p_stream->lang[1] = p_stream->lang[2] = 0; if( p_input->p_fmt->psz_language ) { char *psz = p_input->p_fmt->psz_language; const iso639_lang_t *pl = NULL; if( strlen( psz ) == 2 ) { pl = GetLang_1( psz ); } else if( strlen( psz ) == 3 ) { pl = GetLang_2B( psz ); if( !strcmp( pl->psz_iso639_1, "??" ) ) { pl = GetLang_2T( psz ); } } if( pl && strcmp( pl->psz_iso639_1, "??" ) ) { p_stream->lang[0] = pl->psz_iso639_2T[0]; p_stream->lang[1] = pl->psz_iso639_2T[1]; p_stream->lang[2] = pl->psz_iso639_2T[2]; msg_Dbg( p_mux, " - lang=%c%c%c", p_stream->lang[0], p_stream->lang[1], p_stream->lang[2] ); } } return VLC_SUCCESS;error: free( p_stream ); return VLC_EGENERIC;}/***************************************************************************** * DelStream: *****************************************************************************/static int DelStream( sout_mux_t *p_mux, sout_input_t *p_input ){ sout_mux_sys_t *p_sys = p_mux->p_sys; ps_stream_t *p_stream =(ps_stream_t*)p_input->p_sys; msg_Dbg( p_mux, "removing input" ); switch( p_input->p_fmt->i_codec ) { case VLC_FOURCC( 'm', 'p', 'g', 'v' ): StreamIdRelease( p_sys->stream_id_mpgv, 0xe0, p_stream->i_stream_id ); break; case VLC_FOURCC( 'l', 'p', 'c', 'm' ): StreamIdRelease( p_sys->stream_id_lpcm, 0xa0, p_stream->i_stream_id&0xff ); break; case VLC_FOURCC( 'd', 't', 's', ' ' ): StreamIdRelease( p_sys->stream_id_dts, 0x88, p_stream->i_stream_id&0xff ); break; case VLC_FOURCC( 'a', '5', '2', ' ' ): StreamIdRelease( p_sys->stream_id_a52, 0x80, p_stream->i_stream_id&0xff ); break; case VLC_FOURCC( 'm', 'p', 'g', 'a' ): StreamIdRelease( p_sys->stream_id_mpga, 0xc0, p_stream->i_stream_id ); break; case VLC_FOURCC( 's', 'p', 'u', ' ' ): StreamIdRelease( p_sys->stream_id_spu, 0x20, p_stream->i_stream_id&0xff ); break; default: /* Never reached */
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -