音视频数据处理入门:RTP协议解析

协议分析

image-20210501093710855

1、RTP(Real-time Transport Protocol)

RTP报文由两部分组成:报头和有效载荷。

RTP头部:
image-20210501094719200

这里只说明几个我们需要用到的字段:

V(2位):协议版本号

pt(7位):payload type载荷类型

sequence number(2字节):序列号,每个报文都有一个序列号;接收者据此来检测报文丢失情况,重组报文。

timestamp(4字节):时间戳,反映采样时间;接收者据此来计算延迟和延迟抖动,并进行同步控制。


2、MPEG-TS
image-20210501110807922 音频数据和视频数据经压缩编码后,形成音频基本流和视频基本流(ES:Elementary Stream);基本流经过打包、附加某些必要信息后,形成打包基本流(PES)。音频PES和视频PES可以经节目复合器形成节目流,也可以经传输流复合器形成传输流。

TS包结构:包头(4字节)、自适配域(可选,一般不选)、包数据(最大184字节)

image-20210501110215045

包头信息:

image-20210501102841377

包数据有两种:

1)视频、音频的PES包

2)PSI节目专用信息

PSI:节目特定信息(ProgramSpecificInformation),由节目关联表PAT,条件接收表CAT,节目映射表PMT和网络信息表NIT组成。


程序代码

①引用头文件与RTP头结构定义

#include <stdio.h>
#include <stdint.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>

/*
 *  RTP data header: 12Bytes
 */
typedef struct {
    unsigned char cc:4;        /* CSRC count */
    unsigned char x:1;         /* header extension flag */
    unsigned char p:1;         /* padding flag */
    unsigned char version:2;   /* protocol version */
    unsigned char pt:7;        /* payload type */
    unsigned char m:1;         /* marker bit */
    unsigned short seq;      /* sequence number */
    uint32_t ts;              /* timestamp */
    uint32_t ssrc;            /* synchronization source */
    uint32_t csrc[0];         /* optional CSRC list */
} rtp_header_t;
// 网络传输 -》大端模式, 高地址放低字节
// 注意非单字节数据要用ntoh转换

/*
 * MPEGTS header: 4Bytes
 */
typedef struct {
    unsigned int  sync_byte;                        // 同步字节,恒为0x47
    unsigned int  ts_error_indicator: 1;            // 传输误码指示符
    unsigned int  payload_unit_start_indicator: 1;  // 有效单元载荷起始指示
    unsigned int  ts_priority: 1;                   // 优先传输
    unsigned int  PID: 13;                          // 包标识符,流类型
    unsigned int scrambling_control: 2;             // 传输控制标识
    unsigned int adaptation_field_exist: 2;         // 自适应区标识,1时表示
有附加数据
    unsigned int continue_counter: 4;               // 连续计数器
} mpegts_header_t;

②创建socket并绑定端口进行监听

static struct sockaddr_in local_addr;
int udp_socket_init() {
    int sockfd = socket(AF_INET, SOCK_DGRAM, 0);
    if (sockfd < 0) {
        perror("socket()");
        return -1;
    }

    // sockfd绑定127.0.0.1:1998
    local_addr.sin_family = AF_INET;
    local_addr.sin_port = htons(1998);
    inet_pton(AF_INET, "0.0.0.0", &local_addr.sin_addr);
    if (bind(sockfd, (void *)&local_addr, sizeof(local_addr)) < 0) {
        perror("bind()");
        return -2;
    }
    return sockfd;
}

③接收RTP包并解析RTP头信息

void rtp_parser(void) {
    struct sockaddr_in remote_addr;
    socklen_t len = sizeof(remote_addr);

    int sockfd = udp_socket_init();
    if (sockfd < 0) return ;
    
    int cnt = 0;
    char recvbuf[10000];
    while (1) {
        int pktsize = recvfrom(sockfd, recvbuf, 10000, 0, (void *)&remote_ad
dr, &len);
        if (pktsize < 0) continue;

        // 解析RTP头信息
        rtp_header_t rtpHead;
        int rtphead_size = sizeof(rtpHead);
        memcpy((void *)&rtpHead, recvbuf, rtphead_size);
        
        // 在结构体中pt要放在m前面
        char pt = rtpHead.pt;
        char payload_str[10];
        switch(pt) {
            case 18: sprintf(payload_str,"Audio");break;
            case 31: sprintf(payload_str,"H.261");break;
            case 32: sprintf(payload_str,"MPV");break;
            case 33: sprintf(payload_str,"MP2T");break;
            case 34: sprintf(payload_str,"H.263");break;
            case 96: sprintf(payload_str,"H.264");break;
            default: sprintf(payload_str,"other");break;
        }

        unsigned int timestamp = ntohl(rtpHead.ts);
        unsigned int seq_no = ntohs(rtpHead.seq);
        printf("[RTP packet] %5d| %6s| %10u| %5d| %5d|\n", cnt++, payload_st
r, timestamp, seq_no, pktsize);
    
        // 解析MPEG-TS头信息
        char *rtp_data = recvbuf + rtphead_size;
        int rtp_data_size = pktsize - rtphead_size;

        if (pt == 33) {
            // mpegts_header_t tsHead;
            // memcpy((void *)&tsHead, &rtp_data[i], sizeof(tsHead));
            // 逐一解析包头的每个字段...
            for (int i = 0; i < rtp_data_size; i += 188) {
                if (rtp_data[i] != 0x47) // 检验同步字节
                    break;
                printf("[MPEG-TS packet]\n");
            }
        }
    }

    close(sockfd);
}

ffmpeg推流命令:ffmpeg -re -i xxx.ts -f rtp_mpegts udp://127.0.0.1:1998

测试结果:
image-20210501124600219

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
GB28181是中国国家标准化委员会发布的一项标准,用于统一视频监控系统及设备的通信协议。在GB28181标准中,音视频的发送主要使用的是实时传输协议(Real-time Transport Protocol,简称RTP)。 RTP是一种用于实时数据传输的协议,主要用于音频、视频等实时数据的传输,其实质是一种应用层协议RTP音视频数据按照时间戳分割为小的数据包,并加入一些额外的信息,如传输序列号、时间戳、同步信号等,以保证传输过程的实时性和准确性。 在GB28181中,音视频设备如摄像头、麦克风等通过网络将音视频数据封装成RTP数据包进行传输。发送端会根据数据类型(音频、视频)设定不同的RTP端口,将数据经过RTP协议封装成RTP数据包,并通过UDP或者TCP等传输层协议发送到接收端。 接收端根据RTP数据包中的头部信息解析音视频数据,并进行相应的解码和处理,最终通过显示设备或者扬声器播放出来。通过RTP协议的使用,音视频数据能够以实时、高效的方式进行传输,保证了监控系统的稳定性和可靠性。 总结来说,GB28181标准中的音视频发送主要使用RTP协议进行数据的封装和传输,通过UDP或者TCP等传输层协议RTP数据包发送到接收端,并通过解码处理后播放出来。这种方式保证了音视频数据的实时性和准确性,满足了监控系统对于实时性、稳定性和可靠性的要求。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值