RTP RTCP 客户端接收RTSP中媒体数据

    在使用live555 作为RTSP服务端的的时候,它建立的RTSP服务器其中包括了RTSP协议用来接收和处理客户端的会话请求命令,RTP用来发送媒体数据,RTCP用来发送一些控制信息。在调试RTSP协议的时候知道,live555的RTSP服务器,再它没有建立RTSP会话的时候,它的RTP和RTCP协议它就已经建立起来了。并且它的RTCP和RTP协议都是以多播的形式发送,不管是否有客户端在接收数据,它都是按设置好的周期发送。为了更加方便的了解live555的通信数据,我做了一个小程序,用来单独接收RTP或是RTCP中的数据。

注明如下:

    服务端程序:live555 testProgs 中的 testH264VideoStreamer

    服务端地址:192.168.0.6

    多播地址    :232.165.193.149

    RTP端口    :18888

    RTCP端口 :18889

    客户端地址:192.168.0.5

rtp_rtcp_client.c

/*=============================================================================  
 *     FileName: rtp_rtcp_client.c
 *         Desc: a client to recevie the rtp or rtcp data from live555 RTSP  
 *       Author: licaibiao  
 *   LastChange: 2017-04-20   
 *=============================================================================*/ 
#include<stdio.h>
#include<stdlib.h>  
#include<string.h>     
#include<unistd.h>      
#include<sys/types.h>    
#include<sys/socket.h>     
#include<arpa/inet.h>    
#include<netinet/in.h>    

#define MAXBUF 1024

int init_udpsocket(int port, struct sockaddr_in *servaddr, char *mcast_addr);
void release_udpsocket(int socket_fd, char *mcast_addr);
void proc_udpsocket(int socket_fd, struct sockaddr_in servaddr);

void main(void)
{
	int socket_fd;
	int rtp_port = 18888;
	int rtcp_port = 18888 + 1;
	char mcast_addr[] = "232.165.193.149";   
	struct sockaddr_in servaddr;

	socket_fd = init_udpsocket(rtcp_port, &servaddr, mcast_addr);
	proc_udpsocket(socket_fd, servaddr);
	release_udpsocket(socket_fd, mcast_addr);	
}

int init_udpsocket(int port, struct sockaddr_in *servaddr, char *mcast_addr)
{
	
	int err = -1;
    int socket_fd;                                      
	
    socket_fd = socket(AF_INET, SOCK_DGRAM, 0);    
    if (socket_fd < 0 )
    {
        perror("socket()");
        return -1;
    }  
   
    memset(servaddr, 0, sizeof(struct sockaddr_in));
    servaddr->sin_family 	   = AF_INET;
    servaddr->sin_addr.s_addr  = htonl(INADDR_ANY);
    servaddr->sin_port 		   = htons(port);
   
    err = bind(socket_fd,(struct sockaddr*)servaddr, sizeof(struct sockaddr_in)) ;
    if(err < 0)
    {
        perror("bind()");
        return -2;
    }
     
	/*set enable MULTICAST LOOP */                                       
    int loop = 1;
    err = setsockopt(socket_fd,IPPROTO_IP, IP_MULTICAST_LOOP,&loop, sizeof(loop));
    if(err < 0)
    {
        perror("setsockopt():IP_MULTICAST_LOOP");
        return -3;
    }
   
	/* Join multicast group */ 
	struct ip_mreq mreq; 
    mreq.imr_multiaddr.s_addr = inet_addr(mcast_addr); 
    mreq.imr_interface.s_addr = htonl(INADDR_ANY);      
    err = setsockopt(socket_fd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &mreq, sizeof(struct ip_mreq));
    if (err < 0)
    {
        perror("setsockopt():IP_ADD_MEMBERSHIP");
        return -4;
    }
	return socket_fd;
}

void release_udpsocket(int socket_fd, char *mcast_addr)
{
	struct ip_mreq mreq; 
    mreq.imr_multiaddr.s_addr = inet_addr(mcast_addr); 
    mreq.imr_interface.s_addr = htonl(INADDR_ANY); 
	setsockopt(socket_fd, IPPROTO_IP, IP_DROP_MEMBERSHIP,&mreq, sizeof(struct ip_mreq));
    close(socket_fd);
}

void proc_udpsocket(int socket_fd, struct sockaddr_in servaddr)
{
	int n;
	int i;
	int addr_len;
	char buff[MAXBUF + 1] = {0};

	memset(buff, 0, MAXBUF);    
	addr_len = sizeof(struct sockaddr_in);                                                        
    n = recvfrom(socket_fd, buff, MAXBUF, 0,(struct sockaddr*)&servaddr, &addr_len);
    if(n == -1)
    {
        perror("recvfrom()");
    }        
    printf("Recv %d byte message from server:\n", n);	
	for(i=0; i<n; i++)
	{
		printf("%d ",buff[i]);
	}
    printf("\n\n");	
}

    经测试,接收数据与Wireshark 工具抓取数据相同。

    代码下载地址:RTP RTCP 客户端接收RTSP中媒体数据
 

​​​​​​​

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

li_wen01

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值