blob: 641c2c7f6d46c245897dbba6d99166ced272dd07 [file] [log] [blame]
kwiberg@webrtc.orgac2d27d2015-02-26 13:59:221/*
2 * Copyright (c) 2012 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
Mirko Bonadei71207422017-09-15 11:58:0911#include "common_types.h" // NOLINT(build/include)
kwiberg@webrtc.orgac2d27d2015-02-26 13:59:2212
13#include <string.h>
eladalond0244c22017-06-08 11:19:1314#include <algorithm>
danilchapef8d7732017-04-19 09:59:4815#include <limits>
16#include <type_traits>
kwiberg@webrtc.orgac2d27d2015-02-26 13:59:2217
Mirko Bonadei92ea95e2017-09-15 04:47:3118#include "rtc_base/checks.h"
Erik Språngbb60a3a2018-03-19 17:25:1019#include "rtc_base/strings/string_builder.h"
Mirko Bonadei92ea95e2017-09-15 04:47:3120#include "rtc_base/stringutils.h"
magjed10165ab2016-11-22 18:16:5721
kwiberg@webrtc.orgac2d27d2015-02-26 13:59:2222namespace webrtc {
23
Niels Möllerdef1ef52018-03-19 12:48:4424bool VideoCodecVP8::operator==(const VideoCodecVP8& other) const {
Niels Möllerdef1ef52018-03-19 12:48:4425 return (complexity == other.complexity &&
26 resilience == other.resilience &&
27 numberOfTemporalLayers == other.numberOfTemporalLayers &&
28 denoisingOn == other.denoisingOn &&
29 automaticResizeOn == other.automaticResizeOn &&
30 frameDroppingOn == other.frameDroppingOn &&
31 keyFrameInterval == other.keyFrameInterval);
32}
33
34bool VideoCodecVP9::operator==(const VideoCodecVP9& other) const {
35 return (complexity == other.complexity &&
36 resilienceOn == other.resilienceOn &&
37 numberOfTemporalLayers == other.numberOfTemporalLayers &&
38 denoisingOn == other.denoisingOn &&
39 frameDroppingOn == other.frameDroppingOn &&
40 keyFrameInterval == other.keyFrameInterval &&
41 adaptiveQpMode == other.adaptiveQpMode &&
42 automaticResizeOn == other.automaticResizeOn &&
43 numberOfSpatialLayers == other.numberOfSpatialLayers &&
44 flexibleMode == other.flexibleMode);
45}
46
47bool VideoCodecH264::operator==(const VideoCodecH264& other) const {
48 return (frameDroppingOn == other.frameDroppingOn &&
49 keyFrameInterval == other.keyFrameInterval &&
50 spsLen == other.spsLen &&
51 ppsLen == other.ppsLen &&
52 profile == other.profile &&
53 (spsLen == 0 || memcmp(spsData, other.spsData, spsLen) == 0) &&
54 (ppsLen == 0 || memcmp(ppsData, other.ppsData, ppsLen) == 0));
55}
56
57bool SpatialLayer::operator==(const SpatialLayer& other) const {
58 return (width == other.width &&
59 height == other.height &&
60 numberOfTemporalLayers == other.numberOfTemporalLayers &&
61 maxBitrate == other.maxBitrate &&
62 targetBitrate == other.targetBitrate &&
63 minBitrate == other.minBitrate &&
64 qpMax == other.qpMax &&
65 active == other.active);
66}
67
hta257dc392016-10-25 16:05:0668VideoCodec::VideoCodec()
69 : codecType(kVideoCodecUnknown),
hta257dc392016-10-25 16:05:0670 plType(0),
71 width(0),
72 height(0),
73 startBitrate(0),
74 maxBitrate(0),
75 minBitrate(0),
76 targetBitrate(0),
77 maxFramerate(0),
Seth Hampsonf6464c92018-01-17 21:55:1478 active(true),
hta257dc392016-10-25 16:05:0679 qpMax(0),
80 numberOfSimulcastStreams(0),
81 simulcastStream(),
82 spatialLayers(),
83 mode(kRealtimeVideo),
Erik Språng08127a92016-11-16 15:41:3084 expect_encode_from_texture(false),
ilnik04f4d122017-06-19 14:18:5585 timing_frame_thresholds({0, 0}),
hta527d3472016-11-17 07:23:0486 codec_specific_() {}
hta257dc392016-10-25 16:05:0687
88VideoCodecVP8* VideoCodec::VP8() {
89 RTC_DCHECK_EQ(codecType, kVideoCodecVP8);
hta527d3472016-11-17 07:23:0490 return &codec_specific_.VP8;
hta257dc392016-10-25 16:05:0691}
92
93const VideoCodecVP8& VideoCodec::VP8() const {
94 RTC_DCHECK_EQ(codecType, kVideoCodecVP8);
hta527d3472016-11-17 07:23:0495 return codec_specific_.VP8;
hta257dc392016-10-25 16:05:0696}
97
98VideoCodecVP9* VideoCodec::VP9() {
99 RTC_DCHECK_EQ(codecType, kVideoCodecVP9);
hta527d3472016-11-17 07:23:04100 return &codec_specific_.VP9;
hta257dc392016-10-25 16:05:06101}
102
103const VideoCodecVP9& VideoCodec::VP9() const {
104 RTC_DCHECK_EQ(codecType, kVideoCodecVP9);
hta527d3472016-11-17 07:23:04105 return codec_specific_.VP9;
hta257dc392016-10-25 16:05:06106}
107
108VideoCodecH264* VideoCodec::H264() {
109 RTC_DCHECK_EQ(codecType, kVideoCodecH264);
hta527d3472016-11-17 07:23:04110 return &codec_specific_.H264;
hta257dc392016-10-25 16:05:06111}
112
113const VideoCodecH264& VideoCodec::H264() const {
114 RTC_DCHECK_EQ(codecType, kVideoCodecH264);
hta527d3472016-11-17 07:23:04115 return codec_specific_.H264;
hta257dc392016-10-25 16:05:06116}
117
Erik Språng08127a92016-11-16 15:41:30118static const char* kPayloadNameVp8 = "VP8";
119static const char* kPayloadNameVp9 = "VP9";
120static const char* kPayloadNameH264 = "H264";
121static const char* kPayloadNameI420 = "I420";
122static const char* kPayloadNameRED = "RED";
123static const char* kPayloadNameULPFEC = "ULPFEC";
Danil Chapovalov3c706972018-01-30 10:11:08124static const char* kPayloadNameFlexfec = "flexfec-03";
Erik Språng08127a92016-11-16 15:41:30125static const char* kPayloadNameGeneric = "Generic";
Emircan Uysalerd7ae3c32018-01-25 21:01:09126static const char* kPayloadNameMultiplex = "Multiplex";
Erik Språng08127a92016-11-16 15:41:30127
magjed10165ab2016-11-22 18:16:57128static bool CodecNamesEq(const char* name1, const char* name2) {
129 return _stricmp(name1, name2) == 0;
130}
131
kthelgason1cdddc92017-08-24 10:52:48132const char* CodecTypeToPayloadString(VideoCodecType type) {
Erik Språng08127a92016-11-16 15:41:30133 switch (type) {
134 case kVideoCodecVP8:
kthelgason1cdddc92017-08-24 10:52:48135 return kPayloadNameVp8;
Erik Språng08127a92016-11-16 15:41:30136 case kVideoCodecVP9:
kthelgason1cdddc92017-08-24 10:52:48137 return kPayloadNameVp9;
Erik Språng08127a92016-11-16 15:41:30138 case kVideoCodecH264:
kthelgason1cdddc92017-08-24 10:52:48139 return kPayloadNameH264;
Erik Språng08127a92016-11-16 15:41:30140 case kVideoCodecI420:
kthelgason1cdddc92017-08-24 10:52:48141 return kPayloadNameI420;
Erik Språng08127a92016-11-16 15:41:30142 case kVideoCodecRED:
kthelgason1cdddc92017-08-24 10:52:48143 return kPayloadNameRED;
Erik Språng08127a92016-11-16 15:41:30144 case kVideoCodecULPFEC:
kthelgason1cdddc92017-08-24 10:52:48145 return kPayloadNameULPFEC;
Danil Chapovalov3c706972018-01-30 10:11:08146 case kVideoCodecFlexfec:
147 return kPayloadNameFlexfec;
Emircan Uysaler0a375472017-12-11 06:51:02148 // Other codecs default to generic.
Emircan Uysalerd7ae3c32018-01-25 21:01:09149 case kVideoCodecMultiplex:
Emircan Uysaler0a375472017-12-11 06:51:02150 case kVideoCodecGeneric:
151 case kVideoCodecUnknown:
kthelgason1cdddc92017-08-24 10:52:48152 return kPayloadNameGeneric;
Erik Språng08127a92016-11-16 15:41:30153 }
Emircan Uysaler0a375472017-12-11 06:51:02154 return kPayloadNameGeneric;
Erik Språng08127a92016-11-16 15:41:30155}
156
kthelgason1cdddc92017-08-24 10:52:48157VideoCodecType PayloadStringToCodecType(const std::string& name) {
magjed10165ab2016-11-22 18:16:57158 if (CodecNamesEq(name.c_str(), kPayloadNameVp8))
kthelgason1cdddc92017-08-24 10:52:48159 return kVideoCodecVP8;
magjed10165ab2016-11-22 18:16:57160 if (CodecNamesEq(name.c_str(), kPayloadNameVp9))
kthelgason1cdddc92017-08-24 10:52:48161 return kVideoCodecVP9;
magjed10165ab2016-11-22 18:16:57162 if (CodecNamesEq(name.c_str(), kPayloadNameH264))
kthelgason1cdddc92017-08-24 10:52:48163 return kVideoCodecH264;
magjed10165ab2016-11-22 18:16:57164 if (CodecNamesEq(name.c_str(), kPayloadNameI420))
kthelgason1cdddc92017-08-24 10:52:48165 return kVideoCodecI420;
magjed10165ab2016-11-22 18:16:57166 if (CodecNamesEq(name.c_str(), kPayloadNameRED))
kthelgason1cdddc92017-08-24 10:52:48167 return kVideoCodecRED;
magjed10165ab2016-11-22 18:16:57168 if (CodecNamesEq(name.c_str(), kPayloadNameULPFEC))
kthelgason1cdddc92017-08-24 10:52:48169 return kVideoCodecULPFEC;
Danil Chapovalov3c706972018-01-30 10:11:08170 if (CodecNamesEq(name.c_str(), kPayloadNameFlexfec))
171 return kVideoCodecFlexfec;
Emircan Uysalerd7ae3c32018-01-25 21:01:09172 if (CodecNamesEq(name.c_str(), kPayloadNameMultiplex))
173 return kVideoCodecMultiplex;
kthelgason1cdddc92017-08-24 10:52:48174 return kVideoCodecGeneric;
175}
176
Erik Språng08127a92016-11-16 15:41:30177const uint32_t BitrateAllocation::kMaxBitrateBps =
178 std::numeric_limits<uint32_t>::max();
179
erikvarga@webrtc.org01f2ec32017-11-15 13:58:23180BitrateAllocation::BitrateAllocation() : sum_(0), bitrates_{}, has_bitrate_{} {}
Erik Språng08127a92016-11-16 15:41:30181
182bool BitrateAllocation::SetBitrate(size_t spatial_index,
183 size_t temporal_index,
184 uint32_t bitrate_bps) {
sprang6d314c72016-12-06 14:08:53185 RTC_CHECK_LT(spatial_index, kMaxSpatialLayers);
186 RTC_CHECK_LT(temporal_index, kMaxTemporalStreams);
187 RTC_CHECK_LE(bitrates_[spatial_index][temporal_index], sum_);
Erik Språng08127a92016-11-16 15:41:30188 uint64_t new_bitrate_sum_bps = sum_;
189 new_bitrate_sum_bps -= bitrates_[spatial_index][temporal_index];
190 new_bitrate_sum_bps += bitrate_bps;
191 if (new_bitrate_sum_bps > kMaxBitrateBps)
192 return false;
193
194 bitrates_[spatial_index][temporal_index] = bitrate_bps;
erikvarga@webrtc.org01f2ec32017-11-15 13:58:23195 has_bitrate_[spatial_index][temporal_index] = true;
Erik Språng08127a92016-11-16 15:41:30196 sum_ = static_cast<uint32_t>(new_bitrate_sum_bps);
197 return true;
198}
199
erikvarga@webrtc.org01f2ec32017-11-15 13:58:23200bool BitrateAllocation::HasBitrate(size_t spatial_index,
201 size_t temporal_index) const {
202 RTC_CHECK_LT(spatial_index, kMaxSpatialLayers);
203 RTC_CHECK_LT(temporal_index, kMaxTemporalStreams);
204 return has_bitrate_[spatial_index][temporal_index];
205}
206
Erik Språng08127a92016-11-16 15:41:30207uint32_t BitrateAllocation::GetBitrate(size_t spatial_index,
208 size_t temporal_index) const {
sprang6d314c72016-12-06 14:08:53209 RTC_CHECK_LT(spatial_index, kMaxSpatialLayers);
210 RTC_CHECK_LT(temporal_index, kMaxTemporalStreams);
Erik Språng08127a92016-11-16 15:41:30211 return bitrates_[spatial_index][temporal_index];
212}
213
erikvarga@webrtc.org01f2ec32017-11-15 13:58:23214// Whether the specific spatial layers has the bitrate set in any of its
215// temporal layers.
216bool BitrateAllocation::IsSpatialLayerUsed(size_t spatial_index) const {
217 RTC_CHECK_LT(spatial_index, kMaxSpatialLayers);
218 for (int i = 0; i < kMaxTemporalStreams; ++i) {
219 if (has_bitrate_[spatial_index][i])
220 return true;
221 }
222 return false;
223}
224
Erik Språng08127a92016-11-16 15:41:30225// Get the sum of all the temporal layer for a specific spatial layer.
226uint32_t BitrateAllocation::GetSpatialLayerSum(size_t spatial_index) const {
sprang6d314c72016-12-06 14:08:53227 RTC_CHECK_LT(spatial_index, kMaxSpatialLayers);
Sergey Silkin86684962018-03-28 17:32:37228 return GetTemporalLayerSum(spatial_index, kMaxTemporalStreams - 1);
229}
230
231uint32_t BitrateAllocation::GetTemporalLayerSum(size_t spatial_index,
232 size_t temporal_index) const {
233 RTC_CHECK_LT(spatial_index, kMaxSpatialLayers);
234 RTC_CHECK_LT(temporal_index, kMaxTemporalStreams);
Erik Språng08127a92016-11-16 15:41:30235 uint32_t sum = 0;
Sergey Silkin86684962018-03-28 17:32:37236 for (size_t i = 0; i <= temporal_index; ++i) {
Erik Språng08127a92016-11-16 15:41:30237 sum += bitrates_[spatial_index][i];
Sergey Silkin86684962018-03-28 17:32:37238 }
Erik Språng08127a92016-11-16 15:41:30239 return sum;
240}
241
Erik Språngbb60a3a2018-03-19 17:25:10242std::vector<uint32_t> BitrateAllocation::GetTemporalLayerAllocation(
243 size_t spatial_index) const {
244 RTC_CHECK_LT(spatial_index, kMaxSpatialLayers);
245 std::vector<uint32_t> temporal_rates;
246
247 // Find the highest temporal layer with a defined bitrate in order to
248 // determine the size of the temporal layer allocation.
249 for (size_t i = kMaxTemporalStreams; i > 0; --i) {
250 if (has_bitrate_[spatial_index][i - 1]) {
251 temporal_rates.resize(i);
252 break;
253 }
254 }
255
256 for (size_t i = 0; i < temporal_rates.size(); ++i) {
257 temporal_rates[i] = bitrates_[spatial_index][i];
258 }
259
260 return temporal_rates;
261}
262
sprangd0fc37a2017-06-22 12:40:25263std::string BitrateAllocation::ToString() const {
264 if (sum_ == 0)
265 return "BitrateAllocation [ [] ]";
266
Erik Språngbb60a3a2018-03-19 17:25:10267 // Max string length in practice is 260, but let's have some overhead and
268 // round up to nearest power of two.
269 char string_buf[512];
270 rtc::SimpleStringBuilder ssb(string_buf);
271
272 ssb << "BitrateAllocation [";
sprangd0fc37a2017-06-22 12:40:25273 uint32_t spatial_cumulator = 0;
274 for (int si = 0; si < kMaxSpatialLayers; ++si) {
275 RTC_DCHECK_LE(spatial_cumulator, sum_);
276 if (spatial_cumulator == sum_)
277 break;
278
279 const uint32_t layer_sum = GetSpatialLayerSum(si);
280 if (layer_sum == sum_) {
Erik Språngbb60a3a2018-03-19 17:25:10281 ssb << " [";
sprangd0fc37a2017-06-22 12:40:25282 } else {
283 if (si > 0)
Erik Språngbb60a3a2018-03-19 17:25:10284 ssb << ",";
285 ssb << '\n' << " [";
sprangd0fc37a2017-06-22 12:40:25286 }
287 spatial_cumulator += layer_sum;
288
289 uint32_t temporal_cumulator = 0;
290 for (int ti = 0; ti < kMaxTemporalStreams; ++ti) {
291 RTC_DCHECK_LE(temporal_cumulator, layer_sum);
292 if (temporal_cumulator == layer_sum)
293 break;
294
295 if (ti > 0)
Erik Språngbb60a3a2018-03-19 17:25:10296 ssb << ", ";
sprangd0fc37a2017-06-22 12:40:25297
298 uint32_t bitrate = bitrates_[si][ti];
Erik Språngbb60a3a2018-03-19 17:25:10299 ssb << bitrate;
sprangd0fc37a2017-06-22 12:40:25300 temporal_cumulator += bitrate;
301 }
Erik Språngbb60a3a2018-03-19 17:25:10302 ssb << "]";
sprangd0fc37a2017-06-22 12:40:25303 }
304
305 RTC_DCHECK_EQ(spatial_cumulator, sum_);
Erik Språngbb60a3a2018-03-19 17:25:10306 ssb << " ]";
307 return ssb.str();
sprangd0fc37a2017-06-22 12:40:25308}
309
kwiberg@webrtc.orgac2d27d2015-02-26 13:59:22310} // namespace webrtc