2020-05-18 15:31:49 +08:00
|
|
|
|
/*
|
2020-05-17 18:00:37 +08:00
|
|
|
|
* Copyright (c) 2020 The ZLMediaKit project authors. All Rights Reserved.
|
|
|
|
|
*
|
2021-01-17 18:31:50 +08:00
|
|
|
|
* This file is part of ZLMediaKit(https://github.com/xia-chu/ZLMediaKit).
|
2020-05-17 18:00:37 +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.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include "HlsPlayer.h"
|
2021-12-24 13:29:16 +08:00
|
|
|
|
|
2020-05-17 18:00:37 +08:00
|
|
|
|
namespace mediakit {
|
|
|
|
|
|
2021-12-24 13:29:16 +08:00
|
|
|
|
HlsPlayer::HlsPlayer(const EventPoller::Ptr &poller) {
|
2021-01-17 18:31:50 +08:00
|
|
|
|
_segment.setOnSegment([this](const char *data, size_t len) { onPacket(data, len); });
|
2020-09-12 19:03:52 +08:00
|
|
|
|
setPoller(poller ? poller : EventPollerPool::Instance().getPoller());
|
2020-05-17 18:00:37 +08:00
|
|
|
|
}
|
|
|
|
|
|
2022-01-20 14:48:45 +08:00
|
|
|
|
void HlsPlayer::play(const string &url) {
|
|
|
|
|
_play_result = false;
|
|
|
|
|
_play_url = url;
|
|
|
|
|
fetchIndexFile();
|
2020-05-17 18:00:37 +08:00
|
|
|
|
}
|
|
|
|
|
|
2022-01-20 14:48:45 +08:00
|
|
|
|
void HlsPlayer::fetchIndexFile() {
|
2021-12-24 13:29:16 +08:00
|
|
|
|
if (waitResponse()) {
|
2021-12-22 11:33:40 +08:00
|
|
|
|
return;
|
|
|
|
|
}
|
2021-12-24 13:29:16 +08:00
|
|
|
|
if (!(*this)[kNetAdapter].empty()) {
|
2020-05-17 18:00:37 +08:00
|
|
|
|
setNetAdapter((*this)[kNetAdapter]);
|
|
|
|
|
}
|
2022-01-19 22:50:44 +08:00
|
|
|
|
setCompleteTimeout((*this)[Client::kTimeoutMS].as<int>());
|
2022-01-20 14:48:45 +08:00
|
|
|
|
setMethod("GET");
|
|
|
|
|
sendRequest(_play_url);
|
2020-05-17 18:00:37 +08:00
|
|
|
|
}
|
|
|
|
|
|
2021-12-24 13:29:16 +08:00
|
|
|
|
void HlsPlayer::teardown_l(const SockException &ex) {
|
2022-01-20 14:48:45 +08:00
|
|
|
|
if (!_play_result) {
|
|
|
|
|
_play_result = true;
|
|
|
|
|
onPlayResult(ex);
|
|
|
|
|
} else {
|
|
|
|
|
onShutdown(ex);
|
|
|
|
|
}
|
2020-05-17 18:00:37 +08:00
|
|
|
|
_timer.reset();
|
|
|
|
|
_timer_ts.reset();
|
|
|
|
|
_http_ts_player.reset();
|
|
|
|
|
shutdown(ex);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void HlsPlayer::teardown() {
|
2021-12-24 13:29:16 +08:00
|
|
|
|
teardown_l(SockException(Err_shutdown, "teardown"));
|
2020-05-17 18:00:37 +08:00
|
|
|
|
}
|
|
|
|
|
|
2022-01-20 14:48:45 +08:00
|
|
|
|
void HlsPlayer::fetchSegment() {
|
2020-05-17 18:00:37 +08:00
|
|
|
|
if (_ts_list.empty()) {
|
|
|
|
|
//播放列表为空,那么立即重新下载m3u8文件
|
|
|
|
|
_timer.reset();
|
2022-01-20 14:48:45 +08:00
|
|
|
|
fetchIndexFile();
|
2020-05-17 18:00:37 +08:00
|
|
|
|
return;
|
|
|
|
|
}
|
2021-12-24 13:29:16 +08:00
|
|
|
|
if (_http_ts_player && _http_ts_player->waitResponse()) {
|
2020-05-17 18:00:37 +08:00
|
|
|
|
//播放器目前还存活,正在下载中
|
|
|
|
|
return;
|
|
|
|
|
}
|
2021-12-24 13:29:16 +08:00
|
|
|
|
weak_ptr<HlsPlayer> weak_self = dynamic_pointer_cast<HlsPlayer>(shared_from_this());
|
|
|
|
|
if (!_http_ts_player) {
|
|
|
|
|
_http_ts_player = std::make_shared<HttpTSPlayer>(getPoller(), false);
|
|
|
|
|
_http_ts_player->setOnCreateSocket([weak_self](const EventPoller::Ptr &poller) {
|
|
|
|
|
auto strong_self = weak_self.lock();
|
|
|
|
|
if (strong_self) {
|
|
|
|
|
return strong_self->createSocket();
|
|
|
|
|
}
|
|
|
|
|
return Socket::createSocket(poller, true);
|
|
|
|
|
});
|
2020-05-17 18:00:37 +08:00
|
|
|
|
|
2021-12-24 13:29:16 +08:00
|
|
|
|
_http_ts_player->setOnPacket([weak_self](const char *data, size_t len) {
|
|
|
|
|
auto strong_self = weak_self.lock();
|
|
|
|
|
if (!strong_self) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
//收到ts包
|
|
|
|
|
strong_self->onPacket_l(data, len);
|
|
|
|
|
});
|
2020-09-12 19:03:52 +08:00
|
|
|
|
|
2021-12-24 13:29:16 +08:00
|
|
|
|
if (!(*this)[kNetAdapter].empty()) {
|
|
|
|
|
_http_ts_player->setNetAdapter((*this)[Client::kNetAdapter]);
|
2020-09-12 19:03:52 +08:00
|
|
|
|
}
|
2021-12-24 13:29:16 +08:00
|
|
|
|
}
|
2020-09-12 19:03:52 +08:00
|
|
|
|
|
2021-12-24 13:29:16 +08:00
|
|
|
|
Ticker ticker;
|
|
|
|
|
auto url = _ts_list.front().url;
|
|
|
|
|
auto duration = _ts_list.front().duration;
|
|
|
|
|
_ts_list.pop_front();
|
|
|
|
|
|
|
|
|
|
_http_ts_player->setOnComplete([weak_self, ticker, duration, url](const SockException &err) {
|
|
|
|
|
auto strong_self = weak_self.lock();
|
|
|
|
|
if (!strong_self) {
|
2020-05-17 18:00:37 +08:00
|
|
|
|
return;
|
|
|
|
|
}
|
2021-12-24 13:29:16 +08:00
|
|
|
|
if (err) {
|
|
|
|
|
WarnL << "download ts segment " << url << " failed:" << err.what();
|
2020-05-17 18:00:37 +08:00
|
|
|
|
}
|
2021-12-24 13:29:16 +08:00
|
|
|
|
//提前半秒下载好
|
|
|
|
|
auto delay = duration - ticker.elapsedTime() / 1000.0f - 0.5;
|
|
|
|
|
if (delay <= 0) {
|
|
|
|
|
//延时最小10ms
|
|
|
|
|
delay = 10;
|
2020-05-17 18:00:37 +08:00
|
|
|
|
}
|
2021-12-24 13:29:16 +08:00
|
|
|
|
//延时下载下一个切片
|
|
|
|
|
strong_self->_timer_ts.reset(new Timer(delay / 1000.0f, [weak_self]() {
|
|
|
|
|
auto strong_self = weak_self.lock();
|
|
|
|
|
if (strong_self) {
|
2022-01-20 14:48:45 +08:00
|
|
|
|
strong_self->fetchSegment();
|
2021-12-24 13:29:16 +08:00
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}, strong_self->getPoller()));
|
2020-05-17 18:00:37 +08:00
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
_http_ts_player->setMethod("GET");
|
2022-01-19 22:50:44 +08:00
|
|
|
|
//ts切片必须在其时长的3倍内下载完毕
|
|
|
|
|
_http_ts_player->setCompleteTimeout(3 * duration * 1000);
|
|
|
|
|
_http_ts_player->sendRequest(url);
|
2020-05-17 18:00:37 +08:00
|
|
|
|
}
|
|
|
|
|
|
2021-12-24 13:29:16 +08:00
|
|
|
|
void HlsPlayer::onParsed(bool is_m3u8_inner, int64_t sequence, const map<int, ts_segment> &ts_map) {
|
2020-05-17 18:00:37 +08:00
|
|
|
|
if (!is_m3u8_inner) {
|
|
|
|
|
//这是ts播放列表
|
|
|
|
|
if (_last_sequence == sequence) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
_last_sequence = sequence;
|
|
|
|
|
for (auto &pr : ts_map) {
|
|
|
|
|
auto &ts = pr.second;
|
|
|
|
|
if (_ts_url_cache.emplace(ts.url).second) {
|
|
|
|
|
//该ts未重复
|
|
|
|
|
_ts_list.emplace_back(ts);
|
|
|
|
|
//按时间排序
|
|
|
|
|
_ts_url_sort.emplace_back(ts.url);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (_ts_url_sort.size() > 2 * ts_map.size()) {
|
|
|
|
|
//去除防重列表中过多的数据
|
|
|
|
|
_ts_url_cache.erase(_ts_url_sort.front());
|
|
|
|
|
_ts_url_sort.pop_front();
|
|
|
|
|
}
|
2022-01-20 14:48:45 +08:00
|
|
|
|
fetchSegment();
|
2020-05-17 18:00:37 +08:00
|
|
|
|
} else {
|
|
|
|
|
//这是m3u8列表,我们播放最高清的子hls
|
|
|
|
|
if (ts_map.empty()) {
|
2022-01-20 14:48:45 +08:00
|
|
|
|
throw invalid_argument("empty sub hls list:" + getUrl());
|
2020-05-17 18:00:37 +08:00
|
|
|
|
}
|
|
|
|
|
_timer.reset();
|
2021-12-24 13:29:16 +08:00
|
|
|
|
weak_ptr<HlsPlayer> weak_self = dynamic_pointer_cast<HlsPlayer>(shared_from_this());
|
2020-05-17 18:00:37 +08:00
|
|
|
|
auto url = ts_map.rbegin()->second.url;
|
2021-12-24 13:29:16 +08:00
|
|
|
|
getPoller()->async([weak_self, url]() {
|
|
|
|
|
auto strong_self = weak_self.lock();
|
|
|
|
|
if (strong_self) {
|
|
|
|
|
strong_self->play(url);
|
2020-05-17 18:00:37 +08:00
|
|
|
|
}
|
|
|
|
|
}, false);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-01-20 14:48:45 +08:00
|
|
|
|
void HlsPlayer::onResponseHeader(const string &status, const HttpClient::HttpHeader &headers) {
|
2020-05-17 18:00:37 +08:00
|
|
|
|
if (status != "200" && status != "206") {
|
|
|
|
|
//失败
|
2022-01-20 14:48:45 +08:00
|
|
|
|
throw invalid_argument("bad http status code:" + status);
|
2020-05-17 18:00:37 +08:00
|
|
|
|
}
|
2022-01-21 16:10:27 +08:00
|
|
|
|
auto content_type = strToLower(const_cast<HttpClient::HttpHeader &>(headers)["Content-Type"]);
|
|
|
|
|
if (content_type.find("application/vnd.apple.mpegurl") != 0 && content_type.find("application/x-mpegurl") != 0) {
|
|
|
|
|
WarnL << "may not a hls video: " << content_type << ", url: " << getUrl();
|
2022-01-05 20:01:19 +08:00
|
|
|
|
}
|
2022-01-20 14:48:45 +08:00
|
|
|
|
_m3u8.clear();
|
2020-05-17 18:00:37 +08:00
|
|
|
|
}
|
|
|
|
|
|
2022-01-20 14:48:45 +08:00
|
|
|
|
void HlsPlayer::onResponseBody(const char *buf, size_t size) {
|
2020-05-17 18:00:37 +08:00
|
|
|
|
_m3u8.append(buf, size);
|
|
|
|
|
}
|
|
|
|
|
|
2022-01-20 14:48:45 +08:00
|
|
|
|
void HlsPlayer::onResponseCompleted(const SockException &ex) {
|
|
|
|
|
if (ex) {
|
|
|
|
|
teardown_l(ex);
|
|
|
|
|
return;
|
2020-05-17 18:00:37 +08:00
|
|
|
|
}
|
2022-01-20 14:48:45 +08:00
|
|
|
|
if (!HlsParser::parse(getUrl(), _m3u8)) {
|
|
|
|
|
teardown_l(SockException(Err_other, "parse m3u8 failed:" + _m3u8));
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (!_play_result) {
|
|
|
|
|
_play_result = true;
|
|
|
|
|
onPlayResult(SockException());
|
|
|
|
|
}
|
|
|
|
|
playDelay();
|
2020-05-17 18:00:37 +08:00
|
|
|
|
}
|
|
|
|
|
|
2021-12-17 02:04:28 +08:00
|
|
|
|
float HlsPlayer::delaySecond() {
|
2020-05-17 18:00:37 +08:00
|
|
|
|
if (HlsParser::isM3u8() && HlsParser::getTargetDur() > 0) {
|
2021-12-17 02:04:28 +08:00
|
|
|
|
float targetOffset;
|
|
|
|
|
if (HlsParser::isLive()) {
|
|
|
|
|
// see RFC 8216, Section 4.4.3.8.
|
|
|
|
|
// 根据rfc刷新index列表的周期应该是分段时间x3, 因为根据规范播放器只处理最后3个Segment
|
|
|
|
|
targetOffset = (float) (3 * HlsParser::getTargetDur());
|
|
|
|
|
} else {
|
|
|
|
|
// 点播则一般m3u8文件不会在改变了, 没必要频繁的刷新, 所以按照总时间来进行刷新
|
|
|
|
|
targetOffset = HlsParser::getTotalDuration();
|
|
|
|
|
}
|
|
|
|
|
// 取最小值, 避免因为分段时长不规则而导致的问题
|
|
|
|
|
if (targetOffset > HlsParser::getTotalDuration()) {
|
|
|
|
|
targetOffset = HlsParser::getTotalDuration();
|
|
|
|
|
}
|
|
|
|
|
// 根据规范为一半的时间
|
2021-12-17 11:40:54 +08:00
|
|
|
|
if (targetOffset / 2 > 1.0f) {
|
2021-12-17 02:04:28 +08:00
|
|
|
|
return targetOffset / 2;
|
|
|
|
|
}
|
2020-05-17 18:00:37 +08:00
|
|
|
|
}
|
2021-01-17 18:31:50 +08:00
|
|
|
|
return 1.0f;
|
2020-05-17 18:00:37 +08:00
|
|
|
|
}
|
|
|
|
|
|
2021-12-24 13:29:16 +08:00
|
|
|
|
bool HlsPlayer::onRedirectUrl(const string &url, bool temporary) {
|
2022-01-20 14:48:45 +08:00
|
|
|
|
_play_url = url;
|
2020-05-17 18:00:37 +08:00
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-24 13:29:16 +08:00
|
|
|
|
void HlsPlayer::playDelay() {
|
|
|
|
|
weak_ptr<HlsPlayer> weak_self = dynamic_pointer_cast<HlsPlayer>(shared_from_this());
|
|
|
|
|
_timer.reset(new Timer(delaySecond(), [weak_self]() {
|
|
|
|
|
auto strong_self = weak_self.lock();
|
|
|
|
|
if (strong_self) {
|
2022-01-20 14:48:45 +08:00
|
|
|
|
strong_self->fetchIndexFile();
|
2020-05-17 18:00:37 +08:00
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}, getPoller()));
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-24 13:29:16 +08:00
|
|
|
|
void HlsPlayer::onPacket_l(const char *data, size_t len) {
|
2021-12-24 19:58:27 +08:00
|
|
|
|
try {
|
|
|
|
|
_segment.input(data, len);
|
|
|
|
|
} catch (...) {
|
|
|
|
|
//ts解析失败,清空缓存数据
|
|
|
|
|
_segment.reset();
|
|
|
|
|
throw;
|
|
|
|
|
}
|
2020-05-17 18:00:37 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
2021-12-01 22:09:05 +08:00
|
|
|
|
void HlsDemuxer::start(const EventPoller::Ptr &poller, TrackListener *listener) {
|
|
|
|
|
_frame_cache.clear();
|
|
|
|
|
_stamp[TrackAudio].setRelativeStamp(0);
|
|
|
|
|
_stamp[TrackVideo].setRelativeStamp(0);
|
|
|
|
|
_stamp[TrackAudio].syncTo(_stamp[TrackVideo]);
|
|
|
|
|
setPlayPosition(0);
|
|
|
|
|
|
|
|
|
|
_delegate.setTrackListener(listener);
|
|
|
|
|
|
|
|
|
|
//每50毫秒执行一次
|
|
|
|
|
weak_ptr<HlsDemuxer> weak_self = shared_from_this();
|
|
|
|
|
_timer = std::make_shared<Timer>(0.05f, [weak_self]() {
|
|
|
|
|
auto strong_self = weak_self.lock();
|
|
|
|
|
if (!strong_self) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
strong_self->onTick();
|
|
|
|
|
return true;
|
|
|
|
|
}, poller);
|
2020-05-17 18:00:37 +08:00
|
|
|
|
}
|
|
|
|
|
|
2021-12-01 22:09:05 +08:00
|
|
|
|
bool HlsDemuxer::inputFrame(const Frame::Ptr &frame) {
|
2021-12-17 14:47:17 +08:00
|
|
|
|
//为了避免track准备时间过长, 因此在没准备好之前, 直接消费掉所有的帧
|
|
|
|
|
if (!_delegate.isAllTrackReady()) {
|
|
|
|
|
_delegate.inputFrame(frame);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
2021-12-24 13:29:16 +08:00
|
|
|
|
|
2020-05-26 23:19:13 +08:00
|
|
|
|
//计算相对时间戳
|
|
|
|
|
int64_t dts, pts;
|
|
|
|
|
_stamp[frame->getTrackType()].revise(frame->dts(), frame->pts(), dts, pts);
|
|
|
|
|
//根据时间戳缓存frame
|
|
|
|
|
_frame_cache.emplace(dts, Frame::getCacheAbleFrame(frame));
|
|
|
|
|
|
2020-09-06 17:56:45 +08:00
|
|
|
|
if (getBufferMS() > 30 * 1000) {
|
|
|
|
|
//缓存超过30秒,强制消费至15秒(减少延时或内存占用)
|
|
|
|
|
while (getBufferMS() > 15 * 1000) {
|
2021-12-01 22:09:05 +08:00
|
|
|
|
_delegate.inputFrame(_frame_cache.begin()->second);
|
2020-05-26 23:19:13 +08:00
|
|
|
|
_frame_cache.erase(_frame_cache.begin());
|
|
|
|
|
}
|
2020-09-06 17:56:45 +08:00
|
|
|
|
//接着播放缓存中最早的帧
|
|
|
|
|
setPlayPosition(_frame_cache.begin()->first);
|
|
|
|
|
}
|
2021-09-27 13:12:53 +08:00
|
|
|
|
return true;
|
2020-09-06 17:56:45 +08:00
|
|
|
|
}
|
|
|
|
|
|
2021-12-01 22:09:05 +08:00
|
|
|
|
int64_t HlsDemuxer::getPlayPosition() {
|
2020-09-06 17:56:45 +08:00
|
|
|
|
return _ticker.elapsedTime() + _ticker_offset;
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-01 22:09:05 +08:00
|
|
|
|
int64_t HlsDemuxer::getBufferMS() {
|
|
|
|
|
if (_frame_cache.empty()) {
|
2020-09-06 17:56:45 +08:00
|
|
|
|
return 0;
|
2020-05-26 23:19:13 +08:00
|
|
|
|
}
|
2020-09-06 17:56:45 +08:00
|
|
|
|
return _frame_cache.rbegin()->first - _frame_cache.begin()->first;
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-01 22:09:05 +08:00
|
|
|
|
void HlsDemuxer::setPlayPosition(int64_t pos) {
|
2020-09-06 17:56:45 +08:00
|
|
|
|
_ticker.resetTime();
|
|
|
|
|
_ticker_offset = pos;
|
2020-05-26 23:19:13 +08:00
|
|
|
|
}
|
|
|
|
|
|
2021-12-01 22:09:05 +08:00
|
|
|
|
void HlsDemuxer::onTick() {
|
2020-05-26 23:19:13 +08:00
|
|
|
|
auto it = _frame_cache.begin();
|
|
|
|
|
while (it != _frame_cache.end()) {
|
2020-09-06 17:56:45 +08:00
|
|
|
|
if (it->first > getPlayPosition()) {
|
2020-05-26 23:19:13 +08:00
|
|
|
|
//这些帧还未到时间播放
|
|
|
|
|
break;
|
|
|
|
|
}
|
2020-09-12 18:53:16 +08:00
|
|
|
|
|
|
|
|
|
if (getBufferMS() < 3 * 1000) {
|
|
|
|
|
//缓存小于3秒,那么降低定时器消费速度(让剩余的数据在3秒后消费完毕)
|
|
|
|
|
//目的是为了防止定时器长时间干等后,数据瞬间消费完毕
|
|
|
|
|
setPlayPosition(_frame_cache.begin()->first);
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-26 23:19:13 +08:00
|
|
|
|
//消费掉已经到期的帧
|
2021-12-01 22:09:05 +08:00
|
|
|
|
_delegate.inputFrame(it->second);
|
2020-05-26 23:19:13 +08:00
|
|
|
|
it = _frame_cache.erase(it);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-01 22:09:05 +08:00
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
|
HlsPlayerImp::HlsPlayerImp(const EventPoller::Ptr &poller) : PlayerImp<HlsPlayer, PlayerBase>(poller) {}
|
|
|
|
|
|
2021-12-24 13:29:16 +08:00
|
|
|
|
void HlsPlayerImp::onPacket(const char *data, size_t len) {
|
2022-01-20 14:48:45 +08:00
|
|
|
|
if (!_decoder && _demuxer) {
|
2021-12-01 22:09:05 +08:00
|
|
|
|
_decoder = DecoderImp::createDecoder(DecoderImp::decoder_ts, _demuxer.get());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (_decoder && _demuxer) {
|
|
|
|
|
_decoder->input((uint8_t *) data, len);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void HlsPlayerImp::addTrackCompleted() {
|
|
|
|
|
PlayerImp<HlsPlayer, PlayerBase>::onPlayResult(SockException(Err_success, "play hls success"));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void HlsPlayerImp::onPlayResult(const SockException &ex) {
|
|
|
|
|
if (ex) {
|
|
|
|
|
PlayerImp<HlsPlayer, PlayerBase>::onPlayResult(ex);
|
|
|
|
|
} else {
|
|
|
|
|
auto demuxer = std::make_shared<HlsDemuxer>();
|
|
|
|
|
demuxer->start(getPoller(), this);
|
|
|
|
|
_demuxer = std::move(demuxer);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void HlsPlayerImp::onShutdown(const SockException &ex) {
|
|
|
|
|
PlayerImp<HlsPlayer, PlayerBase>::onShutdown(ex);
|
|
|
|
|
_demuxer = nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
vector<Track::Ptr> HlsPlayerImp::getTracks(bool ready) const {
|
|
|
|
|
return static_pointer_cast<HlsDemuxer>(_demuxer)->getTracks(ready);
|
|
|
|
|
}
|
2020-05-17 18:00:37 +08:00
|
|
|
|
|
2021-12-17 02:04:28 +08:00
|
|
|
|
}//namespace mediakit
|