/* * Copyright (c) 2016 The ZLMediaKit project authors. All Rights Reserved. * * This file is part of ZLMediaKit(https://github.com/xia-chu/ZLMediaKit). * * 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. */ #ifndef ZLMEDIAKIT_STAMP_H #define ZLMEDIAKIT_STAMP_H #include #include #include "Util/TimeTicker.h" namespace mediakit { class DeltaStamp{ public: DeltaStamp(); ~DeltaStamp() = default; /** * 计算时间戳增量 * @param stamp 绝对时间戳 * @param enable_rollback 是否允许相当时间戳回退 * @return 时间戳增量 */ int64_t deltaStamp(int64_t stamp, bool enable_rollback = true); int64_t relativeStamp(int64_t stamp, bool enable_rollback = true); int64_t relativeStamp(); // 设置最大允许回退或跳跃幅度 void setMaxDelta(size_t max_delta); protected: virtual void needSync() {} protected: int _max_delta; int64_t _last_stamp = 0; int64_t _relative_stamp = 0; }; //该类解决时间戳回环、回退问题 //计算相对时间戳或者产生平滑时间戳 class Stamp : public DeltaStamp{ public: Stamp() = default; ~Stamp() = default; /** * 求取相对时间戳,同时实现了音视频同步、限制dts回退等功能 * @param dts 输入dts,如果为0则根据系统时间戳生成 * @param pts 输入pts,如果为0则等于dts * @param dts_out 输出dts * @param pts_out 输出pts * @param modifyStamp 是否用系统时间戳覆盖 */ void revise(int64_t dts, int64_t pts, int64_t &dts_out, int64_t &pts_out,bool modifyStamp = false); /** * 再设置相对时间戳,用于seek用 * @param relativeStamp 相对时间戳 */ void setRelativeStamp(int64_t relativeStamp); /** * 获取当前相对时间戳 * @return */ int64_t getRelativeStamp() const ; /** * 设置是否为回放模式,回放模式运行时间戳回退 * @param playback 是否为回放模式 */ void setPlayBack(bool playback = true); /** * 音视频同步用,音频应该同步于视频(只修改音频时间戳) * 因为音频时间戳修改后不影响播放速度 */ void syncTo(Stamp &other); /** * 是否允许时间戳回退 */ void enableRollback(bool flag); private: //主要实现音视频时间戳同步功能 void revise_l(int64_t dts, int64_t pts, int64_t &dts_out, int64_t &pts_out,bool modifyStamp = false); //主要实现获取相对时间戳功能 void revise_l2(int64_t dts, int64_t pts, int64_t &dts_out, int64_t &pts_out,bool modifyStamp = false); void needSync() override; private: bool _playback = false; bool _need_sync = false; // 默认不允许时间戳回滚 bool _enable_rollback = false; int64_t _relative_stamp = 0; int64_t _last_dts_in = 0; int64_t _last_dts_out = 0; int64_t _last_pts_out = 0; toolkit::SmoothTicker _ticker; Stamp *_sync_master = nullptr; }; //dts生成器, //pts排序后就是dts class DtsGenerator{ public: DtsGenerator() = default; ~DtsGenerator() = default; bool getDts(uint64_t pts, uint64_t &dts); private: bool getDts_l(uint64_t pts, uint64_t &dts); private: uint64_t _dts_pts_offset = 0; uint64_t _last_dts = 0; uint64_t _last_pts = 0; uint64_t _last_max_pts = 0; size_t _frames_since_last_max_pts = 0; size_t _sorter_max_size = 0; size_t _count_sorter_max_size = 0; std::set _pts_sorter; }; class NtpStamp { public: NtpStamp() = default; ~NtpStamp() = default; void setNtpStamp(uint32_t rtp_stamp, uint64_t ntp_stamp_ms); uint64_t getNtpStamp(uint32_t rtp_stamp, uint32_t sample_rate); private: void update(uint32_t rtp_stamp, uint64_t ntp_stamp_us); uint64_t getNtpStampUS(uint32_t rtp_stamp, uint32_t sample_rate); private: uint32_t _last_rtp_stamp = 0; uint64_t _last_ntp_stamp_us = 0; }; }//namespace mediakit #endif //ZLMEDIAKIT_STAMP_H