登录  
 加关注
   显示下一条  |  关闭
温馨提示!由于新浪微博认证机制调整,您的新浪微博帐号绑定已过期,请重新绑定!立即重新绑定新浪微博》  |  关闭

我的博客

linfengdu

 
 
 

日志

 
 
关于我

男人无所谓正派,正派是因为受到的引诱不够,女人无所谓忠诚,忠诚是因为背叛的筹码太低.(所有文章都来自网络,感谢原作者们!) ------各位有幸看我博客的朋友,帮忙置顶的淘宝链接!!

RTP - 视频流广播  

2010-10-27 12:57:42|  分类: linux |  标签: |举报 |字号 订阅

  下载LOFTER 我的照片书  |

这是用RTP(RFC3350)按RFC2550封装MPEG ES流数据的发送程序。学习RTP的路真的辛苦。在网上收集的有关RTP的程序都是那种只负责RTP数据包发送的库,如jrtplib等,他们的DEMO程序都只是用来发发字符串,编编聊天程序,无论是国内还是国外,都没有结合真正的应用的DEMO。其实我的目的很简单,就是写发个视频流服务器,不用复杂,只用把基本原理弄懂,因为这样你才能有的放矢。与网上和RTP相关的库没有应用不一让,当你尝试以流媒体服务器、linux来baidu或google时,你搜出来完非就那么几类:

1.FFSERVER
     FFMPEG2的DEMO,说它有名只是因为这类程序太少了。FFMPEG2是很好用,我现在还在用,但这个DEMO就有很多“炒作”的嫌疑了。好像在做着FFMPEG2库的演示而不是真的视频流服务器。后来想想,这不正是作者想要的吗,但这不是我想要的。编解码部分我会很偏向FFMPEG这个“大杂会”,其它部分我会选择其它的“强者”

2.Darwin、Helix
     两个都是非常有名软件,也只能称之为软件了,因为就算Darwin有源码,这种代码规模,也不适合用于嵌入式。说回软件本身,真的很有名。它们都是很真真拿来商业化运行的软件,但我是研发人员,不是视频流服务商,对不起,Apple,对不起,Microsoft。

3.LIVE555
     如果说上面两个和我都相关性为零(当然了,也是困扰了N周以后痛苦得出的结论),那LIVE555真的给了我一条出路,它是一个代码规模非常合适,又非常强大的媒体解决方案(称之为方案是因为它功能非常的丰富)。有长一段时间,我想去弄懂它的源码,不过和网上的很多人一样,最后软下来了,毕竟,去把这么多东西揉在一起,框架会弄得很复杂,因为我们要把这些完全不同的东西不断一层一层的抽像,最后抽像成一样(哲学呀)。它结构复杂是我中断分析它原来的其中一个原因,但不是主要原因。它结构的复杂程度也没胡像很多人网上说的那样严重,如果你是一个C++的热忱爱好者,你反而会迷上这段代码,当然了,对C的爱好都来说,当然是一种折磨了。暂时把我自己归类在C++爱好者范畴吧,呵呵,我很欣赏这段原码。主要原因是我不希望被某一个库绑死。LIVE555是有编解码能力,但我更希望它只做服务器的工作。

     因此,最终后回来的老路上来,没有帮助,就得自己帮自己,从最基础的RFC看起。经过了N天(周)的英文,终于领会了如果在RTP承载MPEG数据包。在这个过程中很得到了一些LIVE555的帮助(通过对Ethereal捕捉的LIVE555数据包进行分析)。先把程序弄上来,原理性的以后有空再写,程序只有一个.cpp文件,在vs.net 2003下编译通过,播放的视频文件在http://www.cnitblog.com/Files/tinnal/ES流解释程序.rar  内,播放的客户端采用VLC,其下载地址为http://www.videolan.org/。选择打开网络串流,然后选择“UDP/RTP”端口,输入程序的输出端口1000,然后才运行程序,你将在VLC内看到测试的广播视频,IP不一样的话自己改改就行。其它所谓的原理性的,也就是看RFC 3350、RFC2550以及iso13818-2的一些重点地方。

   // MPEG2RTP.hRTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客#include
<stdio.h>
RTP - 视频流广播 - linfengdu - 我的博客#include
<stdlib.h>
RTP - 视频流广播 - linfengdu - 我的博客#include
<conio.h>
RTP - 视频流广播 - linfengdu - 我的博客#include
<string.h>
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客#include
<winsock2.h>
RTP - 视频流广播 - linfengdu - 我的博客#include
<winsock2.h>
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客
//#include "mem.h"
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客
//< Start code or other signal>
RTP - 视频流广播 - linfengdu - 我的博客
#define PACK_STARTCODE                     (unsigned int)0x000001ba
RTP - 视频流广播 - linfengdu - 我的博客
#define SYSTEM_HEADER_STARTCODE     (unsigned int)0x000001bb
RTP - 视频流广播 - linfengdu - 我的博客
#define PICTURE_START_CODE               (unsigned int)0x00000100
RTP - 视频流广播 - linfengdu - 我的博客
#define GROUP_START_CODE            (unsigned int)0x000000B8
RTP - 视频流广播 - linfengdu - 我的博客
#define ISO_11172_ENDCODE             (unsigned int)0x000001b9
RTP - 视频流广播 - linfengdu - 我的博客
#define SEQUENCE_HEADER_CODE        (unsigned int)0x000001b3
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客
#define PACKET_BUFFER_END            (unsigned int)0x00000000
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客
#define MAX_RTP_PKT_LENGTH     1440
RTP - 视频流广播 - linfengdu - 我的博客
#define HEADER_LENGTH        16
RTP - 视频流广播 - linfengdu - 我的博客
#define DEST_IP                "192.168.0.98"
RTP - 视频流广播 - linfengdu - 我的博客
#define DEST_PORT            1000
RTP - 视频流广播 - linfengdu - 我的博客
#define MPA                    14 /*MPEG PAYLOAD TYPE */
RTP - 视频流广播 - linfengdu - 我的博客
#define MPV                    32
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客typedef
struct 
RTP - 视频流广播 - linfengdu - 我的博客
{
RTP - 视频流广播 - linfengdu - 我的博客   
/* byte 0 */
RTP - 视频流广播 - linfengdu - 我的博客    unsigned
char csrc_len:4;        /* expect 0 */
RTP - 视频流广播 - linfengdu - 我的博客    unsigned
char extension:1;        /* expect 1, see RTP_OP below */
RTP - 视频流广播 - linfengdu - 我的博客    unsigned
char padding:1;        /* expect 0 */
RTP - 视频流广播 - linfengdu - 我的博客    unsigned
char version:2;        /* expect 2 */
RTP - 视频流广播 - linfengdu - 我的博客   
/* byte 1 */
RTP - 视频流广播 - linfengdu - 我的博客    unsigned
char payload:7;        /* RTP_PAYLOAD_RTSP */
RTP - 视频流广播 - linfengdu - 我的博客    unsigned
char marker:1;        /* expect 1 */
RTP - 视频流广播 - linfengdu - 我的博客   
/* bytes 2, 3 */
RTP - 视频流广播 - linfengdu - 我的博客    unsigned
short seq_no;           
RTP - 视频流广播 - linfengdu - 我的博客   
/* bytes 4-7 */
RTP - 视频流广播 - linfengdu - 我的博客    unsigned 
long timestamp;       
RTP - 视频流广播 - linfengdu - 我的博客   
/* bytes 8-11 */
RTP - 视频流广播 - linfengdu - 我的博客    unsigned
long ssrc;            /* stream number is used here. */
RTP - 视频流广播 - linfengdu - 我的博客}
RTP_FIXED_HEADER;
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客typedef
struct {
RTP - 视频流广播 - linfengdu - 我的博客   
//byte 0
RTP - 视频流广播 - linfengdu - 我的博客
    unsigned char TR_high2:2;    /* Temporal Reference high 2 bits*/
RTP - 视频流广播 - linfengdu - 我的博客    unsigned
char T:1;            /* video specific head extension flag */
RTP - 视频流广播 - linfengdu - 我的博客    unsigned
char MBZ:5;        /* unused */
RTP - 视频流广播 - linfengdu - 我的博客   
//byte1
RTP - 视频流广播 - linfengdu - 我的博客
    unsigned char TR_low8:8;    /* Temporal Reference low 8 bits*/
RTP - 视频流广播 - linfengdu - 我的博客   
//byte3
RTP - 视频流广播 - linfengdu - 我的博客
    unsigned char P:3;            /* picture type; 1=I,2=P,3=B,4=D */
RTP - 视频流广播 - linfengdu - 我的博客    unsigned
char E:1;         /* set if last byte of payload is slice end code */
RTP - 视频流广播 - linfengdu - 我的博客    unsigned
char B:1;            /* set if start of payload is slice start code */   
RTP - 视频流广播 - linfengdu - 我的博客    unsigned
char S:1;            /* sequence header present flag */
RTP - 视频流广播 - linfengdu - 我的博客    unsigned
char N:1;            /* N bit; used in MPEG 2 */
RTP - 视频流广播 - linfengdu - 我的博客    unsigned
char AN:1;        /* Active N bit */
RTP - 视频流广播 - linfengdu - 我的博客   
//byte4
RTP - 视频流广播 - linfengdu - 我的博客
    unsigned char FFC:3;        /* forward_f_code */
RTP - 视频流广播 - linfengdu - 我的博客    unsigned
char FFV:1;        /* full_pel_forward_vector */
RTP - 视频流广播 - linfengdu - 我的博客    unsigned
char BFC:3;        /* backward_f_code */
RTP - 视频流广播 - linfengdu - 我的博客    unsigned
char FBV:1;        /* full_pel_backward_vector */
RTP - 视频流广播 - linfengdu - 我的博客}
MPEG_VID_SPECIFIC_HDR; /* 4 BYTES */
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客
enum reading_status {
RTP - 视频流广播 - linfengdu - 我的博客    SLICE_AGAIN,
RTP - 视频流广播 - linfengdu - 我的博客    SLICE_BREAK,
RTP - 视频流广播 - linfengdu - 我的博客    UNKNOWN,
RTP - 视频流广播 - linfengdu - 我的博客    SLICE,
RTP - 视频流广播 - linfengdu - 我的博客    SEQUENCE_HEADER,
RTP - 视频流广播 - linfengdu - 我的博客    GROUP_START,
RTP - 视频流广播 - linfengdu - 我的博客    PICTURE
RTP - 视频流广播 - linfengdu - 我的博客    }
;
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客
void validate_file();
RTP - 视频流广播 - linfengdu - 我的博客
float frame_rate(int buffer_index);
RTP - 视频流广播 - linfengdu - 我的博客unsigned
int read_picture_type(int buffer_index);
RTP - 视频流广播 - linfengdu - 我的博客unsigned
int read_FBV(int buffer_index);
RTP - 视频流广播 - linfengdu - 我的博客unsigned
int read_BFC(int buffer_index);
RTP - 视频流广播 - linfengdu - 我的博客unsigned
int read_FFV(int buffer_index);
RTP - 视频流广播 - linfengdu - 我的博客unsigned
int read_FFC(int buffer_index);
RTP - 视频流广播 - linfengdu - 我的博客unsigned
int extract_temporal_reference(int buffer_index);
RTP - 视频流广播 - linfengdu - 我的博客unsigned
int find_next_start_code(unsigned int *buffer_index);
RTP - 视频流广播 - linfengdu - 我的博客
void reset_buffer_index(void);
RTP - 视频流广播 - linfengdu - 我的博客BOOL InitWinsock();
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客


RTP - 视频流广播 - linfengdu - 我的博客//MPEG2RTP.cpp
    //这个程序主要用于RTP封装MPEG2数据的学习和测试,不作任何其它用途
RTP - 视频流广播 - linfengdu - 我的博客
//软件在VS.net 2003中编译通过,但在linux下作小量修改也应编译通过。
RTP - 视频流广播 - linfengdu - 我的博客
//通过VLC测试,VLC能正确接收和解码由本程序发送的TEST.MPV编码流。
RTP - 视频流广播 - linfengdu - 我的博客
//
RTP - 视频流广播 - linfengdu - 我的博客
//作者:冯富秋 Tinnal
RTP - 视频流广播 - linfengdu - 我的博客
//邮箱:tinnal@163.com
RTP - 视频流广播 - linfengdu - 我的博客

RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客#include
"MPEG2RTP.h"
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客#pragma   comment(lib,
"Ws2_32")
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客unsigned
char            buf[MAX_RTP_PKT_LENGTH + 4]; //input buffer
RTP - 视频流广播 - linfengdu - 我的博客
enum reading_status        state = SEQUENCE_HEADER;
RTP - 视频流广播 - linfengdu - 我的博客unsigned
int            g_index_in_packet_buffer = HEADER_LENGTH;
RTP - 视频流广播 - linfengdu - 我的博客
static unsigned long    g_time_stamp = 0;
RTP - 视频流广播 - linfengdu - 我的博客
static unsigned long    g_time_stamp_current =0;
RTP - 视频流广播 - linfengdu - 我的博客
static float            g_frame_rate = 0;
RTP - 视频流广播 - linfengdu - 我的博客
static unsigned int        g_delay_time = 0;
RTP - 视频流广播 - linfengdu - 我的博客
static unsigned int        g_timetramp_increment = 0;
RTP - 视频流广播 - linfengdu - 我的博客FILE   
*mpfd;
RTP - 视频流广播 - linfengdu - 我的博客SOCKET    socket1;
RTP - 视频流广播 - linfengdu - 我的博客RTP_FIXED_HEADER       
*rtp_hdr;
RTP - 视频流广播 - linfengdu - 我的博客MPEG_VID_SPECIFIC_HDR   
*mpeg_hdr;
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客
#if 0
RTP - 视频流广播 - linfengdu - 我的博客
void Send_RTP_Packet(unsigned char *buf,int bytes)
RTP - 视频流广播 - linfengdu - 我的博客
{
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客   
int i = 0;
RTP - 视频流广播 - linfengdu - 我的博客   
int count = 0;
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客    printf(
"\nPacket length %d\n",bytes);
RTP - 视频流广播 - linfengdu - 我的博客    printf(
"RTP Header: [M]:%s [sequence number]:0x%lx [timestamp]:0x%lx\n",
RTP - 视频流广播 - linfengdu - 我的博客        rtp_hdr
->marker == 1?"TRUE":"FALSE",
RTP - 视频流广播 - linfengdu - 我的博客        rtp_hdr
->seq_no,
RTP - 视频流广播 - linfengdu - 我的博客        rtp_hdr
->timestamp);
RTP - 视频流广播 - linfengdu - 我的博客    printf(
" [TR]:%d [AN]:%d [N]:%d [Sequence Header]:%s \
RTP - 视频流广播 - linfengdu - 我的博客
        \n [Begin Slice]:%s [End Slice]:%s \
RTP - 视频流广播 - linfengdu - 我的博客        \n [Pictute Type]:
%d \
RTP - 视频流广播 - linfengdu - 我的博客        \n [FBV]:
%d [BFC]:%d [FFV]:%d [FFC]:%d\n",
RTP - 视频流广播 - linfengdu - 我的博客
        (mpeg_hdr->TR_high2 << 8 | mpeg_hdr->TR_low8),
RTP - 视频流广播 - linfengdu - 我的博客        mpeg_hdr
->AN, mpeg_hdr->N, mpeg_hdr->S == 1?"TRUE":"FALSE",
RTP - 视频流广播 - linfengdu - 我的博客        mpeg_hdr
->B ==1?"TRUE":"FALSE", mpeg_hdr->E ==1?"TRUE":"FALSE",
RTP - 视频流广播 - linfengdu - 我的博客        mpeg_hdr
->P,
RTP - 视频流广播 - linfengdu - 我的博客        mpeg_hdr
->FBV, mpeg_hdr->BFC, mpeg_hdr->FFV, mpeg_hdr->FFC);
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客   
while(bytes --)
RTP - 视频流广播 - linfengdu - 我的博客   
{
RTP - 视频流广播 - linfengdu - 我的博客        printf(
"%02x ",buf[count++]);
RTP - 视频流广播 - linfengdu - 我的博客       
if(++i == 16)
RTP - 视频流广播 - linfengdu - 我的博客       
{
RTP - 视频流广播 - linfengdu - 我的博客            i
=0;
RTP - 视频流广播 - linfengdu - 我的博客            printf(
"\n");
RTP - 视频流广播 - linfengdu - 我的博客        }

RTP - 视频流广播 - linfengdu - 我的博客    }

RTP - 视频流广播 - linfengdu - 我的博客    printf(
"\n");
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客}

RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客
#else
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客Send_RTP_Packet(unsigned
char *buf,int bytes)
RTP - 视频流广播 - linfengdu - 我的博客
{
RTP - 视频流广播 - linfengdu - 我的博客   
return send( socket1,  (char*) buf, bytes, 0 );
RTP - 视频流广播 - linfengdu - 我的博客}

RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客
#endif
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客
void main(int argc, char *argv[])
RTP - 视频流广播 - linfengdu - 我的博客
{
RTP - 视频流广播 - linfengdu - 我的博客    unsigned
int next_start_code;
RTP - 视频流广播 - linfengdu - 我的博客    unsigned
int next_start_code_index;
RTP - 视频流广播 - linfengdu - 我的博客    unsigned
int sent_bytes;
RTP - 视频流广播 - linfengdu - 我的博客    unsigned
short seq_num =0;
RTP - 视频流广播 - linfengdu - 我的博客    unsigned
short stream_num = 10;
RTP - 视频流广播 - linfengdu - 我的博客   
struct sockaddr_in server;
RTP - 视频流广播 - linfengdu - 我的博客   
int len =sizeof(server);
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客
#if 0
RTP - 视频流广播 - linfengdu - 我的博客    mpfd
= fopen("E:\\tinnal\\live555\\vc_proj\\es\\Debug\\test.mpv", "rb"); 
RTP - 视频流广播 - linfengdu - 我的博客
#else
RTP - 视频流广播 - linfengdu - 我的博客   
if (argc < 2)
RTP - 视频流广播 - linfengdu - 我的博客   
{
RTP - 视频流广播 - linfengdu - 我的博客        printf(
"\nUSAGE: %s mpegfile\nExiting..\n\n",argv[0]);
RTP - 视频流广播 - linfengdu - 我的博客                exit(
0);
RTP - 视频流广播 - linfengdu - 我的博客    }

RTP - 视频流广播 - linfengdu - 我的博客    mpfd
= fopen(argv[1], "rb");
RTP - 视频流广播 - linfengdu - 我的博客
#endif
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客   
if (mpfd == NULL )
RTP - 视频流广播 - linfengdu - 我的博客   
{
RTP - 视频流广播 - linfengdu - 我的博客        printf(
"\nERROR: could not open input file %s\n\n",argv[1]);
RTP - 视频流广播 - linfengdu - 我的博客        exit(
0);
RTP - 视频流广播 - linfengdu - 我的博客    }

RTP - 视频流广播 - linfengdu - 我的博客    rtp_hdr
= (RTP_FIXED_HEADER*)&buf[0];
RTP - 视频流广播 - linfengdu - 我的博客    mpeg_hdr
= (MPEG_VID_SPECIFIC_HDR*)&buf[12];
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客    memset((
void *)rtp_hdr,0,12); //zero-out the rtp fixed hdr
RTP - 视频流广播 - linfengdu - 我的博客
    memset((void *)mpeg_hdr,0,4); //zero-out the video specific hdr
RTP - 视频流广播 - linfengdu - 我的博客
    memset((void *)buf,0,MAX_RTP_PKT_LENGTH + 4);
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客    InitWinsock();
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客    server.sin_family
=AF_INET;
RTP - 视频流广播 - linfengdu - 我的博客    server.sin_port
=htons(DEST_PORT);          //server的监听端口
RTP - 视频流广播 - linfengdu - 我的博客
    server.sin_addr.s_addr=inet_addr(DEST_IP); //server的地址
RTP - 视频流广播 - linfengdu - 我的博客

RTP - 视频流广播 - linfengdu - 我的博客    socket1
=socket(AF_INET,SOCK_DGRAM,0);
RTP - 视频流广播 - linfengdu - 我的博客    connect(socket1, (
const sockaddr *)&server, len) ;
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客   
//read the first packet from the mpeg file
RTP - 视频流广播 - linfengdu - 我的博客   
//always read 4 extra bytes in (in case there's a startcode there)
RTP - 视频流广播 - linfengdu - 我的博客   
//but dont send  more than MAX_RTP_PKT_LENGTH in one packet
RTP - 视频流广播 - linfengdu - 我的博客
    fread(&(buf[HEADER_LENGTH]), MAX_RTP_PKT_LENGTH-HEADER_LENGTH+4, 1,mpfd);
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客    validate_file();
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客   
do
RTP - 视频流广播 - linfengdu - 我的博客   
{   
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客       
/* initialization of the two RTP headers */
RTP - 视频流广播 - linfengdu - 我的博客        rtp_hdr
->seq_no     = htons(seq_num ++);
RTP - 视频流广播 - linfengdu - 我的博客        rtp_hdr
->payload     = MPV;
RTP - 视频流广播 - linfengdu - 我的博客        rtp_hdr
->version     = 2;
RTP - 视频流广播 - linfengdu - 我的博客        rtp_hdr
->marker    = 0;
RTP - 视频流广播 - linfengdu - 我的博客        rtp_hdr
->ssrc        = htonl(stream_num);   
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客        mpeg_hdr
->S = mpeg_hdr->E = mpeg_hdr->B= 0;
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客       
do{
RTP - 视频流广播 - linfengdu - 我的博客            next_start_code
= find_next_start_code(&next_start_code_index);
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客           
if ((next_start_code >0x100) && (next_start_code<0x1b0) )
RTP - 视频流广播 - linfengdu - 我的博客           
{
RTP - 视频流广播 - linfengdu - 我的博客               
//< We reach the first slice start code in current packet buffer>
RTP - 视频流广播 - linfengdu - 我的博客               
//< Set the B flag of the mpeg special header>
RTP - 视频流广播 - linfengdu - 我的博客
                if(state == SEQUENCE_HEADER
RTP - 视频流广播 - linfengdu - 我的博客                   
|| state ==GROUP_START
RTP - 视频流广播 - linfengdu - 我的博客                   
|| state ==PICTURE
RTP - 视频流广播 - linfengdu - 我的博客                   
|| state == UNKNOWN)
RTP - 视频流广播 - linfengdu - 我的博客               
{
RTP - 视频流广播 - linfengdu - 我的博客                    state
= SLICE;
RTP - 视频流广播 - linfengdu - 我的博客                    mpeg_hdr
->B = 1;
RTP - 视频流广播 - linfengdu - 我的博客                }

RTP - 视频流广播 - linfengdu - 我的博客               
//< We reach slice start code in current packet again>
RTP - 视频流广播 - linfengdu - 我的博客               
//< Set the E flag of the mpeg special header>
RTP - 视频流广播 - linfengdu - 我的博客               
//< and update the sent_bytes to the last slice data end>
RTP - 视频流广播 - linfengdu - 我的博客
                else if (state == SLICE ||state == SLICE_AGAIN)
RTP - 视频流广播 - linfengdu - 我的博客               
{
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客                    state
= SLICE_AGAIN;
RTP - 视频流广播 - linfengdu - 我的博客                    sent_bytes
= next_start_code_index;
RTP - 视频流广播 - linfengdu - 我的博客                    mpeg_hdr
->E     = 1;
RTP - 视频流广播 - linfengdu - 我的博客                }

RTP - 视频流广播 - linfengdu - 我的博客               
//< We reach slice start codethe previous slice end>
RTP - 视频流广播 - linfengdu - 我的博客               
//< for a broken slice set the E flag>
RTP - 视频流广播 - linfengdu - 我的博客               
//< According to RFC we shouldnt put another slice data to this packet>
RTP - 视频流广播 - linfengdu - 我的博客               
//< instead of sent it out>
RTP - 视频流广播 - linfengdu - 我的博客
                else if (state == SLICE_BREAK)
RTP - 视频流广播 - linfengdu - 我的博客               
{
RTP - 视频流广播 - linfengdu - 我的博客                    state
= UNKNOWN;
RTP - 视频流广播 - linfengdu - 我的博客                    sent_bytes
= next_start_code_index;
RTP - 视频流广播 - linfengdu - 我的博客                    mpeg_hdr
->E     = 1;
RTP - 视频流广播 - linfengdu - 我的博客                   
goto Sent_Packet;
RTP - 视频流广播 - linfengdu - 我的博客                }

RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客            }
 
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客           
switch(next_start_code)
RTP - 视频流广播 - linfengdu - 我的博客           
{
RTP - 视频流广播 - linfengdu - 我的博客           
case SEQUENCE_HEADER_CODE:
RTP - 视频流广播 - linfengdu - 我的博客               
//< SEQUENCE_HEADER_CODE after SLICE_START_CODE>
RTP - 视频流广播 - linfengdu - 我的博客               
//< we must sent the packet now so that the SEQUENCE_HEADER_CODE>
RTP - 视频流广播 - linfengdu - 我的博客               
//< will appear at the start of the next packet>
RTP - 视频流广播 - linfengdu - 我的博客
                if(state == SLICE || state == SLICE_AGAIN)
RTP - 视频流广播 - linfengdu - 我的博客               
{               
RTP - 视频流广播 - linfengdu - 我的博客                    state
= SEQUENCE_HEADER;
RTP - 视频流广播 - linfengdu - 我的博客                    sent_bytes
= next_start_code_index;
RTP - 视频流广播 - linfengdu - 我的博客                   
//< Accord to RFC >
RTP - 视频流广播 - linfengdu - 我的博客                   
//< at the end of a frame we should set RTP marker bit to >
RTP - 视频流广播 - linfengdu - 我的博客
                    rtp_hdr->marker = 1;
RTP - 视频流广播 - linfengdu - 我的博客                   
goto Sent_Packet;
RTP - 视频流广播 - linfengdu - 我的博客                }

RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客                state
= SEQUENCE_HEADER;
RTP - 视频流广播 - linfengdu - 我的博客                g_frame_rate
= frame_rate(next_start_code_index);
RTP - 视频流广播 - linfengdu - 我的博客                g_delay_time
= (unsigned int)(1000.0 / g_frame_rate +0.5); //ms
RTP - 视频流广播 - linfengdu - 我的博客
                g_timetramp_increment = (unsigned int)(90000.0 / g_frame_rate +0.5); //90K Hz
RTP - 视频流广播 - linfengdu - 我的博客
                mpeg_hdr->S=1;
RTP - 视频流广播 - linfengdu - 我的博客               
break;
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客           
case GROUP_START_CODE:
RTP - 视频流广播 - linfengdu - 我的博客               
//< GROUP_START_CODE after SLICE_START_CODE>
RTP - 视频流广播 - linfengdu - 我的博客               
//< we must sent the packet now so that the GROUP_START_CODE>
RTP - 视频流广播 - linfengdu - 我的博客               
//< will appear at the start of the next packet>
RTP - 视频流广播 - linfengdu - 我的博客
                if(state == SLICE || state == SLICE_AGAIN)
RTP - 视频流广播 - linfengdu - 我的博客               
{
RTP - 视频流广播 - linfengdu - 我的博客                    state
= GROUP_START;
RTP - 视频流广播 - linfengdu - 我的博客                    sent_bytes
= next_start_code_index;
RTP - 视频流广播 - linfengdu - 我的博客                   
//< Accord to RFC >
RTP - 视频流广播 - linfengdu - 我的博客                   
//< at the end of a frame we should set RTP marker bit to >
RTP - 视频流广播 - linfengdu - 我的博客
                    rtp_hdr->marker = 1;
RTP - 视频流广播 - linfengdu - 我的博客                   
goto Sent_Packet;
RTP - 视频流广播 - linfengdu - 我的博客                }
   
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客                state
= GROUP_START;
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客           
case PICTURE_START_CODE:
RTP - 视频流广播 - linfengdu - 我的博客               
//< PICTURE_START_CODE after PICTURE_START_CODE>
RTP - 视频流广播 - linfengdu - 我的博客               
//< we must sent the packet now so that the PICTURE_START_CODE>
RTP - 视频流广播 - linfengdu - 我的博客               
//< will appear at the start of the next packet>
RTP - 视频流广播 - linfengdu - 我的博客
                if(state == SLICE || state == SLICE_AGAIN)
RTP - 视频流广播 - linfengdu - 我的博客               
{
RTP - 视频流广播 - linfengdu - 我的博客                    state
= PICTURE;
RTP - 视频流广播 - linfengdu - 我的博客                    sent_bytes
= next_start_code_index;
RTP - 视频流广播 - linfengdu - 我的博客                   
//< Accord to RFC >
RTP - 视频流广播 - linfengdu - 我的博客                   
//< at the end of a frame we should set RTP marker bit to >
RTP - 视频流广播 - linfengdu - 我的博客
                    rtp_hdr->marker = 1;
RTP - 视频流广播 - linfengdu - 我的博客                   
goto Sent_Packet;
RTP - 视频流广播 - linfengdu - 我的博客                }

RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客                state
= PICTURE;
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客                mpeg_hdr
->TR_high2    = (extract_temporal_reference(next_start_code_index) & 0x300 )>> 8;
RTP - 视频流广播 - linfengdu - 我的博客                mpeg_hdr
->TR_low8     =  extract_temporal_reference(next_start_code_index) & 0xff;
RTP - 视频流广播 - linfengdu - 我的博客                mpeg_hdr
->P             = read_picture_type(next_start_code_index);
RTP - 视频流广播 - linfengdu - 我的博客               
//now read the motion vectors information
RTP - 视频流广播 - linfengdu - 我的博客
                if( (mpeg_hdr->P==2) || (mpeg_hdr->P==3))
RTP - 视频流广播 - linfengdu - 我的博客               
{ //if B- or P-type picture, need forward mv
RTP - 视频流广播 - linfengdu - 我的博客
                    mpeg_hdr->FFV = read_FFV(next_start_code_index);
RTP - 视频流广播 - linfengdu - 我的博客                    mpeg_hdr
->FFC = read_FFC(next_start_code_index);
RTP - 视频流广播 - linfengdu - 我的博客                }

RTP - 视频流广播 - linfengdu - 我的博客               
if( mpeg_hdr->P==3)
RTP - 视频流广播 - linfengdu - 我的博客               
{ // if B-type pictue, need backward mv
RTP - 视频流广播 - linfengdu - 我的博客
                    mpeg_hdr->FBV = read_FBV(next_start_code_index);
RTP - 视频流广播 - linfengdu - 我的博客                    mpeg_hdr
->BFC = read_BFC(next_start_code_index);
RTP - 视频流广播 - linfengdu - 我的博客                }

RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客               
//< Time stamp only increate per frame>
RTP - 视频流广播 - linfengdu - 我的博客               
//< But I or P frame>
RTP - 视频流广播 - linfengdu - 我的博客
                if( mpeg_hdr->P== 1 || mpeg_hdr->P == 2 ){
RTP - 视频流广播 - linfengdu - 我的博客                    g_time_stamp
+= g_timetramp_increment;
RTP - 视频流广播 - linfengdu - 我的博客                    g_time_stamp_current
= g_time_stamp;
RTP - 视频流广播 - linfengdu - 我的博客                }
else{
RTP - 视频流广播 - linfengdu - 我的博客                    g_time_stamp
+= g_timetramp_increment;
RTP - 视频流广播 - linfengdu - 我的博客                }

RTP - 视频流广播 - linfengdu - 我的博客               
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客               
break;
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客           
case PACKET_BUFFER_END:
RTP - 视频流广播 - linfengdu - 我的博客               
//< There is one more slice in the packet buffer>
RTP - 视频流广播 - linfengdu - 我的博客               
//< Anyway we only sent the integrated slice>
RTP - 视频流广播 - linfengdu - 我的博客
                if(state == SLICE_AGAIN) {
RTP - 视频流广播 - linfengdu - 我的博客                    state
= UNKNOWN;
RTP - 视频流广播 - linfengdu - 我的博客                   
goto Sent_Packet;
RTP - 视频流广播 - linfengdu - 我的博客                }

RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客               
//< There is one Slice in the packet buffer>
RTP - 视频流广播 - linfengdu - 我的博客               
//< But the Slice is to big so we break the slice>
RTP - 视频流广播 - linfengdu - 我的博客
                if(state == SLICE)
RTP - 视频流广播 - linfengdu - 我的博客               
{
RTP - 视频流广播 - linfengdu - 我的博客                    state
= SLICE_BREAK;
RTP - 视频流广播 - linfengdu - 我的博客                    sent_bytes
= next_start_code_index;
RTP - 视频流广播 - linfengdu - 我的博客                   
goto Sent_Packet;
RTP - 视频流广播 - linfengdu - 我的博客                }

RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客               
//< There if a broke slice but in current packet buffer>
RTP - 视频流广播 - linfengdu - 我的博客               
//< we could not find the end of the slice>
RTP - 视频流广播 - linfengdu - 我的博客               
//< Let it in the broke state>
RTP - 视频流广播 - linfengdu - 我的博客
                if(state == SLICE_BREAK )
RTP - 视频流广播 - linfengdu - 我的博客               
{
RTP - 视频流广播 - linfengdu - 我的博客                    state
= SLICE_BREAK;
RTP - 视频流广播 - linfengdu - 我的博客                    sent_bytes
= next_start_code_index;
RTP - 视频流广播 - linfengdu - 我的博客                   
goto Sent_Packet;
RTP - 视频流广播 - linfengdu - 我的博客                }

RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客               
break;
RTP - 视频流广播 - linfengdu - 我的博客            }

RTP - 视频流广播 - linfengdu - 我的博客        }
while(next_start_code != PACKET_BUFFER_END);
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客Sent_Packet:
RTP - 视频流广播 - linfengdu - 我的博客        rtp_hdr
->timestamp = htonl(g_time_stamp_current);
RTP - 视频流广播 - linfengdu - 我的博客        Send_RTP_Packet(buf, sent_bytes);
RTP - 视频流广播 - linfengdu - 我的博客       
RTP - 视频流广播 - linfengdu - 我的博客       
//copy the tail data to the head of the packet buffer
RTP - 视频流广播 - linfengdu - 我的博客
        memmove(&buf[HEADER_LENGTH], &buf[sent_bytes], MAX_RTP_PKT_LENGTH-sent_bytes);
RTP - 视频流广播 - linfengdu - 我的博客       
//reset the buffer index to zero
RTP - 视频流广播 - linfengdu - 我的博客
        reset_buffer_index();
RTP - 视频流广播 - linfengdu - 我的博客       
//reading data into buffer again
RTP - 视频流广播 - linfengdu - 我的博客
        fread(&(buf[(MAX_RTP_PKT_LENGTH-sent_bytes)+HEADER_LENGTH]), sent_bytes -HEADER_LENGTH , 1,mpfd);
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客       
// sleep g_delay_time msec for sending next picture data
RTP - 视频流广播 - linfengdu - 我的博客
        if(rtp_hdr->marker ==1) Sleep( g_delay_time );
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客    }
while(!feof(mpfd));
RTP - 视频流广播 - linfengdu - 我的博客    closesocket(socket1);
RTP - 视频流广播 - linfengdu - 我的博客    fclose(mpfd);
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客    printf(
"stream end.\n");
RTP - 视频流广播 - linfengdu - 我的博客}

RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客
//==================================================================
RTP - 视频流广播 - linfengdu - 我的博客

RTP - 视频流广播 - linfengdu - 我的博客unsigned
int find_next_start_code(unsigned int *next_start_code_index) //NOTE: all start codes ARE byte-aligned
RTP - 视频流广播 - linfengdu - 我的博客
{
RTP - 视频流广播 - linfengdu - 我的博客    unsigned
int byte0=0,byte1=0,byte2=0,byte3=0,startcode=0;
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客   
//while not startcode and have not exceeded max packet length
RTP - 视频流广播 - linfengdu - 我的博客
    while (g_index_in_packet_buffer < MAX_RTP_PKT_LENGTH)
RTP - 视频流广播 - linfengdu - 我的博客   
{
RTP - 视频流广播 - linfengdu - 我的博客       
if (buf[g_index_in_packet_buffer+0] == 0
RTP - 视频流广播 - linfengdu - 我的博客           
&& buf[g_index_in_packet_buffer+1] == 0
RTP - 视频流广播 - linfengdu - 我的博客           
&& buf[g_index_in_packet_buffer+2] ==1)
RTP - 视频流广播 - linfengdu - 我的博客       
{
RTP - 视频流广播 - linfengdu - 我的博客           
//printf("FOUND startcode %d\n",indx);
RTP - 视频流广播 - linfengdu - 我的博客
            byte0=(int)buf[g_index_in_packet_buffer+0];
RTP - 视频流广播 - linfengdu - 我的博客            byte1
=(int)buf[g_index_in_packet_buffer+1];
RTP - 视频流广播 - linfengdu - 我的博客            byte2
=(int)buf[g_index_in_packet_buffer+2];
RTP - 视频流广播 - linfengdu - 我的博客            byte3
=(int)buf[g_index_in_packet_buffer+3];
RTP - 视频流广播 - linfengdu - 我的博客            startcode
=(byte0 << 24) + (byte1 << 16) + (byte2 << 8) + byte3;
RTP - 视频流广播 - linfengdu - 我的博客           
*next_start_code_index = g_index_in_packet_buffer;
RTP - 视频流广播 - linfengdu - 我的博客            g_index_in_packet_buffer
= g_index_in_packet_buffer+4;
RTP - 视频流广播 - linfengdu - 我的博客           
return(startcode);
RTP - 视频流广播 - linfengdu - 我的博客        }

RTP - 视频流广播 - linfengdu - 我的博客       
else
RTP - 视频流广播 - linfengdu - 我的博客            g_index_in_packet_buffer
++;
RTP - 视频流广播 - linfengdu - 我的博客    }

RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客   
//< reach the end of the packet buffer>
RTP - 视频流广播 - linfengdu - 我的博客
    if (g_index_in_packet_buffer >= (MAX_RTP_PKT_LENGTH))
RTP - 视频流广播 - linfengdu - 我的博客   
{
RTP - 视频流广播 - linfengdu - 我的博客       
*next_start_code_index = g_index_in_packet_buffer -1;
RTP - 视频流广播 - linfengdu - 我的博客        g_index_in_packet_buffer
= HEADER_LENGTH;
RTP - 视频流广播 - linfengdu - 我的博客       
return PACKET_BUFFER_END;
RTP - 视频流广播 - linfengdu - 我的博客    }

RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客    printf(
"Error reading buffer..\n");
RTP - 视频流广播 - linfengdu - 我的博客    exit(
-1);
RTP - 视频流广播 - linfengdu - 我的博客   
return -1;
RTP - 视频流广播 - linfengdu - 我的博客}

RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客
void reset_buffer_index(void)
RTP - 视频流广播 - linfengdu - 我的博客
{
RTP - 视频流广播 - linfengdu - 我的博客    g_index_in_packet_buffer
= HEADER_LENGTH;
RTP - 视频流广播 - linfengdu - 我的博客}

RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客
//========================================================
RTP - 视频流广播 - linfengdu - 我的博客
float frame_rate(int buffer_index)
RTP - 视频流广播 - linfengdu - 我的博客
{
RTP - 视频流广播 - linfengdu - 我的博客    unsigned
char frame_rate_code;
RTP - 视频流广播 - linfengdu - 我的博客    frame_rate_code
= (unsigned char)buf[buffer_index +7] & 0xf;
RTP - 视频流广播 - linfengdu - 我的博客   
switch(frame_rate_code)
RTP - 视频流广播 - linfengdu - 我的博客   
{
RTP - 视频流广播 - linfengdu - 我的博客   
case 0x1:
RTP - 视频流广播 - linfengdu - 我的博客       
return 23.976;
RTP - 视频流广播 - linfengdu - 我的博客   
case 0x2:
RTP - 视频流广播 - linfengdu - 我的博客       
return 24.0;
RTP - 视频流广播 - linfengdu - 我的博客   
case 0x3:
RTP - 视频流广播 - linfengdu - 我的博客       
return 25.0;
RTP - 视频流广播 - linfengdu - 我的博客   
case 0x4:
RTP - 视频流广播 - linfengdu - 我的博客       
return 29.97;
RTP - 视频流广播 - linfengdu - 我的博客   
case 0x5:
RTP - 视频流广播 - linfengdu - 我的博客       
return 30.0;
RTP - 视频流广播 - linfengdu - 我的博客   
case 0x6:
RTP - 视频流广播 - linfengdu - 我的博客       
return 50.0;
RTP - 视频流广播 - linfengdu - 我的博客   
case 0x7:
RTP - 视频流广播 - linfengdu - 我的博客       
return 59.94;
RTP - 视频流广播 - linfengdu - 我的博客   
case 0x8:
RTP - 视频流广播 - linfengdu - 我的博客       
return 60.0;
RTP - 视频流广播 - linfengdu - 我的博客   
default:
RTP - 视频流广播 - linfengdu - 我的博客       
return 0;
RTP - 视频流广播 - linfengdu - 我的博客    }

RTP - 视频流广播 - linfengdu - 我的博客}

RTP - 视频流广播 - linfengdu - 我的博客
//========================================================
RTP - 视频流广播 - linfengdu - 我的博客
unsigned int extract_temporal_reference(int buffer_index) // 10 bits
RTP - 视频流广播 - linfengdu - 我的博客
{
RTP - 视频流广播 - linfengdu - 我的博客    unsigned
int low2bits=0,TR=0; // TR = temporal reference;
RTP - 视频流广播 - linfengdu - 我的博客

RTP - 视频流广播 - linfengdu - 我的博客    TR
= (unsigned int) (buf[buffer_index+4]);
RTP - 视频流广播 - linfengdu - 我的博客    TR
<<= 2;
RTP - 视频流广播 - linfengdu - 我的博客    low2bits
= (unsigned int) (buf[buffer_index+5]);
RTP - 视频流广播 - linfengdu - 我的博客    TR
|= (low2bits >> 6);
RTP - 视频流广播 - linfengdu - 我的博客   
return(TR);
RTP - 视频流广播 - linfengdu - 我的博客}

RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客
//========================================================
RTP - 视频流广播 - linfengdu - 我的博客

RTP - 视频流广播 - linfengdu - 我的博客unsigned
int read_picture_type(int buffer_index)
RTP - 视频流广播 - linfengdu - 我的博客
{
RTP - 视频流广播 - linfengdu - 我的博客    unsigned
int pictype=0;
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客    pictype
= (unsigned int) buf[buffer_index+5];
RTP - 视频流广播 - linfengdu - 我的博客    pictype
= (pictype >> 3) & (0x7);
RTP - 视频流广播 - linfengdu - 我的博客   
return (pictype);
RTP - 视频流广播 - linfengdu - 我的博客}

RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客
//=======================================================
RTP - 视频流广播 - linfengdu - 我的博客
unsigned int read_FFV(int buffer_index) // 1 bit
RTP - 视频流广播 - linfengdu - 我的博客
{
RTP - 视频流广播 - linfengdu - 我的博客   
return( (int) ((buf[buffer_index+7] & (0x4)) >> 2));
RTP - 视频流广播 - linfengdu - 我的博客}

RTP - 视频流广播 - linfengdu - 我的博客
//=======================================================
RTP - 视频流广播 - linfengdu - 我的博客
unsigned int read_FFC(int buffer_index) // 3 bits
RTP - 视频流广播 - linfengdu - 我的博客
{
RTP - 视频流广播 - linfengdu - 我的博客    unsigned
int FFC=0,lowbit=0;
RTP - 视频流广播 - linfengdu - 我的博客    FFC
= (int) (buf[buffer_index+7] & (0x3));
RTP - 视频流广播 - linfengdu - 我的博客    FFC
<<= 1;
RTP - 视频流广播 - linfengdu - 我的博客    lowbit
= (int) ((buf[buffer_index+8]) & (0x80));
RTP - 视频流广播 - linfengdu - 我的博客    FFC
= FFC | (lowbit >> 7 );
RTP - 视频流广播 - linfengdu - 我的博客   
return(FFC);
RTP - 视频流广播 - linfengdu - 我的博客}

RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客
//=======================================================
RTP - 视频流广播 - linfengdu - 我的博客
unsigned int read_FBV(int buffer_index) // 1 bit
RTP - 视频流广播 - linfengdu - 我的博客
{      
RTP - 视频流广播 - linfengdu - 我的博客   
return( (int) ((buf[buffer_index+8] & (0x40))>>6) );
RTP - 视频流广播 - linfengdu - 我的博客}

RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客
//=======================================================
RTP - 视频流广播 - linfengdu - 我的博客
unsigned int read_BFC(int buffer_index) // 3 bits
RTP - 视频流广播 - linfengdu - 我的博客
{               
RTP - 视频流广播 - linfengdu - 我的博客   
return( (int) ( (buf[buffer_index+8] & (0x38) ) >> 3 ) );
RTP - 视频流广播 - linfengdu - 我的博客}

RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客
void validate_file()
RTP - 视频流广播 - linfengdu - 我的博客
{
RTP - 视频流广播 - linfengdu - 我的博客   
/* to validate the file, ensure the existance of a startcode */
RTP - 视频流广播 - linfengdu - 我的博客   
int j=0,valid=0;
RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客   
while ((j++<MAX_RTP_PKT_LENGTH) && (!valid))
RTP - 视频流广播 - linfengdu - 我的博客   
{
RTP - 视频流广播 - linfengdu - 我的博客       
if (!((int)buf[j+0] + (int)buf[j+1]) && (((int)buf[j+2])==1))
RTP - 视频流广播 - linfengdu - 我的博客            valid
=1;
RTP - 视频流广播 - linfengdu - 我的博客    }

RTP - 视频流广播 - linfengdu - 我的博客   
if (!valid)
RTP - 视频流广播 - linfengdu - 我的博客   
{
RTP - 视频流广播 - linfengdu - 我的博客        printf(
"\nERROR: start code not found. \
RTP - 视频流广播 - linfengdu - 我的博客
               \nInput file must be a valid MPEG I file.\n");
RTP - 视频流广播 - linfengdu - 我的博客
        exit(0);
RTP - 视频流广播 - linfengdu - 我的博客    }
          
RTP - 视频流广播 - linfengdu - 我的博客}

RTP - 视频流广播 - linfengdu - 我的博客
RTP - 视频流广播 - linfengdu - 我的博客BOOL InitWinsock()
RTP - 视频流广播 - linfengdu - 我的博客
{
RTP - 视频流广播 - linfengdu - 我的博客   
int Error;
RTP - 视频流广播 - linfengdu - 我的博客    WORD VersionRequested;
RTP - 视频流广播 - linfengdu - 我的博客    WSADATA WsaData;
RTP - 视频流广播 - linfengdu - 我的博客    VersionRequested
=MAKEWORD(2,2);
RTP - 视频流广播 - linfengdu - 我的博客    Error
=WSAStartup(VersionRequested,&WsaData); //启动WinSock2
RTP - 视频流广播 - linfengdu - 我的博客
    if(Error!=0)
RTP - 视频流广播 - linfengdu - 我的博客   
{
RTP - 视频流广播 - linfengdu - 我的博客       
return FALSE;
RTP - 视频流广播 - linfengdu - 我的博客    }

RTP - 视频流广播 - linfengdu - 我的博客   
else
RTP - 视频流广播 - linfengdu - 我的博客   
{
RTP - 视频流广播 - linfengdu - 我的博客       
if(LOBYTE(WsaData.wVersion)!=2||HIBYTE(WsaData.wHighVersion)!=2)
RTP - 视频流广播 - linfengdu - 我的博客       
{
RTP - 视频流广播 - linfengdu - 我的博客            WSACleanup();
RTP - 视频流广播 - linfengdu - 我的博客           
return FALSE;
RTP - 视频流广播 - linfengdu - 我的博客        }

RTP - 视频流广播 - linfengdu - 我的博客       
RTP - 视频流广播 - linfengdu - 我的博客    }

RTP - 视频流广播 - linfengdu - 我的博客   
return TRUE;
RTP - 视频流广播 - linfengdu - 我的博客}

RTP - 视频流广播 - linfengdu - 我的博客

 

      完成这个测试程序后,我有了很大的信心,又重复看了RFC3550几编,其实,如果你真看了程序,你发现我只发送了RTP,并没有发送RTCP数据包,因此,我们是不能同步多个RTP流的。我没去编码下去,因为我觉得已经够了。这里强调,没用说的RTP没有了RTCP就不行!接下来的工作,就是把这个程序的下层发包函数去掉,采用RTP库JRTPLIB,我觉得这才应该是JRTPLIB的DEMO!如果有人问,就这样的一个程序就能完成任务了,要JRTPLIB干嘛,其实,我不写RTCP相关代码的原因为多个:

    1.RTCP里头有很多关于RTCP发送简隔的时间计算,RTP信息的统计,这种操作不是难,而是烦,我不想去写
    2.RTCP和RTP一开始出来的时候并不是因为视频的点播等应用的,而是视频会议。RTCP有管理与会者的层面含义,这一功能在很多场合并不会用到。
    3.我想简单,没有写多个流间的同步,如一个影片的视频和音频流。这些其实是RTCP来完成的。

    我懒得去写,因为这些功作RTP的各个库类都做得很好。我觉得用库的最大优点就在这吧。

  评论这张
 
阅读(1822)| 评论(0)

历史上的今天

评论

<#--最新日志,群博日志--> <#--推荐日志--> <#--引用记录--> <#--博主推荐--> <#--随机阅读--> <#--首页推荐--> <#--历史上的今天--> <#--被推荐日志--> <#--上一篇,下一篇--> <#-- 热度 --> <#-- 网易新闻广告 --> <#--右边模块结构--> <#--评论模块结构--> <#--引用模块结构--> <#--博主发起的投票-->
 
 
 
 
 
 
 
 
 
 
 
 
 
 

页脚

网易公司版权所有 ©1997-2018