MediaSink.cpp 4.5 KB
Newer Older
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
 */
xiongziliang committed
10

11 12
#include "MediaSink.h"

13
//最多等待未初始化的Track 10秒,超时之后会忽略未初始化的Track
14 15 16 17 18
#define MAX_WAIT_MS_READY 10000

//如果添加Track,最多等待3秒
#define MAX_WAIT_MS_ADD_TRACK 3000

19 20 21 22

namespace mediakit{

void MediaSink::addTrack(const Track::Ptr &track_in) {
23
    lock_guard<recursive_mutex> lck(_mtx);
24
    //克隆Track,只拷贝其数据,不拷贝其数据转发关系
25 26 27
    auto track = track_in->clone();
    auto codec_id = track->getCodecId();
    _track_map[codec_id] = track;
28 29
    _allTrackReady = false;
    _trackReadyCallback[codec_id] = [this, track]() {
30 31
        onTrackReady(track);
    };
32
    _ticker.resetTime();
xiongziliang committed
33

34 35
    track->addDelegate(std::make_shared<FrameWriterInterfaceHelper>([this](const Frame::Ptr &frame) {
        if (_allTrackReady) {
xiongziliang committed
36
            onTrackFrame(frame);
xiongziliang committed
37 38
        }
    }));
39 40
}

41
void MediaSink::resetTracks() {
xiongziliang committed
42
    lock_guard<recursive_mutex> lck(_mtx);
43 44 45 46
    _allTrackReady = false;
    _track_map.clear();
    _trackReadyCallback.clear();
    _ticker.resetTime();
47
    _max_track_size = 2;
48 49
}

50
void MediaSink::inputFrame(const Frame::Ptr &frame) {
51
    lock_guard<recursive_mutex> lck(_mtx);
52
    auto it = _track_map.find(frame->getCodecId());
53 54 55 56
    if (it == _track_map.end()) {
        return;
    }
    it->second->inputFrame(frame);
57 58 59 60 61 62 63 64 65 66 67
    checkTrackIfReady(it->second);
}

void MediaSink::checkTrackIfReady_l(const Track::Ptr &track){
    //Track由未就绪状态转换成就绪状态,我们就触发onTrackReady回调
    auto it_callback = _trackReadyCallback.find(track->getCodecId());
    if (it_callback != _trackReadyCallback.end() && track->ready()) {
        it_callback->second();
        _trackReadyCallback.erase(it_callback);
    }
}
68

69 70 71 72 73 74 75 76
void MediaSink::checkTrackIfReady(const Track::Ptr &track){
    if (!_allTrackReady && !_trackReadyCallback.empty()) {
        if (track) {
            checkTrackIfReady_l(track);
        } else {
            for (auto &pr : _track_map) {
                checkTrackIfReady_l(pr.second);
            }
77 78 79
        }
    }

80 81 82
    if(!_allTrackReady){
        if(_ticker.elapsedTime() > MAX_WAIT_MS_READY){
            //如果超过规定时间,那么不再等待并忽略未准备好的Track
83
            emitAllTrackReady();
84 85 86
            return;
        }

87
        if(!_trackReadyCallback.empty()){
88 89
            //在超时时间内,如果存在未准备好的Track,那么继续等待
            return;
90 91
        }

92
        if(_track_map.size() == _max_track_size){
93
            //如果已经添加了音视频Track,并且不存在未准备好的Track,那么说明所有Track都准备好了
94
            emitAllTrackReady();
95
            return;
96
        }
97 98 99

        if(_track_map.size() == 1 && _ticker.elapsedTime() > MAX_WAIT_MS_ADD_TRACK){
            //如果只有一个Track,那么在该Track添加后,我们最多还等待若干时间(可能后面还会添加Track)
100
            emitAllTrackReady();
101 102 103 104 105
            return;
        }
    }
}

106
void MediaSink::addTrackCompleted(){
107 108 109 110 111
    {
        lock_guard<recursive_mutex> lck(_mtx);
        _max_track_size = _track_map.size();
    }
    checkTrackIfReady(nullptr);
112 113 114
}

void MediaSink::emitAllTrackReady() {
115 116 117 118 119
    if (_allTrackReady) {
        return;
    }

    if (!_trackReadyCallback.empty()) {
120 121 122
        //这是超时强制忽略未准备好的Track
        _trackReadyCallback.clear();
        //移除未准备好的Track
123 124
        for (auto it = _track_map.begin(); it != _track_map.end();) {
            if (!it->second->ready()) {
125
                WarnL << "该track长时间未被初始化,已忽略:" << it->second->getCodecName();
126 127 128 129 130 131 132
                it = _track_map.erase(it);
                continue;
            }
            ++it;
        }
    }

133
    if (!_track_map.empty()) {
134
        //最少有一个有效的Track
135
        _allTrackReady = true;
136
        onAllTrackReady();
137 138 139
    }
}

xiongziliang committed
140 141
vector<Track::Ptr> MediaSink::getTracks(bool trackReady) const{
    vector<Track::Ptr> ret;
142
    lock_guard<recursive_mutex> lck(_mtx);
143
    for (auto &pr : _track_map){
xiongziliang committed
144 145
        if(trackReady && !pr.second->ready()){
            continue;
146
        }
xiongziliang committed
147
        ret.emplace_back(pr.second);
148
    }
xiongziliang committed
149
    return std::move(ret);
150 151
}

152 153

}//namespace mediakit