RTP的接收程序分析

//          程序名:rtprecv.c

//          类型:学习笔记

#include

#include
#include
#ifndef _WIN32
#include
#include
#include
#endif
int cond=1;    //变量cond控制着while循环。
void stop_handler(int signum)
{
cond=0;
}
void ssrc_cb(RtpSession *session)    //SSRC环境变量改变时调用此函数。
{
printf("hey, the ssrc has changed !\n");
}
  //输入9个参数
static char *help="usage: rtprecv  filename loc_port [--format format] [--soundcard] [--noadapt] [--with-jitter ]\n";
#define MULAW 0   
#define ALAW 1
#if defined(__hpux) && HAVE_SYS_AUDIO_H   //如果不适用音频设备。则if 与 else之间的函数可删!
#include
int sound_init(int format)
{
int fd;
fd=open("/dev/audio",O_WRONLY);
if (fd<0){
perror("Can't open /dev/audio");
return -1;
}
ioctl(fd,AUDIO_RESET,0);
ioctl(fd,AUDIO_SET_SAMPLE_RATE,8000);
ioctl(fd,AUDIO_SET_CHANNELS,1);
if (format==MULAW)
ioctl(fd,AUDIO_SET_DATA_FORMAT,AUDIO_FORMAT_ULAW);
else ioctl(fd,AUDIO_SET_DATA_FORMAT,AUDIO_FORMAT_ALAW);
return fd;
}
#else
int sound_init(int format)          //没有音频设备时,执行此函数。
{
return -1;
}
#endif
int main(int argc, char*argv[])
{
RtpSession *session;
unsigned char buffer[160];
int err;
uint32_t ts=0;
int stream_received=0;
FILE *outfile;
int local_port;
int have_more; 
int i;
int format=0;       //待删
int soundcard=0;    //待删
int sound_fd=0;     //待删
int jittcomp=40;
bool_t adapt=TRUE;

/* init the lib */
if (argc<3){
printf("%s",help);
return -1;
}
local_port=atoi(argv[2]);    //设置端口
if (local_port<=0) {
printf("%s",help);
return -1;
}
for (i=3;i {
if (strcmp(argv[i],"--noadapt")==0) adapt=FALSE;
if (strcmp(argv[i],"--format")==0){           //音频相关,待删
i++;
if (i if (strcmp(argv[i],"mulaw")==0){
format=MULAW;
}else
if (strcmp(argv[i],"alaw")==0){
format=ALAW;
}else{
printf("Unsupported format %s\n",argv[i]);
return -1;
}
}
}
else if (strcmp(argv[i],"--soundcard")==0){    //音频相关,待删
soundcard=1;
}
else if (strcmp(argv[i],"--with-jitter")==0){  
i++;
if (i jittcomp=atoi(argv[i]);
printf("Using a jitter buffer of %i milliseconds.\n",jittcomp);
}
}
}
outfile=fopen(argv[1],"wb");
if (outfile==NULL) {
perror("Cannot open file for writing");
return -1;
}
if (soundcard){         //音频相关,待删
sound_fd=sound_init(format);
}
ortp_init();   
ortp_scheduler_init(); 
ortp_set_log_level_mask(ORTP_DEBUG|ORTP_MESSAGE|ORTP_WARNING|ORTP_ERROR);
signal(SIGINT,stop_handler);   //设置信号的处理函数
session=rtp_session_new(RTP_SESSION_RECVONLY);
rtp_session_set_scheduling_mode(session,1);  
//设置会话的调度模式。当第二个参数为真时,标示会话可以使用调度模式,例如:阻塞模式。
rtp_session_set_blocking_mode(session,1);
//当第二个参数为真,则使能调度模式。直接影响函数rtp_session_recv_with_ts()与rtp_session_send_with_ts()的行为!
rtp_session_set_local_addr(session,"0.0.0.0",atoi(argv[2]),-1);
//设置本地网络地址,端口号。
rtp_session_set_connected_mode(session,TRUE);
////一个connect()在第一个数据包接收后将对源地址进行使用。
//连接一个socket会造成拒绝所有不是connect()里指定的地址发来的数据。
rtp_session_set_symmetric_rtp(session,TRUE);
//穿越防火墙
rtp_session_enable_adaptive_jitter_compensation(session,adapt);
//自适应补偿功能
rtp_session_set_jitter_compensation(session,jittcomp);
//设置补偿时间
rtp_session_set_payload_type(session,0);
//设置负载的类型。H.264的类型为:payload_type_h264
rtp_session_signal_connect(session,"ssrc_changed",(RtpCallback)ssrc_cb,0);
rtp_session_signal_connect(session,"ssrc_changed",(RtpCallback)rtp_session_reset,0);
//有指定的信号出现时,将调用执行函数。同一个信号可定义多个执行函数。
//第四个参数为执行函数的参数。
while(cond)
{
have_more=1;
while (have_more){
err=rtp_session_recv_with_ts(session,buffer,160,ts,&have_more); 
//have_more标识缓冲区是否还有数据没有接收。当用户缓冲区不够大,数据为读取完时,
//则标识have_more指向的数据为1,希望用户再次调用本函数。
if (err>0) stream_received=1;
/* this is to avoid to write to disk some silence before the first RTP packet is returned*/
if ((stream_received) && (err>0)) {
size_t ret = fwrite(buffer,1,err,outfile);   //从缓冲区写入到输出文件!
if (sound_fd>0){        //if函数为向声音设备传送文件,待删除!!!
ret = write(sound_fd,buffer,err);
if (ret==-1){
fprintf(stderr,"write to sound card failed (%s)",strerror(errno));
}
}
}
}
ts+=160;
//ortp_message("Receiving packet.");
}
rtp_session_destroy(session);
ortp_exit();
ortp_global_stats_display();
return 0;
}

你可能感兴趣的:(流媒体,学习方法)