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
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#ifndef SRC_RTMP_RTMPPLAYERIMP_H_
|
|
|
|
|
#define SRC_RTMP_RTMPPLAYERIMP_H_
|
|
|
|
|
|
|
|
|
|
#include <memory>
|
2017-04-25 11:35:41 +08:00
|
|
|
|
#include <functional>
|
2017-05-02 17:15:12 +08:00
|
|
|
|
#include "Common/config.h"
|
2017-04-01 16:35:56 +08:00
|
|
|
|
#include "RtmpPlayer.h"
|
2018-09-18 21:40:26 +08:00
|
|
|
|
#include "RtmpMediaSource.h"
|
2019-06-28 17:30:13 +08:00
|
|
|
|
#include "RtmpDemuxer.h"
|
2017-04-25 11:35:41 +08:00
|
|
|
|
#include "Poller/Timer.h"
|
2017-04-01 16:35:56 +08:00
|
|
|
|
#include "Util/TimeTicker.h"
|
2018-10-24 17:17:55 +08:00
|
|
|
|
using namespace toolkit;
|
2019-03-27 18:56:49 +08:00
|
|
|
|
using namespace mediakit::Client;
|
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
|
|
|
|
class RtmpPlayerImp: public PlayerImp<RtmpPlayer,PlayerBase>, private TrackListener {
|
2017-04-01 16:35:56 +08:00
|
|
|
|
public:
|
2021-11-10 10:58:43 +08:00
|
|
|
|
using Ptr = std::shared_ptr<RtmpPlayerImp>;
|
|
|
|
|
using Super = PlayerImp<RtmpPlayer,PlayerBase>;
|
2020-08-30 10:48:34 +08:00
|
|
|
|
|
2021-11-10 10:58:43 +08:00
|
|
|
|
RtmpPlayerImp(const EventPoller::Ptr &poller) : Super(poller) {};
|
2020-08-30 10:48:34 +08:00
|
|
|
|
|
|
|
|
|
~RtmpPlayerImp() override {
|
|
|
|
|
DebugL << endl;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
float getProgress() const override {
|
|
|
|
|
if (getDuration() > 0) {
|
2018-10-26 14:12:16 +08:00
|
|
|
|
return getProgressMilliSecond() / (getDuration() * 1000);
|
2017-04-01 16:35:56 +08:00
|
|
|
|
}
|
2017-08-14 22:20:53 +08:00
|
|
|
|
return PlayerBase::getProgress();
|
2020-08-30 10:48:34 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void seekTo(float fProgress) override {
|
|
|
|
|
fProgress = MAX(float(0), MIN(fProgress, float(1.0)));
|
2021-01-17 18:31:50 +08:00
|
|
|
|
seekToMilliSecond((uint32_t)(fProgress * getDuration() * 1000));
|
2020-08-30 10:48:34 +08:00
|
|
|
|
}
|
|
|
|
|
|
2021-08-09 18:28:43 +08:00
|
|
|
|
void seekTo(uint32_t seekPos) override {
|
2021-08-12 16:07:31 +08:00
|
|
|
|
uint32_t pos = MAX(float(0), MIN(seekPos, getDuration())) * 1000;
|
2021-08-09 18:28:43 +08:00
|
|
|
|
seekToMilliSecond(pos);
|
|
|
|
|
}
|
2021-11-10 10:58:43 +08:00
|
|
|
|
|
|
|
|
|
float getDuration() const override {
|
|
|
|
|
return _demuxer ? _demuxer->getDuration() : 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
vector<Track::Ptr> getTracks(bool ready = true) const override {
|
|
|
|
|
return _demuxer ? _demuxer->getTracks(ready) : Super::getTracks(ready);
|
2019-03-01 14:23:28 +08:00
|
|
|
|
}
|
2020-08-30 10:48:34 +08:00
|
|
|
|
|
2017-04-01 16:35:56 +08:00
|
|
|
|
private:
|
|
|
|
|
//派生类回调函数
|
2019-09-24 11:42:27 +08:00
|
|
|
|
bool onCheckMeta(const AMFValue &val) override {
|
2021-11-10 15:25:24 +08:00
|
|
|
|
//无metadata或metadata中无track信息时,需要从数据包中获取track
|
|
|
|
|
_wait_track_ready = (*this)[Client::kWaitTrackReady].as<bool>() || RtmpDemuxer::trackCount(val) == 0;
|
2021-11-10 10:58:43 +08:00
|
|
|
|
onCheckMeta_l(val);
|
2018-10-25 17:39:19 +08:00
|
|
|
|
return true;
|
2017-04-01 16:35:56 +08:00
|
|
|
|
}
|
2020-08-30 10:48:34 +08:00
|
|
|
|
|
2021-02-04 17:58:51 +08:00
|
|
|
|
void onMediaData(RtmpPacket::Ptr chunkData) override {
|
2021-11-10 10:58:43 +08:00
|
|
|
|
if (!_demuxer) {
|
|
|
|
|
//有些rtmp流没metadata
|
|
|
|
|
onCheckMeta_l(TitleMeta().getMetadata());
|
2021-02-04 17:58:51 +08:00
|
|
|
|
}
|
2021-11-10 10:58:43 +08:00
|
|
|
|
_demuxer->inputRtmp(chunkData);
|
2020-08-30 10:48:34 +08:00
|
|
|
|
if (_rtmp_src) {
|
2021-02-04 17:58:51 +08:00
|
|
|
|
_rtmp_src->onWrite(std::move(chunkData));
|
2018-10-25 17:39:19 +08:00
|
|
|
|
}
|
2017-04-01 16:35:56 +08:00
|
|
|
|
}
|
2020-08-30 10:48:34 +08:00
|
|
|
|
|
2021-11-10 10:58:43 +08:00
|
|
|
|
void onPlayResult(const SockException &ex) override {
|
2021-11-10 15:25:24 +08:00
|
|
|
|
if (!_wait_track_ready || ex) {
|
2021-11-10 10:58:43 +08:00
|
|
|
|
Super::onPlayResult(ex);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool addTrack(const Track::Ptr &track) override { return true; }
|
|
|
|
|
|
|
|
|
|
void addTrackCompleted() override {
|
2021-11-10 15:25:24 +08:00
|
|
|
|
if (_wait_track_ready) {
|
2021-11-10 13:56:04 +08:00
|
|
|
|
Super::onPlayResult(SockException(Err_success, "play success"));
|
|
|
|
|
}
|
2021-11-10 10:58:43 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
void onCheckMeta_l(const AMFValue &val) {
|
|
|
|
|
_rtmp_src = dynamic_pointer_cast<RtmpMediaSource>(_media_src);
|
|
|
|
|
if (_rtmp_src) {
|
|
|
|
|
_rtmp_src->setMetaData(val);
|
|
|
|
|
}
|
|
|
|
|
_demuxer = std::make_shared<RtmpDemuxer>();
|
2021-11-10 15:25:24 +08:00
|
|
|
|
_demuxer->setTrackListener(this, _wait_track_ready);
|
2021-11-10 10:58:43 +08:00
|
|
|
|
_demuxer->loadMetaData(val);
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-18 21:40:26 +08:00
|
|
|
|
private:
|
2021-11-10 15:25:24 +08:00
|
|
|
|
bool _wait_track_ready = true;
|
2021-11-10 10:58:43 +08:00
|
|
|
|
RtmpDemuxer::Ptr _demuxer;
|
2020-08-30 10:48:34 +08:00
|
|
|
|
RtmpMediaSource::Ptr _rtmp_src;
|
2017-04-01 16:35:56 +08:00
|
|
|
|
};
|
2019-03-01 14:23:28 +08:00
|
|
|
|
|
|
|
|
|
|
2018-10-24 17:17:55 +08:00
|
|
|
|
} /* namespace mediakit */
|
|
|
|
|
|
2017-04-01 16:35:56 +08:00
|
|
|
|
#endif /* SRC_RTMP_RTMPPLAYERIMP_H_ */
|