blob: 1122d5e891ac596fa058e72bb62a1b44d836dbaf [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 Jansson30bd4032018-04-13 13:56:17 +020013#include <stdint.h>
14#include <cmath>
15#include <limits>
16#include <string>
17
18#include "rtc_base/checks.h"
Sebastian Jansson942b3602018-05-30 15:47:44 +020019#include "rtc_base/numerics/safe_conversions.h"
Sebastian Jansson30bd4032018-04-13 13:56:17 +020020
Sebastian Jansson6fae6ec2018-05-08 10:43:18 +020021#include "api/units/data_size.h"
22#include "api/units/time_delta.h"
Sebastian Janssone31be152018-04-16 12:41:47 +020023
Sebastian Jansson30bd4032018-04-13 13:56:17 +020024namespace webrtc {
25namespace data_rate_impl {
26constexpr int64_t kPlusInfinityVal = std::numeric_limits<int64_t>::max();
Sebastian Jansson66fa5352018-04-30 16:54:57 +020027
28inline int64_t Microbits(const DataSize& size) {
29 constexpr int64_t kMaxBeforeConversion =
30 std::numeric_limits<int64_t>::max() / 8000000;
31 RTC_DCHECK_LE(size.bytes(), kMaxBeforeConversion)
32 << "size is too large to be expressed in microbytes";
33 return size.bytes() * 8000000;
34}
Sebastian Jansson30bd4032018-04-13 13:56:17 +020035} // namespace data_rate_impl
36
37// DataRate is a class that represents a given data rate. This can be used to
Sebastian Jansson66fa5352018-04-30 16:54:57 +020038// represent bandwidth, encoding bitrate, etc. The internal storage is bits per
39// second (bps).
Sebastian Jansson30bd4032018-04-13 13:56:17 +020040class DataRate {
41 public:
Sebastian Jansson3b69b192018-05-07 13:51:51 +020042 DataRate() = delete;
Sebastian Jansson8e064192018-08-07 12:34:33 +020043 static constexpr DataRate Zero() { return DataRate(0); }
44 static constexpr DataRate Infinity() {
Sebastian Jansson30bd4032018-04-13 13:56:17 +020045 return DataRate(data_rate_impl::kPlusInfinityVal);
46 }
Sebastian Jansson942b3602018-05-30 15:47:44 +020047
48 template <
49 typename T,
50 typename std::enable_if<std::is_integral<T>::value>::type* = nullptr>
51 static DataRate bps(T bits_per_second) {
52 RTC_DCHECK_GE(bits_per_second, 0);
53 RTC_DCHECK_LT(bits_per_second, data_rate_impl::kPlusInfinityVal);
54 return DataRate(rtc::dchecked_cast<int64_t>(bits_per_second));
Sebastian Jansson30bd4032018-04-13 13:56:17 +020055 }
Sebastian Jansson942b3602018-05-30 15:47:44 +020056 template <
57 typename T,
58 typename std::enable_if<std::is_integral<T>::value>::type* = nullptr>
59 static DataRate kbps(T kilobits_per_sec) {
60 RTC_DCHECK_GE(kilobits_per_sec, 0);
61 RTC_DCHECK_LT(kilobits_per_sec, data_rate_impl::kPlusInfinityVal / 1000);
62 return DataRate::bps(rtc::dchecked_cast<int64_t>(kilobits_per_sec) * 1000);
Sebastian Jansson30bd4032018-04-13 13:56:17 +020063 }
Sebastian Jansson942b3602018-05-30 15:47:44 +020064
65 template <typename T,
66 typename std::enable_if<std::is_floating_point<T>::value>::type* =
67 nullptr>
68 static DataRate bps(T bits_per_second) {
69 if (bits_per_second == std::numeric_limits<T>::infinity()) {
70 return Infinity();
71 } else {
72 RTC_DCHECK(!std::isnan(bits_per_second));
73 RTC_DCHECK_GE(bits_per_second, 0);
74 RTC_DCHECK_LT(bits_per_second, data_rate_impl::kPlusInfinityVal);
75 return DataRate(rtc::dchecked_cast<int64_t>(bits_per_second));
76 }
Sebastian Jansson30bd4032018-04-13 13:56:17 +020077 }
Sebastian Jansson942b3602018-05-30 15:47:44 +020078 template <typename T,
79 typename std::enable_if<std::is_floating_point<T>::value>::type* =
80 nullptr>
81 static DataRate kbps(T kilobits_per_sec) {
82 return DataRate::bps(kilobits_per_sec * 1e3);
83 }
84
85 template <typename T = int64_t>
86 typename std::enable_if<std::is_integral<T>::value, T>::type bps() const {
Sebastian Jansson30bd4032018-04-13 13:56:17 +020087 RTC_DCHECK(IsFinite());
Sebastian Jansson942b3602018-05-30 15:47:44 +020088 return rtc::dchecked_cast<T>(bits_per_sec_);
Sebastian Jansson30bd4032018-04-13 13:56:17 +020089 }
Sebastian Jansson942b3602018-05-30 15:47:44 +020090 template <typename T = int64_t>
91 typename std::enable_if<std::is_integral<T>::value, T>::type kbps() const {
92 return rtc::dchecked_cast<T>((bps() + 500) / 1000);
93 }
94
95 template <typename T>
96 typename std::enable_if<std::is_floating_point<T>::value, T>::type bps()
97 const {
98 if (IsInfinite()) {
99 return std::numeric_limits<T>::infinity();
100 } else {
101 return bits_per_sec_;
102 }
103 }
104 template <typename T>
105 typename std::enable_if<std::is_floating_point<T>::value, T>::type kbps()
106 const {
107 return bps<T>() * 1e-3;
108 }
109
Sebastian Jansson8e064192018-08-07 12:34:33 +0200110 constexpr bool IsZero() const { return bits_per_sec_ == 0; }
111 constexpr bool IsInfinite() const {
Sebastian Jansson30bd4032018-04-13 13:56:17 +0200112 return bits_per_sec_ == data_rate_impl::kPlusInfinityVal;
113 }
Sebastian Jansson8e064192018-08-07 12:34:33 +0200114 constexpr bool IsFinite() const { return !IsInfinite(); }
Sebastian Jansson66fa5352018-04-30 16:54:57 +0200115
Sebastian Jansson942b3602018-05-30 15:47:44 +0200116 double operator/(const DataRate& other) const {
117 return bps<double>() / other.bps<double>();
118 }
Sebastian Jansson30bd4032018-04-13 13:56:17 +0200119 bool operator==(const DataRate& other) const {
120 return bits_per_sec_ == other.bits_per_sec_;
121 }
122 bool operator!=(const DataRate& other) const {
123 return bits_per_sec_ != other.bits_per_sec_;
124 }
125 bool operator<=(const DataRate& other) const {
126 return bits_per_sec_ <= other.bits_per_sec_;
127 }
128 bool operator>=(const DataRate& other) const {
129 return bits_per_sec_ >= other.bits_per_sec_;
130 }
131 bool operator>(const DataRate& other) const {
132 return bits_per_sec_ > other.bits_per_sec_;
133 }
134 bool operator<(const DataRate& other) const {
135 return bits_per_sec_ < other.bits_per_sec_;
136 }
137
138 private:
139 // Bits per second used internally to simplify debugging by making the value
140 // more recognizable.
Sebastian Jansson8e064192018-08-07 12:34:33 +0200141 explicit constexpr DataRate(int64_t bits_per_second)
142 : bits_per_sec_(bits_per_second) {}
Sebastian Jansson30bd4032018-04-13 13:56:17 +0200143 int64_t bits_per_sec_;
144};
Sebastian Jansson66fa5352018-04-30 16:54:57 +0200145
146inline DataRate operator*(const DataRate& rate, const double& scalar) {
Sebastian Jansson942b3602018-05-30 15:47:44 +0200147 return DataRate::bps(std::round(rate.bps() * scalar));
Sebastian Jansson66fa5352018-04-30 16:54:57 +0200148}
Sebastian Jansson30bd4032018-04-13 13:56:17 +0200149inline DataRate operator*(const double& scalar, const DataRate& rate) {
150 return rate * scalar;
151}
Sebastian Jansson66fa5352018-04-30 16:54:57 +0200152inline DataRate operator*(const DataRate& rate, const int64_t& scalar) {
Sebastian Jansson942b3602018-05-30 15:47:44 +0200153 return DataRate::bps(rate.bps() * scalar);
Sebastian Jansson66fa5352018-04-30 16:54:57 +0200154}
Sebastian Jansson30bd4032018-04-13 13:56:17 +0200155inline DataRate operator*(const int64_t& scalar, const DataRate& rate) {
156 return rate * scalar;
157}
Sebastian Jansson66fa5352018-04-30 16:54:57 +0200158inline DataRate operator*(const DataRate& rate, const int32_t& scalar) {
Sebastian Jansson942b3602018-05-30 15:47:44 +0200159 return DataRate::bps(rate.bps() * scalar);
Sebastian Jansson66fa5352018-04-30 16:54:57 +0200160}
Sebastian Jansson30bd4032018-04-13 13:56:17 +0200161inline DataRate operator*(const int32_t& scalar, const DataRate& rate) {
162 return rate * scalar;
163}
164
Sebastian Jansson66fa5352018-04-30 16:54:57 +0200165inline DataRate operator/(const DataSize& size, const TimeDelta& duration) {
Sebastian Jansson942b3602018-05-30 15:47:44 +0200166 return DataRate::bps(data_rate_impl::Microbits(size) / duration.us());
Sebastian Jansson66fa5352018-04-30 16:54:57 +0200167}
168inline TimeDelta operator/(const DataSize& size, const DataRate& rate) {
Sebastian Jansson942b3602018-05-30 15:47:44 +0200169 return TimeDelta::us(data_rate_impl::Microbits(size) / rate.bps());
Sebastian Jansson66fa5352018-04-30 16:54:57 +0200170}
171inline DataSize operator*(const DataRate& rate, const TimeDelta& duration) {
Sebastian Jansson942b3602018-05-30 15:47:44 +0200172 int64_t microbits = rate.bps() * duration.us();
Sebastian Jansson66fa5352018-04-30 16:54:57 +0200173 return DataSize::bytes((microbits + 4000000) / 8000000);
174}
175inline DataSize operator*(const TimeDelta& duration, const DataRate& rate) {
176 return rate * duration;
177}
Sebastian Janssone31be152018-04-16 12:41:47 +0200178
Sebastian Jansson30bd4032018-04-13 13:56:17 +0200179std::string ToString(const DataRate& value);
Sebastian Janssone31be152018-04-16 12:41:47 +0200180
Sebastian Jansson30bd4032018-04-13 13:56:17 +0200181} // namespace webrtc
182
Sebastian Jansson6fae6ec2018-05-08 10:43:18 +0200183#endif // API_UNITS_DATA_RATE_H_