⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 rtp-mpv.c

📁 spook是一个linux下开源的流媒体服务器
💻 C
字号:
/* * Copyright (C) 2004 Nathan Lutchansky <lutchann@litech.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-1307, USA. */#include <sys/types.h>#include <stdlib.h>#include <stdio.h>#include <string.h>#include <unistd.h>#include <sys/time.h>#include <sys/socket.h>#include <netinet/in.h>#include <fcntl.h>#include <errno.h>#include <event.h>#include <log.h>#include <frame.h>#include <stream.h>#include <outputs.h>#include <rtp.h>#include <rtp_media.h>#include <conf_parse.h>static const int frame_rate_tab[][2] = {	{ 0, 0 },	{ 1001, 24000 },	{ 1, 24 },	{ 1, 25 },	{ 1001, 30000 },	{ 1, 30 },	{ 1, 50 },	{ 1001, 60000 },	{ 1, 60 },	{ 0, 0 },	{ 0, 0 },	{ 0, 0 },	{ 0, 0 },	{ 0, 0 },	{ 0, 0 },	{ 0, 0 }};struct rtp_mpv {	struct {		unsigned char *d;		int len;	} blk[48];	int blk_count;	int temporal_reference;	int picture_type;	int vectors;	unsigned char vsh[4096];	int vsh_len;	int init_done;	int ts_incr;	unsigned int timestamp;};static int find_next_code( unsigned char *d, int len ){	int i;	/* Start at 1 to ignore the code sitting at d[0] */	for( i = 1; i < len - 2; ++i )		if( d[i] == 0x00 && d[i+1] == 0x00 && d[i+2] == 0x01 )			return i;	return 0;}static unsigned int get_field( unsigned char *d, int bits, int *offset ){	unsigned int v = 0;	int i;	for( i = 0; i < bits; )	{		if( bits - i >= 8 && *offset % 8 == 0 )		{			v <<= 8;			v |= d[*offset/8];			i += 8;			*offset += 8;		} else		{			v <<= 1;			v |= ( d[*offset/8] >> ( 7 - *offset % 8 ) ) & 1;			++i;			++(*offset);		}	}	return v;}static void parse_video_sequence_header( struct rtp_mpv *out,				unsigned char *d, int len ){	int off = 32 + 12 + 12 + 4, frc;	frc = get_field( d, 4, &off );	out->ts_incr = 90000 * frame_rate_tab[frc][0] / frame_rate_tab[frc][1];	out->init_done = 1;}static void parse_picture_header( struct rtp_mpv *out,				unsigned char *d, int len ){	int off = 32;	if( ! out->init_done ) return;	out->timestamp += out->ts_incr;	out->temporal_reference = get_field( d, 10, &off );	out->picture_type = get_field( d, 3, &off );	out->vectors = 0;	off += 16;	if( out->picture_type == 2 || out->picture_type == 3 )	{		out->vectors = get_field( d, 1, &off ) << 3; /* FFV */		out->vectors |= len > 8 ? get_field( d, 3, &off ) : /* FFC */					( get_field( d, 2, &off ) << 1 );		if( out->picture_type == 3 && len > 8 )		{			out->vectors |= get_field( d, 1, &off ) << 7; /* FBV */			out->vectors |= get_field( d, 3, &off ) << 4; /* BFC */		}	}}static int mpv_process_frame( struct frame *f, void *d ){	struct rtp_mpv *out = (struct rtp_mpv *)d;	int flen, start, in_picture = 0, have_vsh = 0;	unsigned int start_code;	out->blk_count = 0;	for( start = 0; start < f->length; start += flen )	{		if( out->blk_count == 48 )		{			spook_log( SL_WARN,				"rtp-mpv: too many elements to send" );			break;		}		flen = find_next_code( f->d + start, f->length - start );		if( flen == 0 ) flen = f->length - start;		start_code = GET_32( f->d + start );		if( start_code <= 0x000001AF || start_code == 0x000001B8 )			in_picture = 1;		if( in_picture )		{			if( start_code == 0x00000100 )				parse_picture_header( out, f->d + start, flen );			out->blk[out->blk_count].d = f->d + start;			out->blk[out->blk_count].len = flen;			++out->blk_count;		} else /* we have not seen a GOP or picture start code yet */		{			if( start_code == 0x000001B3 )			{				out->vsh_len = 0;				have_vsh = 1;				parse_video_sequence_header( out,						f->d + start, flen );			}			if( have_vsh )			{				memcpy( out->vsh + out->vsh_len,						f->d + start, flen );				out->vsh_len += flen;			}		}	}	return out->init_done;}static int mpv_get_sdp( char *dest, int len, int payload, int port, void *d ){	return snprintf( dest, len, "m=video %d RTP/AVP 32\r\n", port );}static int mpv_get_payload( int payload, void *d ){	return 32;}static int mpv_send( struct rtp_endpoint *ep, void *d ){	struct rtp_mpv *out = (struct rtp_mpv *)d;	int i, j, space, off, min_space;	struct iovec v[48];	unsigned char vhdr[4];	/* MPEG video header always follows the RTP header */	PUT_16( vhdr, out->temporal_reference );	vhdr[2] = out->picture_type; /* will be different for each frame */	vhdr[3] = out->vectors;	v[1].iov_base = vhdr;	v[1].iov_len = 4;	i = 0;	j = 2;	space = ep->max_data_size - 4;	/* If this is an I frame, insert the saved Video Sequence Header */	if( out->picture_type == 1 )	{		vhdr[2] |= 0x20; /* Sequence-header-present bit */		/* add the block to the frame */		v[j].iov_base = out->vsh;		v[j].iov_len = out->vsh_len;		space -= v[j].iov_len;		++j;	}	for( i = 0; i < out->blk_count; ++i )	{		/* We can fragment slices, but after the initial fragment,		 * all the remaining fragments must be put into their own		 * packets.  This means fragmentation only makes sense if		 * the slice is larger than our MTU. */		if( out->blk[i].len > ep->max_data_size - 4 ) min_space = 4;		else min_space = out->blk[i].len;		/* If we don't have enough space for this entire block, or if		 * we're fragmenting and we don't have enough space for the		 * start code, first send out the previous blocks. */		if( space < min_space )		{			if( send_rtp_packet( ep, v, j, out->timestamp, 0 ) < 0 )				return -1;			j = 2;			vhdr[2] = out->picture_type;			space = ep->max_data_size - 4;		}		/* add the block to the frame */		v[j].iov_base = out->blk[i].d;		v[j].iov_len = out->blk[i].len;		/* if this is a slice, set the Beginning-of-slice bit */		if( out->blk[i].d[3] > 0 && out->blk[i].d[3] <= 0xAF )			vhdr[2] |= 0x10;		/* if the entire block fit, go on to the next block */		if( v[j].iov_len <= space )		{			/* if this is a slice, set the End-of-slice bit */			if( out->blk[i].d[3] > 0 && out->blk[i].d[3] <= 0xAF )				vhdr[2] |= 0x08;			space -= v[j].iov_len;			++j;			continue;		}		/* block did not fit, so send the first fragment */		v[j].iov_len = space;		/* the last byte of the packet is not the end of a slice, so		 * clear the End-of-slice bit */		vhdr[2] &= ~0x08;		if( send_rtp_packet( ep, v, j + 1, out->timestamp, 0 ) < 0 )			return -1;		/* send all remaining fragments by themselves */		for( off = space; off < out->blk[i].len;				off += v[2].iov_len )		{			vhdr[2] = out->picture_type;			v[2].iov_base = out->blk[i].d + off;			v[2].iov_len = out->blk[i].len - off;			if( v[2].iov_len > ep->max_data_size - 4 )				v[2].iov_len = ep->max_data_size - 4;			else				vhdr[2] |= 0x08; /* End-of-slice bit */			if( send_rtp_packet( ep, v, 3, out->timestamp,					( vhdr[2] & 0x08 ) &&					( ( i + 1 ) == out->blk_count ) ) < 0 )				return -1;		}		j = 2;		vhdr[2] = out->picture_type;		space = ep->max_data_size - 4;	}	/* send any unsent blocks */	if( j > 2 && send_rtp_packet( ep, v, j, out->timestamp, 1 ) < 0 )		return -1;	return 0;}struct rtp_media *new_rtp_media_mpv(void){	struct rtp_mpv *out;	out = (struct rtp_mpv *)malloc( sizeof( struct rtp_mpv ) );	out->init_done = 0;	out->timestamp = 0;	return new_rtp_media( mpv_get_sdp, mpv_get_payload, mpv_process_frame,			mpv_send, out );}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -