TsMuxer.cpp 6.33 KB
Newer Older
xiongziliang committed
1
/*
xiongziliang committed
2
 * Copyright (c) 2016 The ZLMediaKit project authors. All Rights Reserved.
3 4 5
 *
 * This file is part of ZLMediaKit(https://github.com/xiongziliang/ZLMediaKit).
 *
xiongziliang committed
6 7 8
 * Use of this source code is governed by MIT license that can be found in the
 * LICENSE file in the root of the source tree. All contributing project authors
 * may be found in the AUTHORS file in the root of the source tree.
9 10
 */

11
#if defined(ENABLE_HLS)
12
#include "TsMuxer.h"
13
#include "mpeg-ts-proto.h"
14
#include "mpeg-ts.h"
15
#include "Extension/H264.h"
16 17 18 19 20 21 22 23 24 25 26

namespace mediakit {

TsMuxer::TsMuxer() {
    init();
}

TsMuxer::~TsMuxer() {
    uninit();
}

xiongziliang committed
27
void TsMuxer::stampSync(){
28
    if (_codec_to_trackid.size() < 2) {
xiongziliang committed
29 30 31 32
        return;
    }

    Stamp *audio = nullptr, *video = nullptr;
33 34
    for (auto &pr : _codec_to_trackid) {
        switch (getTrackType((CodecId) pr.first)) {
xiongziliang committed
35 36 37 38 39 40
            case TrackAudio : audio = &pr.second.stamp; break;
            case TrackVideo : video = &pr.second.stamp; break;
            default : break;
        }
    }

41
    if (audio && video) {
xiongziliang committed
42 43 44 45 46
        //音频时间戳同步于视频,因为音频时间戳被修改后不影响播放
        audio->syncTo(*video);
    }
}

47
void TsMuxer::addTrack(const Track::Ptr &track) {
48
    switch (track->getCodecId()) {
49
        case CodecH264: {
50
            _have_video = true;
xiongziliang committed
51
            _codec_to_trackid[track->getCodecId()].track_id = mpeg_ts_add_stream(_context, PSI_STREAM_H264, nullptr, 0);
52
            break;
xiongziliang committed
53 54
        }

55
        case CodecH265: {
56
            _have_video = true;
xiongziliang committed
57
            _codec_to_trackid[track->getCodecId()].track_id = mpeg_ts_add_stream(_context, PSI_STREAM_H265, nullptr, 0);
58
            break;
xiongziliang committed
59 60
        }

61
        case CodecAAC: {
xiongziliang committed
62
            _codec_to_trackid[track->getCodecId()].track_id = mpeg_ts_add_stream(_context, PSI_STREAM_AAC, nullptr, 0);
xiongziliang committed
63
            break;
64
        }
xiongziliang committed
65 66 67

        case CodecG711A: {
            _codec_to_trackid[track->getCodecId()].track_id = mpeg_ts_add_stream(_context, PSI_STREAM_AUDIO_G711A, nullptr, 0);
68
            break;
xiongziliang committed
69
        }
70 71

        case CodecG711U: {
xiongziliang committed
72
            _codec_to_trackid[track->getCodecId()].track_id = mpeg_ts_add_stream(_context, PSI_STREAM_AUDIO_G711U, nullptr, 0);
73
            break;
xiongziliang committed
74
        }
75

xiongziliang committed
76 77 78 79 80
        case CodecOpus: {
            _codec_to_trackid[track->getCodecId()].track_id = mpeg_ts_add_stream(_context, PSI_STREAM_AUDIO_OPUS, nullptr, 0);
            break;
        }

xiongziliang committed
81
        default: WarnL << "mpeg-ts 不支持该编码格式,已忽略:" << track->getCodecName(); break;
82
    }
xiongziliang committed
83 84 85

    //尝试音视频同步
    stampSync();
86 87 88
}

void TsMuxer::inputFrame(const Frame::Ptr &frame) {
89
    auto it = _codec_to_trackid.find(frame->getCodecId());
90
    if (it == _codec_to_trackid.end()) {
91 92
        return;
    }
93 94
    auto &track_info = it->second;
    int64_t dts_out, pts_out;
95
    _is_idr_fast_packet = !_have_video;
96 97
    switch (frame->getCodecId()){
        case CodecH264: {
98 99
            int type = H264_TYPE(*((uint8_t *) frame->data() + frame->prefixSize()));
            if (type == H264Frame::NAL_SEI) {
100 101 102
                break;
            }
        }
103

104
        case CodecH265: {
105 106 107 108
            //这里的代码逻辑是让SPS、PPS、IDR这些时间戳相同的帧打包到一起当做一个帧处理,
            if (!_frameCached.empty() && _frameCached.back()->dts() != frame->dts()) {
                Frame::Ptr back = _frameCached.back();
                Buffer::Ptr merged_frame = back;
109
                if (_frameCached.size() != 1) {
110 111
                    BufferLikeString merged;
                    merged.reserve(back->size() + 1024);
112 113 114 115 116 117
                    _frameCached.for_each([&](const Frame::Ptr &frame) {
                        if (frame->prefixSize()) {
                            merged.append(frame->data(), frame->size());
                        } else {
                            merged.append("\x00\x00\x00\x01", 4);
                            merged.append(frame->data(), frame->size());
118
                        }
119
                        if (frame->keyFrame()) {
120 121
                            _is_idr_fast_packet = true;
                        }
122
                    });
123
                    merged_frame = std::make_shared<BufferOffset<BufferLikeString> >(std::move(merged));
124
                }
125 126
                track_info.stamp.revise(back->dts(), back->pts(), dts_out, pts_out);
                //取视频时间戳为TS的时间戳
127
                _timestamp = dts_out;
128
                mpeg_ts_write(_context, track_info.track_id, back->keyFrame() ? 0x0001 : 0, pts_out * 90LL,dts_out * 90LL, merged_frame->data(), merged_frame->size());
129
                _frameCached.clear();
130
            }
131
            _frameCached.emplace_back(Frame::getCacheAbleFrame(frame));
132
            break;
133
        }
134 135 136

        case CodecAAC: {
            if (frame->prefixSize() == 0) {
xiongziliang committed
137
                WarnL << "必须提供adts头才能mpeg-ts打包";
138 139 140 141
                break;
            }
        }

142
        default: {
143 144 145 146 147
            track_info.stamp.revise(frame->dts(), frame->pts(), dts_out, pts_out);
            if(!_have_video){
                //没有视频时,才以音频时间戳为TS的时间戳
                _timestamp = dts_out;
            }
148
            mpeg_ts_write(_context, track_info.track_id, frame->keyFrame() ? 0x0001 : 0, pts_out * 90LL, dts_out * 90LL, frame->data(), frame->size());
149
            break;
150
        }
151 152 153 154
    }
}

void TsMuxer::resetTracks() {
155
    _have_video = false;
156
    //通知片段中断
157
    onTs(nullptr, 0, _timestamp, 0);
158 159 160 161 162
    uninit();
    init();
}

void TsMuxer::init() {
xiongziliang committed
163 164 165
    static mpeg_ts_func_t s_func = {
            [](void *param, size_t bytes) {
                TsMuxer *muxer = (TsMuxer *) param;
166
                assert(sizeof(TsMuxer::_tsbuf) >= bytes);
xiongziliang committed
167
                return (void *) muxer->_tsbuf;
168
            },
xiongziliang committed
169
            [](void *param, void *packet) {
170 171
                //do nothing
            },
xiongziliang committed
172 173 174
            [](void *param, const void *packet, size_t bytes) {
                TsMuxer *muxer = (TsMuxer *) param;
                muxer->onTs(packet, bytes, muxer->_timestamp, muxer->_is_idr_fast_packet);
175
                muxer->_is_idr_fast_packet = false;
xiongziliang committed
176
                return 0;
177 178
            }
    };
179 180
    if (_context == nullptr) {
        _context = mpeg_ts_create(&s_func, this);
181 182 183 184
    }
}

void TsMuxer::uninit() {
xiongziliang committed
185
    if (_context) {
186 187 188
        mpeg_ts_destroy(_context);
        _context = nullptr;
    }
189
    _codec_to_trackid.clear();
190 191
}

192 193
}//namespace mediakit
#endif// defined(ENABLE_HLS)