2017-10-09 22:11:01 +08:00
|
|
|
|
/*
|
2020-04-04 20:30:09 +08:00
|
|
|
|
* Copyright (c) 2016 The ZLMediaKit project authors. All Rights Reserved.
|
2017-09-27 16:20:30 +08:00
|
|
|
|
*
|
2021-01-17 18:31:50 +08:00
|
|
|
|
* This file is part of ZLMediaKit(https://github.com/xia-chu/ZLMediaKit).
|
2017-09-27 16:20:30 +08:00
|
|
|
|
*
|
2020-04-04 20:30:09 +08:00
|
|
|
|
* 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.
|
2017-04-01 16:35:56 +08:00
|
|
|
|
*/
|
|
|
|
|
|
2017-04-25 11:35:41 +08:00
|
|
|
|
#include <algorithm>
|
2017-04-01 16:35:56 +08:00
|
|
|
|
#include "PlayerBase.h"
|
2017-04-25 11:35:41 +08:00
|
|
|
|
#include "Rtsp/RtspPlayerImp.h"
|
|
|
|
|
#include "Rtmp/RtmpPlayerImp.h"
|
2020-05-17 18:00:37 +08:00
|
|
|
|
#include "Http/HlsPlayer.h"
|
2021-11-10 10:58:43 +08:00
|
|
|
|
|
2018-10-24 17:17:55 +08:00
|
|
|
|
using namespace toolkit;
|
2017-04-01 16:35:56 +08:00
|
|
|
|
|
2018-10-24 17:17:55 +08:00
|
|
|
|
namespace mediakit {
|
2017-04-01 16:35:56 +08:00
|
|
|
|
|
2021-11-10 10:58:43 +08:00
|
|
|
|
PlayerBase::Ptr PlayerBase::createPlayer(const EventPoller::Ptr &poller, const string &url_in) {
|
|
|
|
|
static auto releasePlayer = [](PlayerBase *ptr) {
|
|
|
|
|
onceToken token(nullptr, [&]() {
|
|
|
|
|
delete ptr;
|
2020-03-20 11:51:24 +08:00
|
|
|
|
});
|
|
|
|
|
ptr->teardown();
|
|
|
|
|
};
|
2020-05-21 14:10:27 +08:00
|
|
|
|
string url = url_in;
|
|
|
|
|
string prefix = FindField(url.data(), NULL, "://");
|
|
|
|
|
auto pos = url.find('?');
|
|
|
|
|
if (pos != string::npos) {
|
|
|
|
|
//去除?后面的字符串
|
|
|
|
|
url = url.substr(0, pos);
|
|
|
|
|
}
|
2019-07-20 20:53:50 +08:00
|
|
|
|
|
2021-11-10 10:58:43 +08:00
|
|
|
|
if (strcasecmp("rtsps", prefix.data()) == 0) {
|
|
|
|
|
return PlayerBase::Ptr(new TcpClientWithSSL<RtspPlayerImp>(poller), releasePlayer);
|
2020-03-20 11:51:24 +08:00
|
|
|
|
}
|
2019-07-20 20:53:50 +08:00
|
|
|
|
|
2021-11-10 10:58:43 +08:00
|
|
|
|
if (strcasecmp("rtsp", prefix.data()) == 0) {
|
|
|
|
|
return PlayerBase::Ptr(new RtspPlayerImp(poller), releasePlayer);
|
2020-03-20 11:51:24 +08:00
|
|
|
|
}
|
2019-07-20 20:53:50 +08:00
|
|
|
|
|
2021-11-10 10:58:43 +08:00
|
|
|
|
if (strcasecmp("rtmps", prefix.data()) == 0) {
|
|
|
|
|
return PlayerBase::Ptr(new TcpClientWithSSL<RtmpPlayerImp>(poller), releasePlayer);
|
2020-03-20 11:51:24 +08:00
|
|
|
|
}
|
2019-07-20 20:53:50 +08:00
|
|
|
|
|
2021-11-10 10:58:43 +08:00
|
|
|
|
if (strcasecmp("rtmp", prefix.data()) == 0) {
|
|
|
|
|
return PlayerBase::Ptr(new RtmpPlayerImp(poller), releasePlayer);
|
2020-03-20 11:51:24 +08:00
|
|
|
|
}
|
2019-07-20 20:53:50 +08:00
|
|
|
|
|
2021-11-10 10:58:43 +08:00
|
|
|
|
if ((strcasecmp("http", prefix.data()) == 0 || strcasecmp("https", prefix.data()) == 0) && end_with(url, ".m3u8")) {
|
|
|
|
|
return PlayerBase::Ptr(new HlsPlayerImp(poller), releasePlayer);
|
2020-05-17 18:00:37 +08:00
|
|
|
|
}
|
|
|
|
|
|
2021-11-10 10:58:43 +08:00
|
|
|
|
return PlayerBase::Ptr(new RtspPlayerImp(poller), releasePlayer);
|
2017-04-01 16:35:56 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-03-01 14:23:28 +08:00
|
|
|
|
PlayerBase::PlayerBase() {
|
2020-03-20 11:51:24 +08:00
|
|
|
|
this->mINI::operator[](kTimeoutMS) = 10000;
|
|
|
|
|
this->mINI::operator[](kMediaTimeoutMS) = 5000;
|
|
|
|
|
this->mINI::operator[](kBeatIntervalMS) = 5000;
|
2021-11-10 13:56:04 +08:00
|
|
|
|
this->mINI::operator[](kWaitTrackReady) = true;
|
2019-03-01 14:23:28 +08:00
|
|
|
|
}
|
|
|
|
|
|
2021-11-10 13:44:29 +08:00
|
|
|
|
///////////////////////////DemuxerSink//////////////////////////////
|
|
|
|
|
|
|
|
|
|
class DemuxerSink : public MediaSink {
|
|
|
|
|
public:
|
|
|
|
|
DemuxerSink() = default;
|
|
|
|
|
~DemuxerSink() override = default;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 设置track监听器
|
|
|
|
|
*/
|
|
|
|
|
void setTrackListener(TrackListener *listener);
|
|
|
|
|
vector<Track::Ptr> getTracks(bool ready = true) const override;
|
|
|
|
|
|
|
|
|
|
protected:
|
|
|
|
|
bool addTrack(const Track::Ptr & track) override;
|
|
|
|
|
void resetTracks() override;
|
|
|
|
|
bool onTrackReady(const Track::Ptr & track) override;
|
|
|
|
|
void onAllTrackReady() override;
|
|
|
|
|
bool onTrackFrame(const Frame::Ptr &frame) override;
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
Track::Ptr _tracks[TrackMax];
|
|
|
|
|
TrackListener *_listener = nullptr;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
bool DemuxerSink::addTrack(const Track::Ptr &track) {
|
2021-11-10 10:58:43 +08:00
|
|
|
|
auto ret = MediaSink::addTrack(track);
|
|
|
|
|
if (ret) {
|
|
|
|
|
track->addDelegate(std::make_shared<FrameWriterInterfaceHelper>([this](const Frame::Ptr &frame) {
|
|
|
|
|
return inputFrame(frame);
|
|
|
|
|
}));
|
2020-03-20 11:51:24 +08:00
|
|
|
|
}
|
2020-09-21 14:32:56 +08:00
|
|
|
|
return ret;
|
2018-11-15 15:14:05 +08:00
|
|
|
|
}
|
|
|
|
|
|
2021-11-10 13:44:29 +08:00
|
|
|
|
void DemuxerSink::setTrackListener(TrackListener *listener) {
|
2021-11-10 10:58:43 +08:00
|
|
|
|
_listener = listener;
|
2018-11-15 15:14:05 +08:00
|
|
|
|
}
|
2019-12-26 11:53:19 +08:00
|
|
|
|
|
2021-11-10 13:44:29 +08:00
|
|
|
|
bool DemuxerSink::onTrackReady(const Track::Ptr &track) {
|
2021-11-10 10:58:43 +08:00
|
|
|
|
_tracks[track->getTrackType()] = track->clone();
|
|
|
|
|
return true;
|
2019-12-26 11:53:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
2021-11-10 13:44:29 +08:00
|
|
|
|
void DemuxerSink::onAllTrackReady() {
|
2021-11-10 10:58:43 +08:00
|
|
|
|
if (!_listener) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
for (auto &track : _tracks) {
|
|
|
|
|
if (track) {
|
|
|
|
|
_listener->addTrack(track);
|
|
|
|
|
}
|
2021-04-20 17:53:43 +08:00
|
|
|
|
}
|
2021-11-10 10:58:43 +08:00
|
|
|
|
_listener->addTrackCompleted();
|
|
|
|
|
}
|
|
|
|
|
|
2021-11-10 13:44:29 +08:00
|
|
|
|
bool DemuxerSink::onTrackFrame(const Frame::Ptr &frame) {
|
2021-11-10 10:58:43 +08:00
|
|
|
|
return _tracks[frame->getTrackType()]->inputFrame(frame);
|
2021-04-20 17:53:43 +08:00
|
|
|
|
}
|
|
|
|
|
|
2021-11-10 13:44:29 +08:00
|
|
|
|
void DemuxerSink::resetTracks() {
|
2021-11-10 10:58:43 +08:00
|
|
|
|
MediaSink::resetTracks();
|
|
|
|
|
for (auto &track : _tracks) {
|
|
|
|
|
track = nullptr;
|
|
|
|
|
}
|
2021-04-20 17:53:43 +08:00
|
|
|
|
if (_listener) {
|
|
|
|
|
_listener->resetTracks();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-11-10 13:44:29 +08:00
|
|
|
|
vector<Track::Ptr> DemuxerSink::getTracks(bool ready) const {
|
2021-11-10 10:58:43 +08:00
|
|
|
|
vector<Track::Ptr> ret;
|
|
|
|
|
for (auto &track : _tracks) {
|
|
|
|
|
if (!track || (ready && !track->ready())) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
ret.emplace_back(track);
|
|
|
|
|
}
|
|
|
|
|
return ret;
|
2019-12-26 11:53:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
2021-11-10 13:44:29 +08:00
|
|
|
|
///////////////////////////Demuxer//////////////////////////////
|
|
|
|
|
|
|
|
|
|
void Demuxer::setTrackListener(TrackListener *listener, bool wait_track_ready) {
|
|
|
|
|
if (wait_track_ready) {
|
|
|
|
|
auto sink = std::make_shared<DemuxerSink>();
|
|
|
|
|
sink->setTrackListener(listener);
|
|
|
|
|
_sink = std::move(sink);
|
|
|
|
|
}
|
|
|
|
|
_listener = listener;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool Demuxer::addTrack(const Track::Ptr &track) {
|
|
|
|
|
if (!_sink) {
|
|
|
|
|
_origin_track.emplace_back(track);
|
|
|
|
|
}
|
|
|
|
|
return _sink ? _sink->addTrack(track) : (_listener ? _listener->addTrack(track) : false);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Demuxer::addTrackCompleted() {
|
|
|
|
|
if (_sink) {
|
|
|
|
|
_sink->addTrackCompleted();
|
|
|
|
|
} else if (_listener) {
|
|
|
|
|
_listener->addTrackCompleted();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Demuxer::resetTracks() {
|
|
|
|
|
if (_sink) {
|
|
|
|
|
_sink->resetTracks();
|
|
|
|
|
} else if (_listener) {
|
|
|
|
|
_listener->resetTracks();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
vector<Track::Ptr> Demuxer::getTracks(bool ready) const {
|
|
|
|
|
if (_sink) {
|
|
|
|
|
return _sink->getTracks(ready);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
vector<Track::Ptr> ret;
|
|
|
|
|
for (auto &track : _origin_track) {
|
|
|
|
|
if (ready && !track->ready()) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
ret.emplace_back(track);
|
|
|
|
|
}
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-24 17:17:55 +08:00
|
|
|
|
} /* namespace mediakit */
|