MediaSource.h 8.8 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 10 11 12 13 14 15 16 17 18 19
 */

#ifndef ZLMEDIAKIT_MEDIASOURCE_H
#define ZLMEDIAKIT_MEDIASOURCE_H

#include <mutex>
#include <string>
#include <memory>
#include <functional>
#include <unordered_map>
#include "Common/config.h"
xiongziliang committed
20
#include "Common/Parser.h"
21 22 23
#include "Util/logger.h"
#include "Util/TimeTicker.h"
#include "Util/NoticeCenter.h"
xiongziliang committed
24 25 26
#include "Util/List.h"
#include "Rtsp/Rtsp.h"
#include "Rtmp/Rtmp.h"
xiongziliang committed
27
#include "Extension/Track.h"
28
#include "Record/Recorder.h"
29 30

using namespace std;
xiongziliang committed
31
using namespace toolkit;
32

33 34
namespace toolkit{
    class TcpSession;
xiongziliang committed
35
}// namespace toolkit
36

xiongziliang committed
37
namespace mediakit {
38

39 40
class MediaSource;
class MediaSourceEvent{
41
public:
42
    friend class MediaSource;
43 44
    MediaSourceEvent(){};
    virtual ~MediaSourceEvent(){};
xiongziliang committed
45 46

    // 通知拖动进度条
xiongziliang committed
47 48 49 50
    virtual bool seekTo(MediaSource &sender, uint32_t stamp) { return false; }
    // 通知其停止产生流
    virtual bool close(MediaSource &sender, bool force) { return false; }
    // 获取观看总人数
51
    virtual int totalReaderCount(MediaSource &sender) = 0;
52 53
    // 通知观看人数变化
    virtual void onReaderChanged(MediaSource &sender, int size);
54 55
    //流注册或注销事件
    virtual void onRegist(MediaSource &sender, bool regist) {};
56

xiongziliang committed
57 58 59 60 61
    ////////////////////////仅供MultiMediaSourceMuxer对象继承////////////////////////
    // 开启或关闭录制
    virtual bool setupRecord(MediaSource &sender, Recorder::type type, bool start, const string &custom_path) { return false; };
    // 获取录制状态
    virtual bool isRecording(MediaSource &sender, Recorder::type type) { return false; };
62 63
    // 获取所有track相关信息
    virtual vector<Track::Ptr> getTracks(MediaSource &sender, bool trackReady = true) const { return vector<Track::Ptr>(); };
64 65 66 67
    // 开始发送ps-rtp
    virtual void startSendRtp(MediaSource &sender, const string &dst_url, uint16_t dst_port, uint32_t ssrc, bool is_udp, const function<void(const SockException &ex)> &cb) { cb(SockException(Err_other, "not implemented"));};
    // 停止发送ps-rtp
    virtual bool stopSendRtp(MediaSource &sender) {return false; }
xiongziliang committed
68

69 70
private:
    Timer::Ptr _async_close_timer;
71
};
72

xiongziliang committed
73 74 75 76 77 78 79 80 81
//该对象用于拦截感兴趣的MediaSourceEvent事件
class MediaSourceEventInterceptor : public MediaSourceEvent{
public:
    MediaSourceEventInterceptor(){}
    ~MediaSourceEventInterceptor() override {}

    bool seekTo(MediaSource &sender, uint32_t stamp) override;
    bool close(MediaSource &sender, bool force) override;
    int totalReaderCount(MediaSource &sender) override;
82
    void onReaderChanged(MediaSource &sender, int size) override;
83
    void onRegist(MediaSource &sender, bool regist) override;
xiongziliang committed
84 85
    bool setupRecord(MediaSource &sender, Recorder::type type, bool start, const string &custom_path) override;
    bool isRecording(MediaSource &sender, Recorder::type type) override;
86
    vector<Track::Ptr> getTracks(MediaSource &sender, bool trackReady = true) const override;
87 88
    void startSendRtp(MediaSource &sender, const string &dst_url, uint16_t dst_port, uint32_t ssrc, bool is_udp, const function<void(const SockException &ex)> &cb) override;
    bool stopSendRtp(MediaSource &sender) override;
xiongziliang committed
89 90 91 92 93

protected:
    std::weak_ptr<MediaSourceEvent> _listener;
};

94 95 96
/**
 * 解析url获取媒体相关信息
 */
97
class MediaInfo{
98
public:
xiongziliang committed
99 100 101
    ~MediaInfo() {}
    MediaInfo() {}
    MediaInfo(const string &url) { parse(url); }
102
    void parse(const string &url);
xiongziliang committed
103

104
public:
105 106 107 108 109 110
    string _schema;
    string _host;
    string _port;
    string _vhost;
    string _app;
    string _streamid;
111
    string _param_strs;
112 113
};

xiongziliang committed
114 115 116
/**
 * 媒体源,任何rtsp/rtmp的直播流都源自该对象
 */
xiongziliang committed
117
class MediaSource: public TrackSource, public enable_shared_from_this<MediaSource> {
118 119 120 121 122 123 124
public:
    typedef std::shared_ptr<MediaSource> Ptr;
    typedef unordered_map<string, weak_ptr<MediaSource> > StreamMap;
    typedef unordered_map<string, StreamMap > AppStreamMap;
    typedef unordered_map<string, AppStreamMap > VhostAppStreamMap;
    typedef unordered_map<string, VhostAppStreamMap > SchemaVhostAppStreamMap;

xiongziliang committed
125
    MediaSource(const string &schema, const string &vhost, const string &app, const string &stream_id) ;
xiongziliang committed
126 127
    virtual ~MediaSource() ;

128 129
    ////////////////获取MediaSource相关信息////////////////

xiongziliang committed
130 131 132 133 134 135 136 137 138
    // 获取协议类型
    const string& getSchema() const;
    // 虚拟主机
    const string& getVhost() const;
    // 应用名
    const string& getApp() const;
    // 流id
    const string& getId() const;

139
    // 获取所有Track
xiongziliang committed
140
    vector<Track::Ptr> getTracks(bool ready = true) const override;
141

142 143 144 145 146 147 148
    // 获取流当前时间戳
    virtual uint32_t getTimeStamp(TrackType type) { return 0; };
    // 设置时间戳
    virtual void setTimeStamp(uint32_t stamp) {};

    ////////////////MediaSourceEvent相关接口实现////////////////

xiongziliang committed
149
    // 设置监听者
150
    void setListener(const std::weak_ptr<MediaSourceEvent> &listener);
151 152 153
    // 获取监听者
    const std::weak_ptr<MediaSourceEvent>& getListener() const;

154
    // 本协议获取观看者个数,可能返回本协议的观看人数,也可能返回总人数
xiongziliang committed
155
    virtual int readerCount() = 0;
156 157
    // 观看者个数,包括(hls/rtsp/rtmp)
    virtual int totalReaderCount();
158

xiongziliang committed
159
    // 拖动进度条
xiongziliang committed
160
    bool seekTo(uint32_t stamp);
xiongziliang committed
161 162
    // 关闭该流
    bool close(bool force);
163 164
    // 该流观看人数变化
    void onReaderChanged(int size);
165
    // 开启或关闭录制
166
    bool setupRecord(Recorder::type type, bool start, const string &custom_path);
167
    // 获取录制状态
168
    bool isRecording(Recorder::type type);
169 170 171 172
    // 开始发送ps-rtp
    void startSendRtp(const string &dst_url, uint16_t dst_port, uint32_t ssrc, bool is_udp, const function<void(const SockException &ex)> &cb);
    // 停止发送ps-rtp
    bool stopSendRtp();
173

174 175
    ////////////////static方法,查找或生成MediaSource////////////////

xiongziliang committed
176
    // 同步查找流
177
    static Ptr find(const string &schema, const string &vhost, const string &app, const string &id);
178 179 180 181

    // 忽略类型,同步查找流,可能返回rtmp/rtsp/hls类型
    static Ptr find(const string &vhost, const string &app, const string &stream_id);

xiongziliang committed
182 183 184 185
    // 异步查找流
    static void findAsync(const MediaInfo &info, const std::shared_ptr<TcpSession> &session, const function<void(const Ptr &src)> &cb);
    // 遍历所有流
    static void for_each_media(const function<void(const Ptr &src)> &cb);
186
    // 从mp4文件生成MediaSource
xiongziliang committed
187
    static MediaSource::Ptr createFromMP4(const string &schema, const string &vhost, const string &app, const string &stream, const string &file_path = "", bool check_app = true);
188

189
protected:
xiongziliang committed
190 191
    //媒体注册
    void regist();
192 193

private:
xiongziliang committed
194 195 196 197 198
    //媒体注销
    bool unregist();
    //触发媒体事件
    void emitEvent(bool regist);

199
private:
xiongziliang committed
200 201 202 203 204
    string _schema;
    string _vhost;
    string _app;
    string _stream_id;
    std::weak_ptr<MediaSourceEvent> _listener;
205 206
};

xiongziliang committed
207 208 209
///缓存刷新策略类
class FlushPolicy {
public:
210
    FlushPolicy() = default;
xiongziliang committed
211 212 213 214 215 216 217
    ~FlushPolicy() = default;

    uint32_t getStamp(const RtpPacket::Ptr &packet) {
        return packet->timeStamp;
    }

    uint32_t getStamp(const RtmpPacket::Ptr &packet) {
xiongziliang committed
218
        return packet->time_stamp;
xiongziliang committed
219 220
    }

221 222
    bool isFlushAble(bool is_video, bool is_key, uint32_t new_stamp, int cache_size);

xiongziliang committed
223
private:
224
    uint32_t _last_stamp[2] = {0, 0};
xiongziliang committed
225 226
};

227
/// 合并写缓存模板
xiongziliang committed
228 229 230 231
/// \tparam packet 包类型
/// \tparam policy 刷新缓存策略
/// \tparam packet_list 包缓存类型
template<typename packet, typename policy = FlushPolicy, typename packet_list = List<std::shared_ptr<packet> > >
232
class PacketCache {
xiongziliang committed
233
public:
234
    PacketCache(){
xiongziliang committed
235 236 237
        _cache = std::make_shared<packet_list>();
    }

238
    virtual ~PacketCache() = default;
xiongziliang committed
239

240 241
    void inputPacket(bool is_video, const std::shared_ptr<packet> &pkt, bool key_pos) {
        if (_policy.isFlushAble(is_video, key_pos, _policy.getStamp(pkt), _cache->size())) {
xiongziliang committed
242 243 244 245
            flushAll();
        }

        //追加数据到最后
246
        _cache->emplace_back(pkt);
xiongziliang committed
247 248 249 250 251
        if (key_pos) {
            _key_pos = key_pos;
        }
    }

252 253 254 255
    virtual void clearCache() {
        _cache->clear();
    }

256
    virtual void onFlush(std::shared_ptr<packet_list> &, bool key_pos) = 0;
xiongziliang committed
257 258 259 260 261 262

private:
    void flushAll() {
        if (_cache->empty()) {
            return;
        }
263
        onFlush(_cache, _key_pos);
xiongziliang committed
264 265 266 267 268
        _cache = std::make_shared<packet_list>();
        _key_pos = false;
    }

private:
xiongziliang committed
269
    bool _key_pos = false;
xiongziliang committed
270 271 272 273
    policy _policy;
    std::shared_ptr<packet_list> _cache;
};

xiongziliang committed
274
} /* namespace mediakit */
275
#endif //ZLMEDIAKIT_MEDIASOURCE_H