/* * SRT - Secure, Reliable, Transport * Copyright (c) 2018 Haivision Systems Inc. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * */ /***************************************************************************** Copyright (c) 2001 - 2009, The Board of Trustees of the University of Illinois. All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: * Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. * Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. * Neither the name of the University of Illinois nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. *****************************************************************************/ /***************************************************************************** written by Yunhong Gu, last updated 05/05/2009 modified by Haivision Systems Inc. *****************************************************************************/ #ifndef INC_SRT_BUFFER_TOOLS_H #define INC_SRT_BUFFER_TOOLS_H #include "common.h" namespace srt { /// The AvgBufSize class is used to calculate moving average of the buffer (RCV or SND) class AvgBufSize { typedef sync::steady_clock::time_point time_point; public: AvgBufSize() : m_dBytesCountMAvg(0.0) , m_dCountMAvg(0.0) , m_dTimespanMAvg(0.0) { } public: bool isTimeToUpdate(const time_point& now) const; void update(const time_point& now, int pkts, int bytes, int timespan_ms); public: inline double pkts() const { return m_dCountMAvg; } inline double timespan_ms() const { return m_dTimespanMAvg; } inline double bytes() const { return m_dBytesCountMAvg; } private: time_point m_tsLastSamplingTime; double m_dBytesCountMAvg; double m_dCountMAvg; double m_dTimespanMAvg; }; /// The class to estimate source bitrate based on samples submitted to the buffer. /// Is currently only used by the CSndBuffer. class CRateEstimator { typedef sync::steady_clock::time_point time_point; typedef sync::steady_clock::duration duration; public: CRateEstimator(); public: uint64_t getInRatePeriod() const { return m_InRatePeriod; } /// Retrieve input bitrate in bytes per second int getInputRate() const { return m_iInRateBps; } void setInputRateSmpPeriod(int period); /// Update input rate calculation. /// @param [in] time current time /// @param [in] pkts number of packets newly added to the buffer /// @param [in] bytes number of payload bytes in those newly added packets void updateInputRate(const time_point& time, int pkts = 0, int bytes = 0); void resetInputRateSmpPeriod(bool disable = false) { setInputRateSmpPeriod(disable ? 0 : INPUTRATE_FAST_START_US); } private: // Constants static const uint64_t INPUTRATE_FAST_START_US = 500000; // 500 ms static const uint64_t INPUTRATE_RUNNING_US = 1000000; // 1000 ms static const int64_t INPUTRATE_MAX_PACKETS = 2000; // ~ 21 Mbps of 1316 bytes payload static const int INPUTRATE_INITIAL_BYTESPS = BW_INFINITE; private: int m_iInRatePktsCount; // number of payload packets added since InRateStartTime. int m_iInRateBytesCount; // number of payload bytes added since InRateStartTime. time_point m_tsInRateStartTime; uint64_t m_InRatePeriod; // usec int m_iInRateBps; // Input Rate in Bytes/sec }; class CSndRateEstimator { typedef sync::steady_clock::time_point time_point; public: CSndRateEstimator(const time_point& tsNow); /// Add sample. /// @param [in] time sample (sending) time. /// @param [in] pkts number of packets in the sample. /// @param [in] bytes number of payload bytes in the sample. void addSample(const time_point& time, int pkts = 0, size_t bytes = 0); /// Retrieve estimated bitrate in bytes per second int getRate() const { return m_iRateBps; } /// Retrieve estimated bitrate in bytes per second inluding the current sampling interval. int getCurrentRate() const; private: static const int NUM_PERIODS = 10; static const int SAMPLE_DURATION_MS = 100; // 100 ms struct Sample { int m_iPktsCount; // number of payload packets int m_iBytesCount; // number of payload bytes void reset() { m_iPktsCount = 0; m_iBytesCount = 0; } Sample() : m_iPktsCount(0) , m_iBytesCount(0) { } Sample(int iPkts, int iBytes) : m_iPktsCount(iPkts) , m_iBytesCount(iBytes) { } Sample operator+(const Sample& other) { return Sample(m_iPktsCount + other.m_iPktsCount, m_iBytesCount + other.m_iBytesCount); } Sample& operator+=(const Sample& other) { *this = *this + other; return *this; } bool empty() const { return m_iPktsCount == 0; } }; int incSampleIdx(int val, int inc = 1) const; Sample m_Samples[NUM_PERIODS]; time_point m_tsFirstSampleTime; //< Start time of the first sameple. int m_iFirstSampleIdx; //< Index of the first sample. int m_iCurSampleIdx; //< Index of the current sample being collected. int m_iRateBps; // Input Rate in Bytes/sec }; } // namespace srt #endif