h264、h265、svac封装成ps流符合gb28181过检码流要求

数据安全 数据安全 2074 人阅读 | 0 人回复

发表于 2022-12-2 18:08:54 | 显示全部楼层 |阅读模式

简单的ps流封装,做分享笔记:
  1. #include <stdint.h>
  2. #include <string>
  3. #include <memory.h>

  4. #define H264_ID        0x1b
  5. #define H265_ID        0x24
  6. #define MPEG_ID        0x10
  7. #define SVACV_ID       0x80

  8. #define G711_ID        0x90
  9. #define SVACA_ID       0x9b

  10. const uint8_t PS_HEAD[] = {
  11.     /*PS头*/
  12.     0x00, 0x00, 0x01, 0xba,
  13.     0x00, 0x00, 0x00, 0x00, 0x00, 0x00,        /*时间戳*/
  14.     0x01, 0x47, 0xb3,
  15.     0xf8
  16. };

  17. const uint8_t SYS_MAP_HEAD[] = {
  18.     /*PS_SYS头*/
  19.     0x00, 0x00, 0x01, 0xbb,
  20.     0x00, 0x0c,                                /*sys头长度,不含自己,6+3*流的数目*/
  21.     0x80, 0xa3, 0xd9,                          /*速率*/
  22.     0x04, 0xe1,                                /*音频流数,视频流数加3个1标识*/
  23.     0xff,                                      /**/
  24.     0xb9, 0xe0, 0x00, 0xb8, 0xc0, 0x40,        /*流信息,b9视频,b8音频*/
  25.     /*PS_MAP头*/
  26.     0x00, 0x00, 0x01, 0xbc,
  27.     0x00, 0x12,                                /*psm长度*/
  28.     0xe1, 0xff,                                /**/
  29.     0x00, 0x00, 0x00, 0x08,                    /*固定2路流*/
  30.     0x1b, 0xe0, 0x00, 0x00,                    /*视频,第一个字节(0x1b), 跟具不同的视频编码改变即可封装不同的流,见开头宏定义*/
  31.     0x90, 0xc0, 0x00, 0x00,                    /*音频,同视频*/
  32.     0x00, 0x00, 0x00, 0x00                     /*4b CRC,暂时没设置*/
  33. };

  34. const uint8_t PES_HEAD[] = {
  35.     /*PS_PES头*/
  36.     0x00, 0x00, 0x01, 0xe0,
  37.     0x00, 0x00,                                /*pes长度*/
  38.     0x80, 0xc0,                                /*附加信息*/
  39.     0x0a,                                      /*附加信息长度*/
  40.     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00  /*pts和pds*/
  41. };

  42. void SetHeaderTimeStamp(uint8_t *dest, uint64_t pts)
  43. {
  44.     uint8_t *scr_buf = dest + 4;
  45.     scr_buf[0] = 0x40 | (((uint8_t)(pts >> 30) & 0x07) << 3) | 0x04 | ((uint8_t)(pts >> 28) & 0x03);
  46.     scr_buf[1] = (uint8_t)((pts >> 20) & 0xff);
  47.     scr_buf[2] = (((uint8_t)(pts >> 15) & 0x1f) << 3) | 0x04 | ((uint8_t)(pts >> 13) & 0x03);
  48.     scr_buf[3] = (uint8_t)((pts >> 5) & 0xff);
  49.     scr_buf[4] = (((uint8_t)pts & 0x1f) << 3) | 0x04;
  50.     scr_buf[5] = 1;
  51. }

  52. // 设置PES头中的PTS和DTS字段
  53. void SetPESTimeStamp(uint8_t *buff, uint64_t ts)
  54. {
  55.     buff += 9;
  56.     // PTS
  57.     buff[0] = (uint8_t)(((ts >> 30) & 0x07) << 1) | 0x30 | 0x01;
  58.     buff[1] = (uint8_t)((ts >> 22) & 0xff);
  59.     buff[2] = (uint8_t)(((ts >> 15) & 0xff) << 1) | 0x01;
  60.     buff[3] = (uint8_t)((ts >> 7) & 0xff);
  61.     buff[4] = (uint8_t)((ts & 0xff) << 1) | 0x01;
  62.     // DTS
  63.     buff[5] = (uint8_t)(((ts >> 30) & 0x07) << 1) | 0x10 | 0x01;
  64.     buff[6] = (uint8_t)((ts >> 22) & 0xff);
  65.     buff[7] = (uint8_t)(((ts >> 15) & 0xff) << 1) | 0x01;
  66.     buff[8] = (uint8_t)((ts >> 7) & 0xff);
  67.     buff[9] = (uint8_t)((ts & 0xff) << 1) | 0x01;
  68. }

  69. int GetSinglePESHeader(uint8_t *header, uint64_t mtime, uint16_t farmLen)
  70. {
  71.     farmLen += 13;
  72.     memcpy(header, PES_HEAD, sizeof(PES_HEAD));
  73.     *(header+4) = (uint8_t)(farmLen>>8);
  74.     *(header+5) = (uint8_t)farmLen;

  75.     SetPESTimeStamp(header, mtime);
  76.     return sizeof(PES_HEAD);
  77. }

  78. int GetPSHeader(uint8_t *header, uint64_t mtime, uint16_t farmLen, int streamType, int farmType)
  79. {
  80.     if(streamType == 2)        //语音包
  81.     {
  82.         GetSinglePESHeader(header, mtime, farmLen);
  83.         *(header+3) = 0xc0;
  84.         return sizeof(PES_HEAD);
  85.     }
  86.     else if(farmType == 1)    //I帧
  87.     {
  88.         memcpy(header, PS_HEAD, sizeof(PS_HEAD));
  89.         SetHeaderTimeStamp(header, mtime);
  90.         header += sizeof(PS_HEAD);

  91.         memcpy(header, SYS_MAP_HEAD, sizeof(SYS_MAP_HEAD));
  92.         header += sizeof(SYS_MAP_HEAD);

  93.         GetSinglePESHeader(header, mtime, farmLen);
  94.         return sizeof(PS_HEAD) + sizeof(SYS_MAP_HEAD) + sizeof(PES_HEAD);
  95.     }
  96.     else
  97.     {
  98.         memcpy(header, PS_HEAD, sizeof(PS_HEAD));
  99.         SetHeaderTimeStamp(header, mtime);
  100.         header += sizeof(PS_HEAD);

  101.         GetSinglePESHeader(header, mtime, farmLen);
  102.         return sizeof(PS_HEAD) + sizeof(PES_HEAD);
  103.     }
  104. }

  105. unsigned char PSFrameBuffer[10*1024*1024]; //转换后的ps帧缓存区
  106. int TransPSFrame(char *pFrame, int nFrameLength, int nIFrameFlag, int nStreamType, u_int nTimeStamp)
  107. {
  108.     if(!pFrame || !nFrameLength)
  109.         return 0;

  110.     // 每个pes最多65400数据
  111.     int PesLenth = nFrameLength > 65400 ? 65400 : nFrameLength;   
  112.     // 第一个pes需要有ps头,其它不需要,音频直接打包pes(00 00 01 c0)
  113.     int psHeadLen = GetPSHeader(PSFrameBuffer, nTimeStamp, PesLenth, nStreamType, nIFrameFlag);
  114.     memcpy(PSFrameBuffer + psHeadLen, pFrame, PesLenth);
  115.     int psSize = psHeadLen + PesLenth;
  116.     int pod = PesLenth;

  117.     nFrameLength -= PesLenth;
  118.     while (nFrameLength > 0)
  119.     {
  120.         PesLenth = nFrameLength > 65400 ? 65400 : nFrameLength;
  121.         psHeadLen = GetSinglePESHeader(PSFrameBuffer + psSize, nTimeStamp, PesLenth);

  122.         memcpy(PSFrameBuffer + psSize + psHeadLen, pFrame + pod, PesLenth);
  123.         psSize += (PesLenth + psHeadLen);   
  124.         pod += PesLenth;
  125.         nFrameLength -= PesLenth;
  126.     }

  127.     //static FILE *fp = fopen("my.ps", "wb");
  128.     //if(fp)
  129.     //    fwrite(PSFrameBuffer, 1, psSize, fp);

  130.     return psSize;
  131. }
复制代码
(之前的固定字段中current_next_indicator 和vesion填了0和5导致打包出来的ps流vlc不能播放,已修复)

TransPSFrame函数为测试所用,应用时宜将得到的ps头部和帧数据直接写到rtp包中,省去memcpy过程。

SYS_MAP_HEAD中的数据是按国标ps流标准写死的,如果要封装格式数据需要修改。

封装h265和svac音频和svac视频时需要修改SYS_MAP_HEAD[30]和34,见注释和本文开头的宏定义

00 00 01 B9 为ps流结束标识符

附加各种编码对应的代码置(SYS_MAP_HEAD[30]和[34]的值):
  1.     PSMUX_ST_RESERVED                      = 0x00,
  2.     PSMUX_ST_VIDEO_MPEG1                = 0x01,
  3.     PSMUX_ST_VIDEO_MPEG2                = 0x02,
  4.     PSMUX_ST_AUDIO_MPEG1                = 0x03,
  5.     PSMUX_ST_AUDIO_MPEG2                = 0x04,
  6.     PSMUX_ST_PRIVATE_SECTIONS      = 0x05,
  7.     PSMUX_ST_PRIVATE_DATA                = 0x06,
  8.     PSMUX_ST_MHEG                                 = 0x07,
  9.     PSMUX_ST_DSMCC                              = 0x08,
  10.     PSMUX_ST_H222_1                               = 0x09,
  11.     PSMUX_ST_AUDIO_AAC                      = 0x0f,
  12.     PSMUX_ST_VIDEO_MPEG4                = 0x10,
  13.     PSMUX_ST_VIDEO_H264                     = 0x1b,
  14.     PSMUX_ST_VIDEO_H265                     = 0x24,
  15.     PSMUX_ST_PS_VIDEO_SVAC            = 0x80,
  16.     PSMUX_ST_PS_AUDIO_AC3               = 0x81,
  17.     PSMUX_ST_PS_AUDIO_DTS               = 0x8a,
  18.     PSMUX_ST_PS_AUDIO_LPCM              = 0x8b,
  19.     PSMUX_ST_PS_AUDIO_G711A             = 0x90,
  20.     PSMUX_ST_PS_AUDIO_G711U             = 0x91,
  21.     PSMUX_ST_PS_AUDIO_G722_1            = 0x92,
  22.     PSMUX_ST_PS_AUDIO_G723_1            = 0x93,
  23.     PSMUX_ST_PS_AUDIO_G729                = 0x99,
  24.     PSMUX_ST_PS_AUDIO_SVAC               = 0x9b,
  25.     PSMUX_ST_PS_DVD_SUBPICTURE     = 0xff,
  26.     //下面定义不是标准里面定义的
  27.     PSMUX_ST_VIDEO_DIRAC                      = 0xD1
复制代码


其实只需要将以上代码保持为一个ps.hpp文件,添加以下代码即可。
  1. #include "stdafx.h"
  2. #include <stdlib.h>
  3. #include "ps.hpp"

  4. int ReadSourceStream(char **frame_data, int *frame_size)
  5. {
  6.         static FILE *fp = NULL;
  7.         if (fp == NULL)
  8.         {
  9.                 fp = fopen("es_stream.h264", "rb");
  10.                 if (fp == NULL)
  11.                 {
  12.                         printf("open file es_stream.h264 error\n");
  13.                         return -1;
  14.                 }
  15.         }

  16.         //文件格式为:
  17.         //     1    2    3    4         n
  18.         //|----|----|----|----|---------
  19.         //|       帧长度       |   数据   
  20.         do {
  21.                 if (4 != fread(frame_size, 1, 4, fp))
  22.                         break;
  23.                 *frame_data = (char*)malloc(*frame_size);
  24.                 if (*frame_size != fread(*frame_data, 1, *frame_size, fp))
  25.                         break;
  26.                 return 0;
  27.         } while (0);
  28.         printf("read file es_stream.h264 eof\n");
  29.         fclose(fp);
  30.         fp = NULL;
  31.         return -2;
  32. }

  33. int KeyFrame(uint8_t nalu)
  34. {
  35.         int nalType = nalu & 0x1f;
  36.         if (nalType == 0x07 || nalType == 0x05 || nalType == 0x08)
  37.                 return 1;
  38.         return 0;
  39. }

  40. int main()
  41. {
  42.         char *frame_data;
  43.         int frame_size, ps_size;
  44.         uint32_t ts = 0;

  45.         FILE *fp = fopen("my_h264.ps", "wb");
  46.         while (0 == ReadSourceStream(&frame_data, &frame_size))
  47.         {
  48.                 // 文件中只有h264视频,第4个参数固定填1,音频为2
  49.                 ps_size = TransPSFrame(frame_data, frame_size, KeyFrame((uint8_t)frame_data[4]), 1, ts);
  50.                 fwrite(PSFrameBuffer, 1, ps_size, fp);
  51.                 ts += 3600;
  52.                 free(frame_data);
  53.         }
  54.         fclose(fp);
  55.         getchar();
  56.     return 0;
  57. }
复制代码


回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则