FFmpeg解析TS私有文本流

 TS包的结构如下:

本文在FFmpeg 4.3.2的基础上修改avformat/mpegts.c中的handle_packet函数,添加了从188个字节的TS包中解析出,pid为0x12的私有流信息。代码做的事情主要是:跳过TS头部4个字节,跳过adaptation填充字节,然后跳过PES的头部,最后找到ES的Payload。

其中PES的结构如下:

 

 

在我的另外一篇博文里也有详细介绍:PES包结构解析_iChenwin的博客-CSDN博客 

最后上代码: 

/* handle one TS packet */
static int handle_packet(MpegTSContext *ts, AVFormatContext *fmt, const uint8_t *packet, int64_t pos)
{
    MpegTSFilter *tss;
    int len, pid, cc, expected_cc, cc_ok, afc, is_start, is_discontinuity,
        has_adaptation, has_payload;
    const uint8_t *p, *p_end;

    pid = AV_RB16(packet + 1) & 0x1fff;

    is_start = packet[1] & 0x40;

    // 私有流解析
    if (fmt != NULL && pid == 0x12 && is_start)
    {
        // The packet contains the desired data.
        int payload_offset = 4;                        //跳过TS头部4个字节
        if ((packet[3] >> 5) & 0x01) {                 //解析Adaptation Field字段,并跳过
            int adaptation_field_length = packet[payload_offset] + 1;
            payload_offset += adaptation_field_length;
        }
        //pes解析:https://blog.csdn.net/ichenwin/article/details/84946333
        payload_offset += 4; 							//跳过PES的起始码00 00 01 BD
        int pes_len = (packet[payload_offset]<<8) | (packet[payload_offset+1]);
        payload_offset += 2; 							//跳过PES的长度表示位

        int pes_header_left_len = packet[payload_offset+2];
        int es_len = pes_len - 3 - pes_header_left_len;		
        payload_offset += 3; 							    //跳过pes头前面3字节
        payload_offset += pes_header_left_len; 				//跳过pes头剩余字节,到达es数据

        const uint8_t *es_payload = packet + payload_offset;

        // Parse the es data as UTF-8.
        char es_data[188];
        memcpy(es_data, es_payload, es_len);
        es_data[es_len] = '\0';

        // parse_private_json(fmt, es_data);           //到这里已经完全解出自己的私有流,我的是json,所以调用自己写的json解析。

        //av_log(ts->stream, AV_LOG_ERROR, "pid %04x, payload:%02x %02x %02x %02x %02x %02x, len:%d,%d, ts:%s\n", pid,
        //	packet[0], packet[1], packet[2], packet[3], packet[4], packet[5], pes_len, es_len, text_data);
    }


    tss = ts->pids[pid];
    if (ts->auto_guess && !tss && is_start) {
        add_pes_stream(ts, pid, -1);
        tss = ts->pids[pid];
    }
    if (!tss)
        return 0;
    if (is_start)
        tss->discard = discard_pid(ts, pid);
    if (tss->discard)
        return 0;
    ts->current_pid = pid;

    afc = (packet[3] >> 4) & 3;
    if (afc == 0) /* reserved value */
        return 0;
    has_adaptation   = afc & 2;
    has_payload      = afc & 1;
    is_discontinuity = has_adaptation &&
                        packet[4] != 0 && /* with length > 0 */
                        (packet[5] & 0x80); /* and discontinuity indicated */

    /* continuity check (currently not used) */
    cc = (packet[3] & 0xf);
    expected_cc = has_payload ? (tss->last_cc + 1) & 0x0f : tss->last_cc;
    cc_ok = pid == 0x1FFF || // null packet PID
            is_discontinuity ||
            tss->last_cc < 0 ||
            expected_cc == cc;

    tss->last_cc = cc;
    if (!cc_ok) {
        av_log(ts->stream, AV_LOG_DEBUG,
                "Continuity check failed for pid %d expected %d got %d\n",
                pid, expected_cc, cc);
        if (tss->type == MPEGTS_PES) {
            PESContext *pc = tss->u.pes_filter.opaque;
            pc->flags |= AV_PKT_FLAG_CORRUPT;
        }
    }

    if (packet[1] & 0x80) {
        av_log(ts->stream, AV_LOG_DEBUG, "Packet had TEI flag set; marking as corrupt\n");
        if (tss->type == MPEGTS_PES) {
            PESContext *pc = tss->u.pes_filter.opaque;
            pc->flags |= AV_PKT_FLAG_CORRUPT;
        }
    }

    p = packet + 4;
    if (has_adaptation) {
        int64_t pcr_h;
        int pcr_l;
        if (parse_pcr(&pcr_h, &pcr_l, packet) == 0)
            tss->last_pcr = pcr_h * 300 + pcr_l;
        /* skip adaptation field */
        p += p[0] + 1;
    }
    /* if past the end of packet, ignore */
    p_end = packet + TS_PACKET_SIZE;
    if (p >= p_end || !has_payload)
        return 0;

    if (pos >= 0) {
        av_assert0(pos >= TS_PACKET_SIZE);
        ts->pos47_full = pos - TS_PACKET_SIZE;
    }

    if (tss->type == MPEGTS_SECTION) {
        if (is_start) {
            /* pointer field present */
            len = *p++;
            if (len > p_end - p)
                return 0;
            if (len && cc_ok) {
                /* write remaining section bytes */
                write_section_data(ts, tss,
                                    p, len, 0);
                /* check whether filter has been closed */
                if (!ts->pids[pid])
                    return 0;
            }
            p += len;
            if (p < p_end) {
                write_section_data(ts, tss,
                                    p, p_end - p, 1);
            }
        } else {
            if (cc_ok) {
                write_section_data(ts, tss,
                                    p, p_end - p, 0);
            }
        }

        // stop find_stream_info from waiting for more streams
        // when all programs have received a PMT
        if (ts->stream->ctx_flags & AVFMTCTX_NOHEADER && ts->scan_all_pmts <= 0) {
            int i;
            for (i = 0; i < ts->nb_prg; i++) {
                if (!ts->prg[i].pmt_found)
                    break;
            }
            if (i == ts->nb_prg && ts->nb_prg > 0) {
                int types = 0;
                for (i = 0; i < ts->stream->nb_streams; i++) {
                    AVStream *st = ts->stream->streams[i];
                    if (st->codecpar->codec_type >= 0)
                        types |= 1<<st->codecpar->codec_type;
                }
                if ((types & (1<<AVMEDIA_TYPE_AUDIO) && types & (1<<AVMEDIA_TYPE_VIDEO)) || pos > 100000) {
                    av_log(ts->stream, AV_LOG_DEBUG, "All programs have pmt, headers found\n");
                    ts->stream->ctx_flags &= ~AVFMTCTX_NOHEADER;
                }
            }
        }

    } else {
        int ret;
        // Note: The position here points actually behind the current packet.
        if (tss->type == MPEGTS_PES) {
            if ((ret = tss->u.pes_filter.pes_cb(tss, p, p_end - p, is_start,
                                                pos - ts->raw_packet_size)) < 0)
                return ret;
        }
    }

    return 0;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
FFmpeg是一个跨平台的音视频处理工具,能够对各种格式的音视频文件进行编码、解码、转码等操作。而MPEG-TS是一种常用的音视频传输格式,通常用于广播、电视等领域。 在FFmpeg中,可以通过命令行参数来实现对MPEG-TS的分操作。首先,需要通过输入参数指定要处理的MPEG-TS文件,例如: ``` ffmpeg -i input.ts ``` 然后,可以通过选择要分离的的索引号来实现分操作。通过使用"-map"参数加上的索引号,可以将特定的输出为新的MPEG-TS文件。例如,下面的命令可以将输入文件中的第一个视频和第一个音频输出为新的MPEG-TS文件: ``` ffmpeg -i input.ts -map 0:0 -map 0:1 -c copy output.ts ``` 上述命令中的"-map 0:0"表示选择输入文件中的第一个视频(0号输入的第一个输出),"-map 0:1"表示选择输入文件中的第一个音频(0号输入的第二个输出)。"-c copy"表示直接复制选定的而不进行重新编码。 除了选择特定的,还可以将多个合并为一个输出文件。通过使用"-map"参数加上合适的选项,可以实现多个的合并。例如,下面的命令可以将输入文件中的第一个视频和第二个音频合并为一个新的MPEG-TS文件: ``` ffmpeg -i input.ts -map 0:0 -map 0:1 -c copy output.ts ``` 上述命令中的"-map 0:0"表示选择输入文件中的第一个视频(0号输入的第一个输出),"-map 0:1"表示选择输入文件中的第二个音频(0号输入的第二个输出)。 通过以上的命令和参数,可以很方便地使用FFmpeg对MPEG-TS进行分操作,实现根据需要选择和合并音视频的功能。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值