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

转自http://blog.csdn.net/li_wen01/article/details/70258000

在使用live555 作为RTSP(Real Time Streaming Protocol视频控制)服务端的的时候,它建立的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

[objc]  view plain  copy
 print ?
  1. /*=============================================================================   
  2.  *     FileName: rtp_rtcp_client.c 
  3.  *         Desc: a client to recevie the rtp or rtcp data from live555 RTSP   
  4.  *       Author: licaibiao   
  5.  *   LastChange: 2017-04-20    
  6.  *=============================================================================*/   
  7. #include<stdio.h>  
  8. #include<stdlib.h>    
  9. #include<string.h>       
  10. #include<unistd.h>        
  11. #include<sys/types.h>      
  12. #include<sys/socket.h>       
  13. #include<arpa/inet.h>      
  14. #include<netinet/in.h>      
  15.   
  16. #define MAXBUF 1024  
  17.   
  18. int init_udpsocket(int port, struct sockaddr_in *servaddr, charchar *mcast_addr);  
  19. void release_udpsocket(int socket_fd, charchar *mcast_addr);  
  20. void proc_udpsocket(int socket_fd, struct sockaddr_in servaddr);  
  21.   
  22. void main(void)  
  23. {  
  24.     int socket_fd;  
  25.     int rtp_port = 18888;  
  26.     int rtcp_port = 18888 + 1;  
  27.     char mcast_addr[] = "232.165.193.149";     
  28.     struct sockaddr_in servaddr;  
  29.   
  30.     socket_fd = init_udpsocket(rtcp_port, &servaddr, mcast_addr);  
  31.     proc_udpsocket(socket_fd, servaddr);  
  32.     release_udpsocket(socket_fd, mcast_addr);     
  33. }  
  34.   
  35. int init_udpsocket(int port, struct sockaddr_in *servaddr, charchar *mcast_addr)  
  36. {  
  37.       
  38.     int err = -1;  
  39.     int socket_fd;                                        
  40.       
  41.     socket_fd = socket(AF_INET, SOCK_DGRAM, 0);      
  42.     if (socket_fd < 0 )  
  43.     {  
  44.         perror("socket()");  
  45.         return -1;  
  46.     }    
  47.      
  48.     memset(servaddr, 0sizeof(struct sockaddr_in));  
  49.     servaddr->sin_family        = AF_INET;  
  50.     servaddr->sin_addr.s_addr  = htonl(INADDR_ANY);  
  51.     servaddr->sin_port          = htons(port);  
  52.      
  53.     err = bind(socket_fd,(struct sockaddr*)servaddr, sizeof(struct sockaddr_in)) ;  
  54.     if(err < 0)  
  55.     {  
  56.         perror("bind()");  
  57.         return -2;  
  58.     }  
  59.        
  60.     /*set enable MULTICAST LOOP */                                         
  61.     int loop = 1;  
  62.     err = setsockopt(socket_fd,IPPROTO_IP, IP_MULTICAST_LOOP,&loop, sizeof(loop));  
  63.     if(err < 0)  
  64.     {  
  65.         perror("setsockopt():IP_MULTICAST_LOOP");  
  66.         return -3;  
  67.     }  
  68.      
  69.     /* Join multicast group */   
  70.     struct ip_mreq mreq;   
  71.     mreq.imr_multiaddr.s_addr = inet_addr(mcast_addr);   
  72.     mreq.imr_interface.s_addr = htonl(INADDR_ANY);        
  73.     err = setsockopt(socket_fd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &mreq, sizeof(struct ip_mreq));  
  74.     if (err < 0)  
  75.     {  
  76.         perror("setsockopt():IP_ADD_MEMBERSHIP");  
  77.         return -4;  
  78.     }  
  79.     return socket_fd;  
  80. }  
  81.   
  82. void release_udpsocket(int socket_fd, charchar *mcast_addr)  
  83. {  
  84.     struct ip_mreq mreq;   
  85.     mreq.imr_multiaddr.s_addr = inet_addr(mcast_addr);   
  86.     mreq.imr_interface.s_addr = htonl(INADDR_ANY);   
  87.     setsockopt(socket_fd, IPPROTO_IP, IP_DROP_MEMBERSHIP,&mreq, sizeof(struct ip_mreq));  
  88.     close(socket_fd);  
  89. }  
  90.   
  91. void proc_udpsocket(int socket_fd, struct sockaddr_in servaddr)  
  92. {  
  93.     int n;  
  94.     int i;  
  95.     int addr_len;  
  96.     char buff[MAXBUF + 1] = {0};  
  97.   
  98.     memset(buff, 0, MAXBUF);      
  99.     addr_len = sizeof(struct sockaddr_in);                                                          
  100.     n = recvfrom(socket_fd, buff, MAXBUF, 0,(struct sockaddr*)&servaddr, &addr_len);  
  101.     if(n == -1)  
  102.     {  
  103.         perror("recvfrom()");  
  104.     }          
  105.     printf("Recv %d byte message from server:\n", n);     
  106.     for(i=0; i<n; i++)  
  107.     {  
  108.         printf("%d ",buff[i]);  
  109.     }  
  110.     printf("\n\n");   
  111. }  
    经测试,接收数据与Wireshark 工具抓取数据相同。

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


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值