blob: 188e89fd8c76244080be88bda85f56b91ef74349 [file] [log] [blame]
Sebastian Jansson98b07e912018-09-27 11:47:011/*
2 * Copyright 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#include "test/scenario/video_stream.h"
11
12#include <algorithm>
Mirko Bonadei317a1f02019-09-17 15:06:1813#include <memory>
Sebastian Jansson98b07e912018-09-27 11:47:0114#include <utility>
15
Mirko Bonadei06d35592020-04-01 11:43:0816#include "absl/strings/match.h"
Artem Titov33f9d2b2019-12-05 14:59:0017#include "api/test/create_frame_generator.h"
18#include "api/test/frame_generator_interface.h"
Danil Chapovalov99b71df2018-10-26 13:57:4819#include "api/test/video/function_video_encoder_factory.h"
Jiawei Ouc2ebe212018-11-08 18:02:5620#include "api/video/builtin_video_bitrate_allocator_factory.h"
Steve Anton10542f22019-01-11 17:11:0021#include "media/base/media_constants.h"
22#include "media/engine/internal_decoder_factory.h"
23#include "media/engine/internal_encoder_factory.h"
24#include "media/engine/webrtc_video_engine.h"
Niels Möller4a5a3da2022-06-17 09:26:0225#include "modules/video_coding/svc/scalability_mode_util.h"
Sebastian Jansson98b07e912018-09-27 11:47:0126#include "test/call_test.h"
27#include "test/fake_encoder.h"
Sebastian Jansson98b07e912018-09-27 11:47:0128#include "test/scenario/hardware_codecs.h"
Steve Anton10542f22019-01-11 17:11:0029#include "test/testsupport/file_utils.h"
Artem Titov8a9f3a82023-04-25 07:56:4930#include "test/video_test_constants.h"
Jonas Oreland1262eb52022-09-27 14:53:0431#include "video/config/encoder_stream_factory.h"
Sebastian Jansson98b07e912018-09-27 11:47:0132
33namespace webrtc {
34namespace test {
35namespace {
Elad Alond8d32482019-02-18 22:45:5736enum : int { // The first valid value is 1.
37 kTransportSequenceNumberExtensionId = 1,
Sebastian Janssone112bb82019-06-13 15:36:0138 kAbsSendTimeExtensionId,
Elad Alond8d32482019-02-18 22:45:5739 kVideoContentTypeExtensionId,
40 kVideoRotationRtpExtensionId,
41};
42
Sebastian Jansson98b07e912018-09-27 11:47:0143uint8_t CodecTypeToPayloadType(VideoCodecType codec_type) {
44 switch (codec_type) {
45 case VideoCodecType::kVideoCodecGeneric:
Artem Titov8a9f3a82023-04-25 07:56:4946 return VideoTestConstants::kFakeVideoSendPayloadType;
Sebastian Jansson98b07e912018-09-27 11:47:0147 case VideoCodecType::kVideoCodecVP8:
Artem Titov8a9f3a82023-04-25 07:56:4948 return VideoTestConstants::kPayloadTypeVP8;
Sebastian Jansson98b07e912018-09-27 11:47:0149 case VideoCodecType::kVideoCodecVP9:
Artem Titov8a9f3a82023-04-25 07:56:4950 return VideoTestConstants::kPayloadTypeVP9;
Sebastian Jansson98b07e912018-09-27 11:47:0151 case VideoCodecType::kVideoCodecH264:
Artem Titov8a9f3a82023-04-25 07:56:4952 return VideoTestConstants::kPayloadTypeH264;
qwu16ae82df72023-09-20 05:10:3153 case VideoCodecType::kVideoCodecH265:
54 return VideoTestConstants::kPayloadTypeH265;
Sebastian Jansson98b07e912018-09-27 11:47:0155 default:
Artem Titovd3251962021-11-15 15:57:0756 RTC_DCHECK_NOTREACHED();
Sebastian Jansson98b07e912018-09-27 11:47:0157 }
58 return {};
59}
60std::string CodecTypeToCodecName(VideoCodecType codec_type) {
61 switch (codec_type) {
62 case VideoCodecType::kVideoCodecGeneric:
63 return "";
64 case VideoCodecType::kVideoCodecVP8:
65 return cricket::kVp8CodecName;
66 case VideoCodecType::kVideoCodecVP9:
67 return cricket::kVp9CodecName;
68 case VideoCodecType::kVideoCodecH264:
69 return cricket::kH264CodecName;
qwu16ae82df72023-09-20 05:10:3170 case VideoCodecType::kVideoCodecH265:
71 return cricket::kH265CodecName;
Sebastian Jansson98b07e912018-09-27 11:47:0172 default:
Artem Titovd3251962021-11-15 15:57:0773 RTC_DCHECK_NOTREACHED();
Sebastian Jansson98b07e912018-09-27 11:47:0174 }
75 return {};
76}
Sebastian Janssonf0c366b2019-02-14 12:18:4277VideoEncoderConfig::ContentType ConvertContentType(
78 VideoStreamConfig::Encoder::ContentType content_type) {
79 switch (content_type) {
80 case VideoStreamConfig::Encoder::ContentType::kVideo:
81 return VideoEncoderConfig::ContentType::kRealtimeVideo;
Sebastian Janssonf0c366b2019-02-14 12:18:4282 case VideoStreamConfig::Encoder::ContentType::kScreen:
83 return VideoEncoderConfig::ContentType::kScreen;
84 }
85}
Niels Möller4a5a3da2022-06-17 09:26:0286
Sebastian Janssonf2727fb2019-02-18 15:54:5587std::string TransformFilePath(std::string path) {
88 static const std::string resource_prefix = "res://";
Mirko Bonadei5686e342020-12-16 06:29:0289 int ext_pos = path.rfind('.');
Sebastian Janssonf2727fb2019-02-18 15:54:5590 if (ext_pos < 0) {
91 return test::ResourcePath(path, "yuv");
Mirko Bonadei06d35592020-04-01 11:43:0892 } else if (absl::StartsWith(path, resource_prefix)) {
Sebastian Janssonf2727fb2019-02-18 15:54:5593 std::string name = path.substr(resource_prefix.length(), ext_pos);
94 std::string ext = path.substr(ext_pos, path.size());
95 return test::ResourcePath(name, ext);
96 }
97 return path;
98}
99
Sebastian Jansson3e66a492020-01-14 11:30:13100VideoSendStream::Config CreateVideoSendStreamConfig(
101 VideoStreamConfig config,
102 std::vector<uint32_t> ssrcs,
103 std::vector<uint32_t> rtx_ssrcs,
104 Transport* send_transport) {
Sebastian Jansson98b07e912018-09-27 11:47:01105 VideoSendStream::Config send_config(send_transport);
106 send_config.rtp.payload_name = CodecTypeToPayloadString(config.encoder.codec);
107 send_config.rtp.payload_type = CodecTypeToPayloadType(config.encoder.codec);
Sebastian Janssonc57b0ee2019-06-26 14:22:39108 send_config.rtp.nack.rtp_history_ms =
109 config.stream.nack_history_time.ms<int>();
Sebastian Jansson98b07e912018-09-27 11:47:01110
111 send_config.rtp.ssrcs = ssrcs;
112 send_config.rtp.extensions = GetVideoRtpExtensions(config);
113
Sebastian Jansson3e66a492020-01-14 11:30:13114 if (config.stream.use_rtx) {
Artem Titov8a9f3a82023-04-25 07:56:49115 send_config.rtp.rtx.payload_type = VideoTestConstants::kSendRtxPayloadType;
Sebastian Jansson3e66a492020-01-14 11:30:13116 send_config.rtp.rtx.ssrcs = rtx_ssrcs;
117 }
Sebastian Jansson98b07e912018-09-27 11:47:01118 if (config.stream.use_flexfec) {
Artem Titov8a9f3a82023-04-25 07:56:49119 send_config.rtp.flexfec.payload_type =
120 VideoTestConstants::kFlexfecPayloadType;
121 send_config.rtp.flexfec.ssrc = VideoTestConstants::kFlexfecSendSsrc;
Sebastian Jansson98b07e912018-09-27 11:47:01122 send_config.rtp.flexfec.protected_media_ssrcs = ssrcs;
123 }
124 if (config.stream.use_ulpfec) {
Artem Titov8a9f3a82023-04-25 07:56:49125 send_config.rtp.ulpfec.red_payload_type =
126 VideoTestConstants::kRedPayloadType;
127 send_config.rtp.ulpfec.ulpfec_payload_type =
128 VideoTestConstants::kUlpfecPayloadType;
129 send_config.rtp.ulpfec.red_rtx_payload_type =
130 VideoTestConstants::kRtxRedPayloadType;
Sebastian Jansson98b07e912018-09-27 11:47:01131 }
132 return send_config;
133}
134rtc::scoped_refptr<VideoEncoderConfig::EncoderSpecificSettings>
Sebastian Jansson06c51452019-02-18 15:55:15135CreateVp9SpecificSettings(VideoStreamConfig video_config) {
136 constexpr auto kScreen = VideoStreamConfig::Encoder::ContentType::kScreen;
137 VideoStreamConfig::Encoder conf = video_config.encoder;
138 VideoCodecVP9 vp9 = VideoEncoder::GetDefaultVp9Settings();
Niels Möller4a5a3da2022-06-17 09:26:02139 // TODO(bugs.webrtc.org/11607): Support separate scalability mode per
140 // simulcast stream.
141 ScalabilityMode scalability_mode = conf.simulcast_streams[0];
Sebastian Jansson06c51452019-02-18 15:55:15142 vp9.keyFrameInterval = conf.key_frame_interval.value_or(0);
Niels Möller4a5a3da2022-06-17 09:26:02143 vp9.numberOfTemporalLayers =
144 ScalabilityModeToNumTemporalLayers(scalability_mode);
145 vp9.numberOfSpatialLayers =
146 ScalabilityModeToNumSpatialLayers(scalability_mode);
147 vp9.interLayerPred = ScalabilityModeToInterLayerPredMode(scalability_mode);
Sebastian Janssonf2727fb2019-02-18 15:54:55148
Sebastian Jansson06c51452019-02-18 15:55:15149 if (conf.content_type == kScreen &&
Niels Möller4a5a3da2022-06-17 09:26:02150 (video_config.source.framerate > 5 || vp9.numberOfSpatialLayers >= 3)) {
Sebastian Jansson06c51452019-02-18 15:55:15151 vp9.flexibleMode = true;
152 }
153
Niels Möller4a5a3da2022-06-17 09:26:02154 if (conf.content_type == kScreen || vp9.numberOfTemporalLayers > 1 ||
155 vp9.numberOfSpatialLayers > 1) {
Sebastian Jansson06c51452019-02-18 15:55:15156 vp9.automaticResizeOn = false;
157 vp9.denoisingOn = false;
158 } else {
159 vp9.automaticResizeOn = conf.single.automatic_scaling;
160 vp9.denoisingOn = conf.single.denoising;
Sebastian Janssonf2727fb2019-02-18 15:54:55161 }
Tommi87f70902021-04-27 12:43:08162 return rtc::make_ref_counted<VideoEncoderConfig::Vp9EncoderSpecificSettings>(
163 vp9);
Sebastian Janssonf0c366b2019-02-14 12:18:42164}
165
166rtc::scoped_refptr<VideoEncoderConfig::EncoderSpecificSettings>
167CreateVp8SpecificSettings(VideoStreamConfig config) {
168 VideoCodecVP8 vp8_settings = VideoEncoder::GetDefaultVp8Settings();
Sebastian Janssonf0c366b2019-02-14 12:18:42169 vp8_settings.keyFrameInterval = config.encoder.key_frame_interval.value_or(0);
Niels Möller4a5a3da2022-06-17 09:26:02170 // TODO(bugs.webrtc.org/11607): Support separate scalability mode per
171 // simulcast stream.
172 ScalabilityMode scalability_mode = config.encoder.simulcast_streams[0];
173 vp8_settings.numberOfTemporalLayers =
174 ScalabilityModeToNumTemporalLayers(scalability_mode);
175 if (vp8_settings.numberOfTemporalLayers > 1 ||
176 config.encoder.simulcast_streams.size() > 1) {
Sebastian Jansson5fbebd52019-02-20 10:16:19177 vp8_settings.automaticResizeOn = false;
178 vp8_settings.denoisingOn = false;
179 } else {
180 vp8_settings.automaticResizeOn = config.encoder.single.automatic_scaling;
181 vp8_settings.denoisingOn = config.encoder.single.denoising;
182 }
Tommi87f70902021-04-27 12:43:08183 return rtc::make_ref_counted<VideoEncoderConfig::Vp8EncoderSpecificSettings>(
184 vp8_settings);
Sebastian Janssonf0c366b2019-02-14 12:18:42185}
186
187rtc::scoped_refptr<VideoEncoderConfig::EncoderSpecificSettings>
188CreateH264SpecificSettings(VideoStreamConfig config) {
Niels Möller4a5a3da2022-06-17 09:26:02189 RTC_DCHECK_EQ(config.encoder.simulcast_streams.size(), 1);
190 RTC_DCHECK(config.encoder.simulcast_streams[0] == ScalabilityMode::kL1T1);
Niels Möllercf2c8912022-05-18 08:45:46191 // TODO(bugs.webrtc.org/6883): Set a key frame interval as a setting that
192 // isn't codec specific.
193 RTC_CHECK_EQ(0, config.encoder.key_frame_interval.value_or(0));
194 return nullptr;
Sebastian Janssonf0c366b2019-02-14 12:18:42195}
196
197rtc::scoped_refptr<VideoEncoderConfig::EncoderSpecificSettings>
Sebastian Jansson98b07e912018-09-27 11:47:01198CreateEncoderSpecificSettings(VideoStreamConfig config) {
199 using Codec = VideoStreamConfig::Encoder::Codec;
200 switch (config.encoder.codec) {
Sebastian Janssonf0c366b2019-02-14 12:18:42201 case Codec::kVideoCodecH264:
202 return CreateH264SpecificSettings(config);
203 case Codec::kVideoCodecVP8:
204 return CreateVp8SpecificSettings(config);
205 case Codec::kVideoCodecVP9:
206 return CreateVp9SpecificSettings(config);
207 case Codec::kVideoCodecGeneric:
Danil Chapovalovdc368292019-11-26 13:48:20208 case Codec::kVideoCodecAV1:
qwu16ae82df72023-09-20 05:10:31209 case Codec::kVideoCodecH265:
Sebastian Janssonf0c366b2019-02-14 12:18:42210 return nullptr;
Sebastian Jansson98b07e912018-09-27 11:47:01211 }
212}
213
214VideoEncoderConfig CreateVideoEncoderConfig(VideoStreamConfig config) {
Jonas Oreland80c87d72022-09-29 13:01:09215 webrtc::VideoEncoder::EncoderInfo encoder_info;
Sebastian Jansson98b07e912018-09-27 11:47:01216 VideoEncoderConfig encoder_config;
217 encoder_config.codec_type = config.encoder.codec;
Sebastian Janssonf0c366b2019-02-14 12:18:42218 encoder_config.content_type = ConvertContentType(config.encoder.content_type);
Sebastian Jansson98b07e912018-09-27 11:47:01219 encoder_config.video_format =
220 SdpVideoFormat(CodecTypeToPayloadString(config.encoder.codec), {});
Sebastian Janssonf0c366b2019-02-14 12:18:42221
Niels Möller4a5a3da2022-06-17 09:26:02222 encoder_config.number_of_streams = config.encoder.simulcast_streams.size();
Sebastian Janssone8226682019-04-30 12:29:09223 encoder_config.simulcast_layers =
Niels Möller4a5a3da2022-06-17 09:26:02224 std::vector<VideoStream>(encoder_config.number_of_streams);
Sebastian Jansson985ee682018-11-12 15:33:29225 encoder_config.min_transmit_bitrate_bps = config.stream.pad_to_rate.bps();
Sebastian Jansson98b07e912018-09-27 11:47:01226
227 std::string cricket_codec = CodecTypeToCodecName(config.encoder.codec);
228 if (!cricket_codec.empty()) {
Sebastian Jansson5fbebd52019-02-20 10:16:19229 bool screenshare = config.encoder.content_type ==
230 VideoStreamConfig::Encoder::ContentType::kScreen;
Sebastian Jansson98b07e912018-09-27 11:47:01231 encoder_config.video_stream_factory =
Tommi87f70902021-04-27 12:43:08232 rtc::make_ref_counted<cricket::EncoderStreamFactory>(
Sergey Silkinb6ef1a72023-10-23 15:11:21233 cricket_codec, cricket::kDefaultVideoMaxQpVpx, screenshare,
234 screenshare, encoder_info);
Sebastian Jansson98b07e912018-09-27 11:47:01235 } else {
236 encoder_config.video_stream_factory =
Tommi87f70902021-04-27 12:43:08237 rtc::make_ref_counted<DefaultVideoStreamFactory>();
Sebastian Jansson98b07e912018-09-27 11:47:01238 }
Sebastian Janssonf0c366b2019-02-14 12:18:42239
240 // TODO(srte): Base this on encoder capabilities.
241 encoder_config.max_bitrate_bps =
Danil Chapovalovcad3e0e2020-02-17 17:46:07242 config.encoder.max_data_rate.value_or(DataRate::KilobitsPerSec(10000))
243 .bps();
Sebastian Janssonf0c366b2019-02-14 12:18:42244
Niels Möller807328f2022-05-12 14:16:39245 encoder_config.frame_drop_enabled = config.encoder.frame_dropping;
Sebastian Jansson98b07e912018-09-27 11:47:01246 encoder_config.encoder_specific_settings =
247 CreateEncoderSpecificSettings(config);
Niels Möller4a5a3da2022-06-17 09:26:02248
249 for (size_t i = 0; i < encoder_config.number_of_streams; ++i) {
250 auto& layer = encoder_config.simulcast_layers[i];
251 if (config.encoder.max_framerate) {
Sebastian Jansson0c32e332018-11-12 15:47:43252 layer.max_framerate = *config.encoder.max_framerate;
Erik Språng279f3702020-10-13 19:55:07253 layer.min_bitrate_bps = config.encoder.min_data_rate->bps_or(-1);
Sebastian Jansson0c32e332018-11-12 15:47:43254 }
Niels Möller4a5a3da2022-06-17 09:26:02255 layer.scalability_mode = config.encoder.simulcast_streams[i];
Sebastian Jansson0c32e332018-11-12 15:47:43256 }
257
Sebastian Jansson98b07e912018-09-27 11:47:01258 return encoder_config;
259}
Sebastian Janssonf0c366b2019-02-14 12:18:42260
Artem Titov33f9d2b2019-12-05 14:59:00261std::unique_ptr<FrameGeneratorInterface> CreateImageSlideGenerator(
Sebastian Janssonf2727fb2019-02-18 15:54:55262 Clock* clock,
263 VideoStreamConfig::Source::Slides slides,
264 int framerate) {
265 std::vector<std::string> paths = slides.images.paths;
266 for (std::string& path : paths)
267 path = TransformFilePath(path);
268 if (slides.images.crop.width || slides.images.crop.height) {
269 TimeDelta pause_duration =
270 slides.change_interval - slides.images.crop.scroll_duration;
Sebastian Jansson5a000162019-04-12 09:21:32271 RTC_CHECK_GE(pause_duration, TimeDelta::Zero());
Sebastian Janssonf2727fb2019-02-18 15:54:55272 int crop_width = slides.images.crop.width.value_or(slides.images.width);
273 int crop_height = slides.images.crop.height.value_or(slides.images.height);
274 RTC_CHECK_LE(crop_width, slides.images.width);
275 RTC_CHECK_LE(crop_height, slides.images.height);
Artem Titov33f9d2b2019-12-05 14:59:00276 return CreateScrollingInputFromYuvFilesFrameGenerator(
Sebastian Janssonf2727fb2019-02-18 15:54:55277 clock, paths, slides.images.width, slides.images.height, crop_width,
278 crop_height, slides.images.crop.scroll_duration.ms(),
279 pause_duration.ms());
280 } else {
Artem Titov33f9d2b2019-12-05 14:59:00281 return CreateFromYuvFileFrameGenerator(
Sebastian Janssonf2727fb2019-02-18 15:54:55282 paths, slides.images.width, slides.images.height,
283 slides.change_interval.seconds<double>() * framerate);
284 }
285}
286
Artem Titov33f9d2b2019-12-05 14:59:00287std::unique_ptr<FrameGeneratorInterface> CreateFrameGenerator(
Sebastian Janssonf0c366b2019-02-14 12:18:42288 Clock* clock,
289 VideoStreamConfig::Source source) {
290 using Capture = VideoStreamConfig::Source::Capture;
291 switch (source.capture) {
292 case Capture::kGenerator:
Artem Titov33f9d2b2019-12-05 14:59:00293 return CreateSquareFrameGenerator(
Sebastian Janssonf0c366b2019-02-14 12:18:42294 source.generator.width, source.generator.height,
295 source.generator.pixel_format, /*num_squares*/ absl::nullopt);
296 case Capture::kVideoFile:
297 RTC_CHECK(source.video_file.width && source.video_file.height);
Artem Titov33f9d2b2019-12-05 14:59:00298 return CreateFromYuvFileFrameGenerator(
Sebastian Janssond37307c2019-02-22 16:07:49299 {TransformFilePath(source.video_file.name)}, source.video_file.width,
Sebastian Janssonf0c366b2019-02-14 12:18:42300 source.video_file.height, /*frame_repeat_count*/ 1);
Sebastian Janssonf2727fb2019-02-18 15:54:55301 case Capture::kGenerateSlides:
Artem Titov33f9d2b2019-12-05 14:59:00302 return CreateSlideFrameGenerator(
Sebastian Janssonf2727fb2019-02-18 15:54:55303 source.slides.generator.width, source.slides.generator.height,
304 source.slides.change_interval.seconds<double>() * source.framerate);
305 case Capture::kImageSlides:
306 return CreateImageSlideGenerator(clock, source.slides, source.framerate);
Sebastian Janssonf0c366b2019-02-14 12:18:42307 }
308}
309
Tommif6f45432022-05-20 13:21:20310VideoReceiveStreamInterface::Config CreateVideoReceiveStreamConfig(
Sebastian Jansson5fbebd52019-02-20 10:16:19311 VideoStreamConfig config,
312 Transport* feedback_transport,
Philip Eliasson2b068ce2020-08-03 15:55:10313 VideoDecoderFactory* decoder_factory,
Tommif6f45432022-05-20 13:21:20314 VideoReceiveStreamInterface::Decoder decoder,
Sebastian Jansson5fbebd52019-02-20 10:16:19315 rtc::VideoSinkInterface<VideoFrame>* renderer,
316 uint32_t local_ssrc,
317 uint32_t ssrc,
318 uint32_t rtx_ssrc) {
Tommif6f45432022-05-20 13:21:20319 VideoReceiveStreamInterface::Config recv(feedback_transport);
Sebastian Jansson5fbebd52019-02-20 10:16:19320 recv.rtp.local_ssrc = local_ssrc;
Sebastian Jansson24cf2602019-06-07 13:24:57321
Sebastian Jansson5fbebd52019-02-20 10:16:19322 RTC_DCHECK(!config.stream.use_rtx ||
323 config.stream.nack_history_time > TimeDelta::Zero());
324 recv.rtp.nack.rtp_history_ms = config.stream.nack_history_time.ms();
325 recv.rtp.protected_by_flexfec = config.stream.use_flexfec;
326 recv.rtp.remote_ssrc = ssrc;
Philip Eliasson2b068ce2020-08-03 15:55:10327 recv.decoder_factory = decoder_factory;
Sebastian Jansson5fbebd52019-02-20 10:16:19328 recv.decoders.push_back(decoder);
329 recv.renderer = renderer;
330 if (config.stream.use_rtx) {
331 recv.rtp.rtx_ssrc = rtx_ssrc;
Artem Titov8a9f3a82023-04-25 07:56:49332 recv.rtp
333 .rtx_associated_payload_types[VideoTestConstants::kSendRtxPayloadType] =
Sebastian Jansson5fbebd52019-02-20 10:16:19334 CodecTypeToPayloadType(config.encoder.codec);
335 }
336 if (config.stream.use_ulpfec) {
Artem Titov8a9f3a82023-04-25 07:56:49337 recv.rtp.red_payload_type = VideoTestConstants::kRedPayloadType;
338 recv.rtp.ulpfec_payload_type = VideoTestConstants::kUlpfecPayloadType;
339 recv.rtp
340 .rtx_associated_payload_types[VideoTestConstants::kRtxRedPayloadType] =
341 VideoTestConstants::kRedPayloadType;
Sebastian Jansson5fbebd52019-02-20 10:16:19342 }
Sebastian Janssoncc5be542019-04-01 11:34:55343 recv.sync_group = config.render.sync_group;
Sebastian Jansson5fbebd52019-02-20 10:16:19344 return recv;
345}
Sebastian Jansson98b07e912018-09-27 11:47:01346} // namespace
347
Per Kb3046c22023-01-12 16:00:22348std::vector<RtpExtension> GetVideoRtpExtensions(
349 const VideoStreamConfig config) {
350 std::vector<RtpExtension> res = {
351 RtpExtension(RtpExtension::kVideoContentTypeUri,
352 kVideoContentTypeExtensionId),
353 RtpExtension(RtpExtension::kVideoRotationUri,
354 kVideoRotationRtpExtensionId)};
355 if (config.stream.packet_feedback) {
356 res.push_back(RtpExtension(RtpExtension::kTransportSequenceNumberUri,
357 kTransportSequenceNumberExtensionId));
358 }
359 if (config.stream.abs_send_time) {
360 res.push_back(
361 RtpExtension(RtpExtension::kAbsSendTimeUri, kAbsSendTimeExtensionId));
362 }
363 return res;
364}
365
Sebastian Jansson98b07e912018-09-27 11:47:01366SendVideoStream::SendVideoStream(CallClient* sender,
367 VideoStreamConfig config,
Sebastian Jansson9a4f38e2018-12-19 12:14:41368 Transport* send_transport,
Sebastian Janssoncf2df2f2019-04-02 09:51:28369 VideoFrameMatcher* matcher)
Sebastian Jansson98b07e912018-09-27 11:47:01370 : sender_(sender), config_(config) {
Mirko Bonadei317a1f02019-09-17 15:06:18371 video_capturer_ = std::make_unique<FrameGeneratorCapturer>(
Danil Chapovalov59d0b8d2023-11-29 18:05:54372 &sender_->env_.clock(),
373 CreateFrameGenerator(&sender_->env_.clock(), config.source),
374 config.source.framerate, sender_->env_.task_queue_factory());
Sebastian Janssonf0c366b2019-02-14 12:18:42375 video_capturer_->Init();
Sebastian Jansson98b07e912018-09-27 11:47:01376
377 using Encoder = VideoStreamConfig::Encoder;
378 using Codec = VideoStreamConfig::Encoder::Codec;
379 switch (config.encoder.implementation) {
380 case Encoder::Implementation::kFake:
Danil Chapovalov41b4bf972024-04-11 16:10:31381 encoder_factory_ = std::make_unique<FunctionVideoEncoderFactory>(
382 [this](const Environment& env, const SdpVideoFormat& format) {
Markus Handella5a4be12020-07-08 14:09:21383 MutexLock lock(&mutex_);
Jonas Olssona4d87372019-07-05 17:08:33384 std::unique_ptr<FakeEncoder> encoder;
385 if (config_.encoder.codec == Codec::kVideoCodecVP8) {
Danil Chapovalov41b4bf972024-04-11 16:10:31386 encoder = std::make_unique<test::FakeVp8Encoder>(env);
Jonas Olssona4d87372019-07-05 17:08:33387 } else if (config_.encoder.codec == Codec::kVideoCodecGeneric) {
Danil Chapovalov41b4bf972024-04-11 16:10:31388 encoder = std::make_unique<test::FakeEncoder>(env);
Jonas Olssona4d87372019-07-05 17:08:33389 } else {
Artem Titovd3251962021-11-15 15:57:07390 RTC_DCHECK_NOTREACHED();
Jonas Olssona4d87372019-07-05 17:08:33391 }
392 fake_encoders_.push_back(encoder.get());
393 if (config_.encoder.fake.max_rate.IsFinite())
394 encoder->SetMaxBitrate(config_.encoder.fake.max_rate.kbps());
395 return encoder;
396 });
Sebastian Jansson98b07e912018-09-27 11:47:01397 break;
398 case VideoStreamConfig::Encoder::Implementation::kSoftware:
399 encoder_factory_.reset(new InternalEncoderFactory());
400 break;
401 case VideoStreamConfig::Encoder::Implementation::kHardware:
402 encoder_factory_ = CreateHardwareEncoderFactory();
403 break;
404 }
405 RTC_CHECK(encoder_factory_);
406
Jiawei Ouc2ebe212018-11-08 18:02:56407 bitrate_allocator_factory_ = CreateBuiltinVideoBitrateAllocatorFactory();
408 RTC_CHECK(bitrate_allocator_factory_);
409
Sebastian Janssonf0c366b2019-02-14 12:18:42410 VideoEncoderConfig encoder_config = CreateVideoEncoderConfig(config);
411 for (size_t i = 0; i < encoder_config.number_of_streams; ++i) {
412 ssrcs_.push_back(sender->GetNextVideoSsrc());
413 rtx_ssrcs_.push_back(sender->GetNextRtxSsrc());
414 }
Sebastian Jansson98b07e912018-09-27 11:47:01415 VideoSendStream::Config send_config =
Sebastian Jansson3e66a492020-01-14 11:30:13416 CreateVideoSendStreamConfig(config, ssrcs_, rtx_ssrcs_, send_transport);
Sebastian Jansson98b07e912018-09-27 11:47:01417 send_config.encoder_settings.encoder_factory = encoder_factory_.get();
Jiawei Ouc2ebe212018-11-08 18:02:56418 send_config.encoder_settings.bitrate_allocator_factory =
419 bitrate_allocator_factory_.get();
Erik Språng279f3702020-10-13 19:55:07420 send_config.suspend_below_min_bitrate =
421 config.encoder.suspend_below_min_bitrate;
Jiawei Ouc2ebe212018-11-08 18:02:56422
Artem Titovf92cc6d2023-06-29 12:56:59423 video_capturer_->Start();
Sebastian Jansson105a10a2019-04-01 07:18:14424 sender_->SendTask([&] {
Sebastian Janssoncc5be542019-04-01 11:34:55425 if (config.stream.fec_controller_factory) {
426 send_stream_ = sender_->call_->CreateVideoSendStream(
427 std::move(send_config), std::move(encoder_config),
Danil Chapovalov55a61892024-01-03 17:39:27428 config.stream.fec_controller_factory->CreateFecController(
429 sender_->env_));
Sebastian Janssoncc5be542019-04-01 11:34:55430 } else {
431 send_stream_ = sender_->call_->CreateVideoSendStream(
432 std::move(send_config), std::move(encoder_config));
433 }
Sebastian Jansson98b07e912018-09-27 11:47:01434
Sebastian Janssoncf2df2f2019-04-02 09:51:28435 if (matcher->Active()) {
Mirko Bonadei317a1f02019-09-17 15:06:18436 frame_tap_ = std::make_unique<ForwardingCapturedFrameTap>(
Danil Chapovalov59d0b8d2023-11-29 18:05:54437 &sender_->env_.clock(), matcher, video_capturer_.get());
Sebastian Jansson105a10a2019-04-01 07:18:14438 send_stream_->SetSource(frame_tap_.get(),
439 config.encoder.degradation_preference);
440 } else {
441 send_stream_->SetSource(video_capturer_.get(),
442 config.encoder.degradation_preference);
443 }
444 });
Sebastian Jansson98b07e912018-09-27 11:47:01445}
446
447SendVideoStream::~SendVideoStream() {
Sebastian Jansson105a10a2019-04-01 07:18:14448 sender_->SendTask(
449 [this] { sender_->call_->DestroyVideoSendStream(send_stream_); });
Sebastian Jansson98b07e912018-09-27 11:47:01450}
451
452void SendVideoStream::Start() {
Sebastian Jansson105a10a2019-04-01 07:18:14453 sender_->SendTask([this] {
454 send_stream_->Start();
455 sender_->call_->SignalChannelNetworkState(MediaType::VIDEO, kNetworkUp);
456 });
Sebastian Jansson98b07e912018-09-27 11:47:01457}
458
Christoffer Rodbro5f6abcf2019-02-08 10:00:10459void SendVideoStream::Stop() {
Sebastian Jansson105a10a2019-04-01 07:18:14460 sender_->SendTask([this] { send_stream_->Stop(); });
Christoffer Rodbro5f6abcf2019-02-08 10:00:10461}
462
Sebastian Jansson0c32e332018-11-12 15:47:43463void SendVideoStream::UpdateConfig(
464 std::function<void(VideoStreamConfig*)> modifier) {
Sebastian Jansson105a10a2019-04-01 07:18:14465 sender_->SendTask([&] {
Markus Handella5a4be12020-07-08 14:09:21466 MutexLock lock(&mutex_);
Sebastian Jansson105a10a2019-04-01 07:18:14467 VideoStreamConfig prior_config = config_;
468 modifier(&config_);
469 if (prior_config.encoder.fake.max_rate != config_.encoder.fake.max_rate) {
470 for (auto* encoder : fake_encoders_) {
471 encoder->SetMaxBitrate(config_.encoder.fake.max_rate.kbps());
472 }
Sebastian Jansson0c32e332018-11-12 15:47:43473 }
Sebastian Jansson105a10a2019-04-01 07:18:14474 // TODO(srte): Add more conditions that should cause reconfiguration.
Erik Språng3e3e1662020-10-06 19:51:21475 if (prior_config.encoder.max_framerate != config_.encoder.max_framerate ||
476 prior_config.encoder.max_data_rate != config_.encoder.max_data_rate) {
Sebastian Jansson105a10a2019-04-01 07:18:14477 VideoEncoderConfig encoder_config = CreateVideoEncoderConfig(config_);
478 send_stream_->ReconfigureVideoEncoder(std::move(encoder_config));
479 }
480 if (prior_config.source.framerate != config_.source.framerate) {
481 SetCaptureFramerate(config_.source.framerate);
482 }
483 });
Sebastian Jansson0c32e332018-11-12 15:47:43484}
485
Sebastian Janssone8226682019-04-30 12:29:09486void SendVideoStream::UpdateActiveLayers(std::vector<bool> active_layers) {
487 sender_->task_queue_.PostTask([=] {
Markus Handella5a4be12020-07-08 14:09:21488 MutexLock lock(&mutex_);
Erik Språng3e3e1662020-10-06 19:51:21489 VideoEncoderConfig encoder_config = CreateVideoEncoderConfig(config_);
490 RTC_CHECK_EQ(encoder_config.simulcast_layers.size(), active_layers.size());
491 for (size_t i = 0; i < encoder_config.simulcast_layers.size(); ++i)
492 encoder_config.simulcast_layers[i].active = active_layers[i];
493 send_stream_->ReconfigureVideoEncoder(std::move(encoder_config));
Sebastian Janssone8226682019-04-30 12:29:09494 });
495}
496
Sebastian Jansson3d4d94a2020-01-14 13:25:41497bool SendVideoStream::UsingSsrc(uint32_t ssrc) const {
498 for (uint32_t owned : ssrcs_) {
499 if (owned == ssrc)
500 return true;
501 }
502 return false;
503}
504
505bool SendVideoStream::UsingRtxSsrc(uint32_t ssrc) const {
506 for (uint32_t owned : rtx_ssrcs_) {
507 if (owned == ssrc)
508 return true;
509 }
510 return false;
511}
512
Sebastian Jansson98b07e912018-09-27 11:47:01513void SendVideoStream::SetCaptureFramerate(int framerate) {
Sebastian Jansson105a10a2019-04-01 07:18:14514 sender_->SendTask([&] { video_capturer_->ChangeFramerate(framerate); });
Sebastian Jansson98b07e912018-09-27 11:47:01515}
516
517VideoSendStream::Stats SendVideoStream::GetStats() const {
518 return send_stream_->GetStats();
519}
520
521ColumnPrinter SendVideoStream::StatsPrinter() {
522 return ColumnPrinter::Lambda(
523 "video_target_rate video_sent_rate width height",
524 [this](rtc::SimpleStringBuilder& sb) {
525 VideoSendStream::Stats video_stats = send_stream_->GetStats();
526 int width = 0;
527 int height = 0;
Mirko Bonadei739baf02019-01-27 16:29:42528 for (const auto& stream_stat : video_stats.substreams) {
Sebastian Jansson98b07e912018-09-27 11:47:01529 width = std::max(width, stream_stat.second.width);
530 height = std::max(height, stream_stat.second.height);
531 }
532 sb.AppendFormat("%.0lf %.0lf %i %i",
533 video_stats.target_media_bitrate_bps / 8.0,
534 video_stats.media_bitrate_bps / 8.0, width, height);
535 },
536 64);
537}
538
539ReceiveVideoStream::ReceiveVideoStream(CallClient* receiver,
540 VideoStreamConfig config,
541 SendVideoStream* send_stream,
542 size_t chosen_stream,
Sebastian Jansson9a4f38e2018-12-19 12:14:41543 Transport* feedback_transport,
Sebastian Janssoncf2df2f2019-04-02 09:51:28544 VideoFrameMatcher* matcher)
Sebastian Jansson0c32e332018-11-12 15:47:43545 : receiver_(receiver), config_(config) {
Sebastian Jansson0c32e332018-11-12 15:47:43546 if (config.encoder.codec ==
Sebastian Jansson3d4d94a2020-01-14 13:25:41547 VideoStreamConfig::Encoder::Codec::kVideoCodecGeneric ||
548 config.encoder.implementation == VideoStreamConfig::Encoder::kFake) {
Mirko Bonadei317a1f02019-09-17 15:06:18549 decoder_factory_ = std::make_unique<FunctionVideoDecoderFactory>(
550 []() { return std::make_unique<FakeDecoder>(); });
Sebastian Jansson0c32e332018-11-12 15:47:43551 } else {
Mirko Bonadei317a1f02019-09-17 15:06:18552 decoder_factory_ = std::make_unique<InternalDecoderFactory>();
Sebastian Jansson0c32e332018-11-12 15:47:43553 }
Sebastian Jansson98b07e912018-09-27 11:47:01554
Tommif6f45432022-05-20 13:21:20555 VideoReceiveStreamInterface::Decoder decoder =
Sebastian Jansson5fbebd52019-02-20 10:16:19556 CreateMatchingDecoder(CodecTypeToPayloadType(config.encoder.codec),
557 CodecTypeToPayloadString(config.encoder.codec));
Niels Möller4a5a3da2022-06-17 09:26:02558 size_t num_streams = config.encoder.simulcast_streams.size();
Sebastian Jansson5fbebd52019-02-20 10:16:19559 for (size_t i = 0; i < num_streams; ++i) {
560 rtc::VideoSinkInterface<VideoFrame>* renderer = &fake_renderer_;
Sebastian Janssoncf2df2f2019-04-02 09:51:28561 if (matcher->Active()) {
Danil Chapovalov59d0b8d2023-11-29 18:05:54562 render_taps_.emplace_back(std::make_unique<DecodedFrameTap>(
563 &receiver_->env_.clock(), matcher, i));
Sebastian Janssoncf2df2f2019-04-02 09:51:28564 renderer = render_taps_.back().get();
Sebastian Jansson5fbebd52019-02-20 10:16:19565 }
566 auto recv_config = CreateVideoReceiveStreamConfig(
Philip Eliasson2b068ce2020-08-03 15:55:10567 config, feedback_transport, decoder_factory_.get(), decoder, renderer,
Sebastian Jansson5fbebd52019-02-20 10:16:19568 receiver_->GetNextVideoLocalSsrc(), send_stream->ssrcs_[i],
569 send_stream->rtx_ssrcs_[i]);
570 if (config.stream.use_flexfec) {
571 RTC_DCHECK(num_streams == 1);
572 FlexfecReceiveStream::Config flexfec(feedback_transport);
Artem Titov8a9f3a82023-04-25 07:56:49573 flexfec.payload_type = VideoTestConstants::kFlexfecPayloadType;
574 flexfec.rtp.remote_ssrc = VideoTestConstants::kFlexfecSendSsrc;
Sebastian Jansson5fbebd52019-02-20 10:16:19575 flexfec.protected_media_ssrcs = send_stream->rtx_ssrcs_;
Tommi1c1f5402021-06-14 08:54:20576 flexfec.rtp.local_ssrc = recv_config.rtp.local_ssrc;
577 receiver_->ssrc_media_types_[flexfec.rtp.remote_ssrc] = MediaType::VIDEO;
Sebastian Jansson342f98b2019-06-18 14:08:23578
579 receiver_->SendTask([this, &flexfec] {
580 flecfec_stream_ = receiver_->call_->CreateFlexfecReceiveStream(flexfec);
581 });
Sebastian Jansson5fbebd52019-02-20 10:16:19582 }
583 receiver_->ssrc_media_types_[recv_config.rtp.remote_ssrc] =
584 MediaType::VIDEO;
585 if (config.stream.use_rtx)
586 receiver_->ssrc_media_types_[recv_config.rtp.rtx_ssrc] = MediaType::VIDEO;
Sebastian Jansson105a10a2019-04-01 07:18:14587 receiver_->SendTask([this, &recv_config] {
588 receive_streams_.push_back(
589 receiver_->call_->CreateVideoReceiveStream(std::move(recv_config)));
590 });
Sebastian Jansson98b07e912018-09-27 11:47:01591 }
Sebastian Jansson98b07e912018-09-27 11:47:01592}
593
594ReceiveVideoStream::~ReceiveVideoStream() {
Sebastian Jansson105a10a2019-04-01 07:18:14595 receiver_->SendTask([this] {
596 for (auto* recv_stream : receive_streams_)
597 receiver_->call_->DestroyVideoReceiveStream(recv_stream);
598 if (flecfec_stream_)
599 receiver_->call_->DestroyFlexfecReceiveStream(flecfec_stream_);
600 });
Sebastian Jansson98b07e912018-09-27 11:47:01601}
602
Sebastian Jansson49a78432018-11-20 15:15:29603void ReceiveVideoStream::Start() {
Sebastian Jansson105a10a2019-04-01 07:18:14604 receiver_->SendTask([this] {
605 for (auto* recv_stream : receive_streams_)
606 recv_stream->Start();
607 receiver_->call_->SignalChannelNetworkState(MediaType::VIDEO, kNetworkUp);
608 });
Sebastian Jansson49a78432018-11-20 15:15:29609}
610
Sebastian Janssonbdfadd62019-02-08 12:34:57611void ReceiveVideoStream::Stop() {
Sebastian Jansson105a10a2019-04-01 07:18:14612 receiver_->SendTask([this] {
613 for (auto* recv_stream : receive_streams_)
614 recv_stream->Stop();
615 });
Sebastian Janssonbdfadd62019-02-08 12:34:57616}
617
Tommif6f45432022-05-20 13:21:20618VideoReceiveStreamInterface::Stats ReceiveVideoStream::GetStats() const {
Sebastian Janssonf4481c82019-04-09 10:48:34619 if (receive_streams_.empty())
Tommif6f45432022-05-20 13:21:20620 return VideoReceiveStreamInterface::Stats();
Sebastian Janssonf4481c82019-04-09 10:48:34621 // TODO(srte): Handle multiple receive streams.
Sebastian Jansson8ad34272020-02-27 11:54:16622 return receive_streams_.back()->GetStats();
Sebastian Janssonf4481c82019-04-09 10:48:34623}
624
Sebastian Jansson98b07e912018-09-27 11:47:01625VideoStreamPair::~VideoStreamPair() = default;
626
Sebastian Janssoncf2df2f2019-04-02 09:51:28627VideoStreamPair::VideoStreamPair(CallClient* sender,
628 CallClient* receiver,
629 VideoStreamConfig config)
Sebastian Jansson98b07e912018-09-27 11:47:01630 : config_(config),
Sebastian Janssoncf2df2f2019-04-02 09:51:28631 matcher_(config.hooks.frame_pair_handlers),
632 send_stream_(sender, config, sender->transport_.get(), &matcher_),
Sebastian Jansson98b07e912018-09-27 11:47:01633 receive_stream_(receiver,
634 config,
635 &send_stream_,
636 /*chosen_stream=*/0,
Sebastian Jansson105a10a2019-04-01 07:18:14637 receiver->transport_.get(),
Sebastian Janssoncf2df2f2019-04-02 09:51:28638 &matcher_) {}
Sebastian Jansson98b07e912018-09-27 11:47:01639
640} // namespace test
641} // namespace webrtc