blob: 4bb988b65888bb934addaadfd0629f6196262e92 [file] [log] [blame]
Sebastian Jansson30bd4032018-04-13 13:56:17 +02001/*
2 * Copyright (c) 2018 The WebRTC project authors. All Rights Reserved.
3 *
4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 */
10
Sebastian Jansson6fae6ec2018-05-08 10:43:18 +020011#ifndef API_UNITS_DATA_RATE_H_
12#define API_UNITS_DATA_RATE_H_
Sebastian Jansson2afd2812018-08-23 14:44:05 +020013
14#ifdef UNIT_TEST
15#include <ostream> // no-presubmit-check TODO(webrtc:8982)
16#endif // UNIT_TEST
17
Sebastian Jansson30bd4032018-04-13 13:56:17 +020018#include <stdint.h>
Sebastian Jansson6c19dec2018-10-05 13:23:57 +020019#include <algorithm>
Sebastian Jansson30bd4032018-04-13 13:56:17 +020020#include <cmath>
21#include <limits>
22#include <string>
23
24#include "rtc_base/checks.h"
Sebastian Jansson942b3602018-05-30 15:47:44 +020025#include "rtc_base/numerics/safe_conversions.h"
Sebastian Jansson30bd4032018-04-13 13:56:17 +020026
Sebastian Jansson6fae6ec2018-05-08 10:43:18 +020027#include "api/units/data_size.h"
28#include "api/units/time_delta.h"
Sebastian Janssone31be152018-04-16 12:41:47 +020029
Sebastian Jansson30bd4032018-04-13 13:56:17 +020030namespace webrtc {
31namespace data_rate_impl {
32constexpr int64_t kPlusInfinityVal = std::numeric_limits<int64_t>::max();
Sebastian Jansson66fa5352018-04-30 16:54:57 +020033
34inline int64_t Microbits(const DataSize& size) {
35 constexpr int64_t kMaxBeforeConversion =
36 std::numeric_limits<int64_t>::max() / 8000000;
37 RTC_DCHECK_LE(size.bytes(), kMaxBeforeConversion)
38 << "size is too large to be expressed in microbytes";
39 return size.bytes() * 8000000;
40}
Sebastian Jansson30bd4032018-04-13 13:56:17 +020041} // namespace data_rate_impl
42
43// DataRate is a class that represents a given data rate. This can be used to
Sebastian Jansson66fa5352018-04-30 16:54:57 +020044// represent bandwidth, encoding bitrate, etc. The internal storage is bits per
45// second (bps).
Sebastian Jansson30bd4032018-04-13 13:56:17 +020046class DataRate {
47 public:
Sebastian Jansson3b69b192018-05-07 13:51:51 +020048 DataRate() = delete;
Sebastian Jansson8e064192018-08-07 12:34:33 +020049 static constexpr DataRate Zero() { return DataRate(0); }
50 static constexpr DataRate Infinity() {
Sebastian Jansson30bd4032018-04-13 13:56:17 +020051 return DataRate(data_rate_impl::kPlusInfinityVal);
52 }
Sebastian Janssonc1c8b8e2018-08-07 15:29:04 +020053 template <int64_t bps>
54 static constexpr DataRate BitsPerSec() {
55 static_assert(bps >= 0, "");
56 static_assert(bps < data_rate_impl::kPlusInfinityVal, "");
57 return DataRate(bps);
58 }
59 template <int64_t kbps>
60 static constexpr DataRate KilobitsPerSec() {
61 static_assert(kbps >= 0, "");
62 static_assert(kbps < data_rate_impl::kPlusInfinityVal / 1000, "");
63 return DataRate(kbps * 1000);
64 }
Sebastian Jansson942b3602018-05-30 15:47:44 +020065
66 template <
67 typename T,
68 typename std::enable_if<std::is_integral<T>::value>::type* = nullptr>
69 static DataRate bps(T bits_per_second) {
70 RTC_DCHECK_GE(bits_per_second, 0);
71 RTC_DCHECK_LT(bits_per_second, data_rate_impl::kPlusInfinityVal);
72 return DataRate(rtc::dchecked_cast<int64_t>(bits_per_second));
Sebastian Jansson30bd4032018-04-13 13:56:17 +020073 }
Sebastian Jansson942b3602018-05-30 15:47:44 +020074 template <
75 typename T,
76 typename std::enable_if<std::is_integral<T>::value>::type* = nullptr>
77 static DataRate kbps(T kilobits_per_sec) {
78 RTC_DCHECK_GE(kilobits_per_sec, 0);
79 RTC_DCHECK_LT(kilobits_per_sec, data_rate_impl::kPlusInfinityVal / 1000);
80 return DataRate::bps(rtc::dchecked_cast<int64_t>(kilobits_per_sec) * 1000);
Sebastian Jansson30bd4032018-04-13 13:56:17 +020081 }
Sebastian Jansson942b3602018-05-30 15:47:44 +020082
83 template <typename T,
84 typename std::enable_if<std::is_floating_point<T>::value>::type* =
85 nullptr>
86 static DataRate bps(T bits_per_second) {
87 if (bits_per_second == std::numeric_limits<T>::infinity()) {
88 return Infinity();
89 } else {
90 RTC_DCHECK(!std::isnan(bits_per_second));
91 RTC_DCHECK_GE(bits_per_second, 0);
92 RTC_DCHECK_LT(bits_per_second, data_rate_impl::kPlusInfinityVal);
93 return DataRate(rtc::dchecked_cast<int64_t>(bits_per_second));
94 }
Sebastian Jansson30bd4032018-04-13 13:56:17 +020095 }
Sebastian Jansson942b3602018-05-30 15:47:44 +020096 template <typename T,
97 typename std::enable_if<std::is_floating_point<T>::value>::type* =
98 nullptr>
99 static DataRate kbps(T kilobits_per_sec) {
100 return DataRate::bps(kilobits_per_sec * 1e3);
101 }
102
103 template <typename T = int64_t>
104 typename std::enable_if<std::is_integral<T>::value, T>::type bps() const {
Sebastian Jansson30bd4032018-04-13 13:56:17 +0200105 RTC_DCHECK(IsFinite());
Sebastian Jansson942b3602018-05-30 15:47:44 +0200106 return rtc::dchecked_cast<T>(bits_per_sec_);
Sebastian Jansson30bd4032018-04-13 13:56:17 +0200107 }
Sebastian Jansson942b3602018-05-30 15:47:44 +0200108 template <typename T = int64_t>
109 typename std::enable_if<std::is_integral<T>::value, T>::type kbps() const {
Sebastian Janssonc1c8b8e2018-08-07 15:29:04 +0200110 RTC_DCHECK(IsFinite());
111 return rtc::dchecked_cast<T>(UnsafeKilobitsPerSec());
Sebastian Jansson942b3602018-05-30 15:47:44 +0200112 }
113
114 template <typename T>
Sebastian Janssonc1c8b8e2018-08-07 15:29:04 +0200115 typename std::enable_if<std::is_floating_point<T>::value,
116 T>::type constexpr bps() const {
117 return IsInfinite() ? std::numeric_limits<T>::infinity() : bits_per_sec_;
Sebastian Jansson942b3602018-05-30 15:47:44 +0200118 }
119 template <typename T>
Sebastian Janssonc1c8b8e2018-08-07 15:29:04 +0200120 typename std::enable_if<std::is_floating_point<T>::value,
121 T>::type constexpr kbps() const {
Sebastian Jansson942b3602018-05-30 15:47:44 +0200122 return bps<T>() * 1e-3;
123 }
124
Sebastian Janssonc1c8b8e2018-08-07 15:29:04 +0200125 constexpr int64_t bps_or(int64_t fallback_value) const {
126 return IsFinite() ? bits_per_sec_ : fallback_value;
127 }
128 constexpr int64_t kbps_or(int64_t fallback_value) const {
129 return IsFinite() ? UnsafeKilobitsPerSec() : fallback_value;
130 }
131
Sebastian Jansson8e064192018-08-07 12:34:33 +0200132 constexpr bool IsZero() const { return bits_per_sec_ == 0; }
133 constexpr bool IsInfinite() const {
Sebastian Jansson30bd4032018-04-13 13:56:17 +0200134 return bits_per_sec_ == data_rate_impl::kPlusInfinityVal;
135 }
Sebastian Jansson8e064192018-08-07 12:34:33 +0200136 constexpr bool IsFinite() const { return !IsInfinite(); }
Sebastian Jansson6c19dec2018-10-05 13:23:57 +0200137 DataRate Clamped(DataRate min_rate, DataRate max_rate) const {
138 return std::max(min_rate, std::min(*this, max_rate));
139 }
140 void Clamp(DataRate min_rate, DataRate max_rate) {
141 *this = Clamped(min_rate, max_rate);
142 }
Sebastian Janssonaf21eab2018-09-04 18:34:45 +0200143 DataRate operator-(const DataRate& other) const {
144 return DataRate::bps(bps() - other.bps());
145 }
146 DataRate operator+(const DataRate& other) const {
147 return DataRate::bps(bps() + other.bps());
148 }
149 DataRate& operator-=(const DataRate& other) {
150 *this = *this - other;
151 return *this;
152 }
153 DataRate& operator+=(const DataRate& other) {
154 *this = *this + other;
155 return *this;
156 }
Sebastian Janssonc1c8b8e2018-08-07 15:29:04 +0200157 constexpr double operator/(const DataRate& other) const {
Sebastian Jansson942b3602018-05-30 15:47:44 +0200158 return bps<double>() / other.bps<double>();
159 }
Sebastian Janssonc1c8b8e2018-08-07 15:29:04 +0200160 constexpr bool operator==(const DataRate& other) const {
Sebastian Jansson30bd4032018-04-13 13:56:17 +0200161 return bits_per_sec_ == other.bits_per_sec_;
162 }
Sebastian Janssonc1c8b8e2018-08-07 15:29:04 +0200163 constexpr bool operator!=(const DataRate& other) const {
Sebastian Jansson30bd4032018-04-13 13:56:17 +0200164 return bits_per_sec_ != other.bits_per_sec_;
165 }
Sebastian Janssonc1c8b8e2018-08-07 15:29:04 +0200166 constexpr bool operator<=(const DataRate& other) const {
Sebastian Jansson30bd4032018-04-13 13:56:17 +0200167 return bits_per_sec_ <= other.bits_per_sec_;
168 }
Sebastian Janssonc1c8b8e2018-08-07 15:29:04 +0200169 constexpr bool operator>=(const DataRate& other) const {
Sebastian Jansson30bd4032018-04-13 13:56:17 +0200170 return bits_per_sec_ >= other.bits_per_sec_;
171 }
Sebastian Janssonc1c8b8e2018-08-07 15:29:04 +0200172 constexpr bool operator>(const DataRate& other) const {
Sebastian Jansson30bd4032018-04-13 13:56:17 +0200173 return bits_per_sec_ > other.bits_per_sec_;
174 }
Sebastian Janssonc1c8b8e2018-08-07 15:29:04 +0200175 constexpr bool operator<(const DataRate& other) const {
Sebastian Jansson30bd4032018-04-13 13:56:17 +0200176 return bits_per_sec_ < other.bits_per_sec_;
177 }
178
179 private:
180 // Bits per second used internally to simplify debugging by making the value
181 // more recognizable.
Sebastian Jansson8e064192018-08-07 12:34:33 +0200182 explicit constexpr DataRate(int64_t bits_per_second)
183 : bits_per_sec_(bits_per_second) {}
Sebastian Janssonc1c8b8e2018-08-07 15:29:04 +0200184 constexpr int64_t UnsafeKilobitsPerSec() const {
185 return (bits_per_sec_ + 500) / 1000;
186 }
Sebastian Jansson30bd4032018-04-13 13:56:17 +0200187 int64_t bits_per_sec_;
188};
Sebastian Jansson66fa5352018-04-30 16:54:57 +0200189
190inline DataRate operator*(const DataRate& rate, const double& scalar) {
Sebastian Jansson942b3602018-05-30 15:47:44 +0200191 return DataRate::bps(std::round(rate.bps() * scalar));
Sebastian Jansson66fa5352018-04-30 16:54:57 +0200192}
Sebastian Jansson30bd4032018-04-13 13:56:17 +0200193inline DataRate operator*(const double& scalar, const DataRate& rate) {
194 return rate * scalar;
195}
Sebastian Jansson66fa5352018-04-30 16:54:57 +0200196inline DataRate operator*(const DataRate& rate, const int64_t& scalar) {
Sebastian Jansson942b3602018-05-30 15:47:44 +0200197 return DataRate::bps(rate.bps() * scalar);
Sebastian Jansson66fa5352018-04-30 16:54:57 +0200198}
Sebastian Jansson30bd4032018-04-13 13:56:17 +0200199inline DataRate operator*(const int64_t& scalar, const DataRate& rate) {
200 return rate * scalar;
201}
Sebastian Jansson66fa5352018-04-30 16:54:57 +0200202inline DataRate operator*(const DataRate& rate, const int32_t& scalar) {
Sebastian Jansson942b3602018-05-30 15:47:44 +0200203 return DataRate::bps(rate.bps() * scalar);
Sebastian Jansson66fa5352018-04-30 16:54:57 +0200204}
Sebastian Jansson30bd4032018-04-13 13:56:17 +0200205inline DataRate operator*(const int32_t& scalar, const DataRate& rate) {
206 return rate * scalar;
207}
208
Sebastian Jansson66fa5352018-04-30 16:54:57 +0200209inline DataRate operator/(const DataSize& size, const TimeDelta& duration) {
Sebastian Jansson942b3602018-05-30 15:47:44 +0200210 return DataRate::bps(data_rate_impl::Microbits(size) / duration.us());
Sebastian Jansson66fa5352018-04-30 16:54:57 +0200211}
212inline TimeDelta operator/(const DataSize& size, const DataRate& rate) {
Sebastian Jansson942b3602018-05-30 15:47:44 +0200213 return TimeDelta::us(data_rate_impl::Microbits(size) / rate.bps());
Sebastian Jansson66fa5352018-04-30 16:54:57 +0200214}
215inline DataSize operator*(const DataRate& rate, const TimeDelta& duration) {
Sebastian Jansson942b3602018-05-30 15:47:44 +0200216 int64_t microbits = rate.bps() * duration.us();
Sebastian Jansson66fa5352018-04-30 16:54:57 +0200217 return DataSize::bytes((microbits + 4000000) / 8000000);
218}
219inline DataSize operator*(const TimeDelta& duration, const DataRate& rate) {
220 return rate * duration;
221}
Sebastian Janssone31be152018-04-16 12:41:47 +0200222
Sebastian Jansson30bd4032018-04-13 13:56:17 +0200223std::string ToString(const DataRate& value);
Sebastian Janssone31be152018-04-16 12:41:47 +0200224
Sebastian Jansson2afd2812018-08-23 14:44:05 +0200225#ifdef UNIT_TEST
226inline std::ostream& operator<<( // no-presubmit-check TODO(webrtc:8982)
227 std::ostream& stream, // no-presubmit-check TODO(webrtc:8982)
228 DataRate value) {
229 return stream << ToString(value);
230}
231#endif // UNIT_TEST
232
Sebastian Jansson30bd4032018-04-13 13:56:17 +0200233} // namespace webrtc
234
Sebastian Jansson6fae6ec2018-05-08 10:43:18 +0200235#endif // API_UNITS_DATA_RATE_H_