
This reverts commit 389b1672a32f2dd49af6c6ed40e8ddf394b986de. Reason for revert: Causes failure (and empty result list) in CallPerfTest.PadsToMinTransmitBitrate Original change's description: > Delete test/constants.h > > It's not possible to use constants.h for all RTP extensions > after the number of extensions exceeds 14, which is the maximum > number of one-byte RTP extensions. This is because some extensions > would have to be assigned a number greater than 14, even if the > test only involves 14 extensions or less. > > For uniformity's sake, this CL also edits some files to use an > enum as the files involved in this CL, rather than free-floating > const-ints. > > Bug: webrtc:10288 > Change-Id: Ib5e58ad72c4d3756f4c4f6521f140ec59617f3f5 > Reviewed-on: https://webrtc-review.googlesource.com/c/123048 > Commit-Queue: Elad Alon <eladalon@webrtc.org> > Reviewed-by: Danil Chapovalov <danilchap@webrtc.org> > Reviewed-by: Karl Wiberg <kwiberg@webrtc.org> > Reviewed-by: Erik Språng <sprang@webrtc.org> > Cr-Commit-Position: refs/heads/master@{#26728} TBR=danilchap@webrtc.org,kwiberg@webrtc.org,eladalon@webrtc.org,sprang@webrtc.org No-Presubmit: True Bug: webrtc:10288, chromium:933127 Change-Id: If1de0bd8992137c52bf0b877b3cb0a2bafc809d4 Reviewed-on: https://webrtc-review.googlesource.com/c/123381 Commit-Queue: Oleh Prypin <oprypin@webrtc.org> Reviewed-by: Oleh Prypin <oprypin@webrtc.org> Cr-Commit-Position: refs/heads/master@{#26744}
520 lines
20 KiB
C++
520 lines
20 KiB
C++
/*
|
|
* Copyright 2018 The WebRTC project authors. All Rights Reserved.
|
|
*
|
|
* Use of this source code is governed by a BSD-style license
|
|
* that can be found in the LICENSE file in the root of the source
|
|
* tree. An additional intellectual property rights grant can be found
|
|
* in the file PATENTS. All contributing project authors may
|
|
* be found in the AUTHORS file in the root of the source tree.
|
|
*/
|
|
#include "test/scenario/video_stream.h"
|
|
|
|
#include <algorithm>
|
|
#include <utility>
|
|
|
|
#include "absl/memory/memory.h"
|
|
#include "api/test/video/function_video_encoder_factory.h"
|
|
#include "api/video/builtin_video_bitrate_allocator_factory.h"
|
|
#include "media/base/media_constants.h"
|
|
#include "media/engine/internal_decoder_factory.h"
|
|
#include "media/engine/internal_encoder_factory.h"
|
|
#include "media/engine/webrtc_video_engine.h"
|
|
#include "test/call_test.h"
|
|
#include "test/fake_encoder.h"
|
|
#include "test/scenario/hardware_codecs.h"
|
|
#include "test/testsupport/file_utils.h"
|
|
|
|
namespace webrtc {
|
|
namespace test {
|
|
namespace {
|
|
constexpr int kDefaultMaxQp = cricket::WebRtcVideoChannel::kDefaultQpMax;
|
|
const int kVideoRotationRtpExtensionId = 4;
|
|
uint8_t CodecTypeToPayloadType(VideoCodecType codec_type) {
|
|
switch (codec_type) {
|
|
case VideoCodecType::kVideoCodecGeneric:
|
|
return CallTest::kFakeVideoSendPayloadType;
|
|
case VideoCodecType::kVideoCodecVP8:
|
|
return CallTest::kPayloadTypeVP8;
|
|
case VideoCodecType::kVideoCodecVP9:
|
|
return CallTest::kPayloadTypeVP9;
|
|
case VideoCodecType::kVideoCodecH264:
|
|
return CallTest::kPayloadTypeH264;
|
|
default:
|
|
RTC_NOTREACHED();
|
|
}
|
|
return {};
|
|
}
|
|
std::string CodecTypeToCodecName(VideoCodecType codec_type) {
|
|
switch (codec_type) {
|
|
case VideoCodecType::kVideoCodecGeneric:
|
|
return "";
|
|
case VideoCodecType::kVideoCodecVP8:
|
|
return cricket::kVp8CodecName;
|
|
case VideoCodecType::kVideoCodecVP9:
|
|
return cricket::kVp9CodecName;
|
|
case VideoCodecType::kVideoCodecH264:
|
|
return cricket::kH264CodecName;
|
|
default:
|
|
RTC_NOTREACHED();
|
|
}
|
|
return {};
|
|
}
|
|
VideoEncoderConfig::ContentType ConvertContentType(
|
|
VideoStreamConfig::Encoder::ContentType content_type) {
|
|
switch (content_type) {
|
|
case VideoStreamConfig::Encoder::ContentType::kVideo:
|
|
return VideoEncoderConfig::ContentType::kRealtimeVideo;
|
|
break;
|
|
case VideoStreamConfig::Encoder::ContentType::kScreen:
|
|
return VideoEncoderConfig::ContentType::kScreen;
|
|
}
|
|
}
|
|
InterLayerPredMode ToInterLayerPredMode(
|
|
VideoStreamConfig::Encoder::Layers::Prediction value) {
|
|
using Pred = VideoStreamConfig::Encoder::Layers::Prediction;
|
|
switch (value) {
|
|
case Pred::kTemporalOnly:
|
|
return InterLayerPredMode::kOff;
|
|
case Pred::kSpatialOnKey:
|
|
return InterLayerPredMode::kOnKeyPic;
|
|
case Pred::kFull:
|
|
return InterLayerPredMode::kOn;
|
|
}
|
|
}
|
|
std::vector<RtpExtension> GetVideoRtpExtensions(
|
|
const VideoStreamConfig config) {
|
|
return {RtpExtension(RtpExtension::kTransportSequenceNumberUri,
|
|
kTransportSequenceNumberExtensionId),
|
|
RtpExtension(RtpExtension::kVideoContentTypeUri,
|
|
kVideoContentTypeExtensionId),
|
|
RtpExtension(RtpExtension::kVideoRotationUri,
|
|
kVideoRotationRtpExtensionId)};
|
|
}
|
|
|
|
std::string TransformFilePath(std::string path) {
|
|
static const std::string resource_prefix = "res://";
|
|
int ext_pos = path.rfind(".");
|
|
if (ext_pos < 0) {
|
|
return test::ResourcePath(path, "yuv");
|
|
} else if (path.find(resource_prefix) == 0) {
|
|
std::string name = path.substr(resource_prefix.length(), ext_pos);
|
|
std::string ext = path.substr(ext_pos, path.size());
|
|
return test::ResourcePath(name, ext);
|
|
}
|
|
return path;
|
|
}
|
|
|
|
VideoSendStream::Config CreateVideoSendStreamConfig(VideoStreamConfig config,
|
|
std::vector<uint32_t> ssrcs,
|
|
Transport* send_transport) {
|
|
VideoSendStream::Config send_config(send_transport);
|
|
send_config.rtp.payload_name = CodecTypeToPayloadString(config.encoder.codec);
|
|
send_config.rtp.payload_type = CodecTypeToPayloadType(config.encoder.codec);
|
|
|
|
send_config.rtp.ssrcs = ssrcs;
|
|
send_config.rtp.extensions = GetVideoRtpExtensions(config);
|
|
|
|
if (config.stream.use_flexfec) {
|
|
send_config.rtp.flexfec.payload_type = CallTest::kFlexfecPayloadType;
|
|
send_config.rtp.flexfec.ssrc = CallTest::kFlexfecSendSsrc;
|
|
send_config.rtp.flexfec.protected_media_ssrcs = ssrcs;
|
|
}
|
|
if (config.stream.use_ulpfec) {
|
|
send_config.rtp.ulpfec.red_payload_type = CallTest::kRedPayloadType;
|
|
send_config.rtp.ulpfec.ulpfec_payload_type = CallTest::kUlpfecPayloadType;
|
|
send_config.rtp.ulpfec.red_rtx_payload_type = CallTest::kRtxRedPayloadType;
|
|
}
|
|
return send_config;
|
|
}
|
|
rtc::scoped_refptr<VideoEncoderConfig::EncoderSpecificSettings>
|
|
CreateVp9SpecificSettings(VideoStreamConfig video_config) {
|
|
constexpr auto kScreen = VideoStreamConfig::Encoder::ContentType::kScreen;
|
|
VideoStreamConfig::Encoder conf = video_config.encoder;
|
|
VideoCodecVP9 vp9 = VideoEncoder::GetDefaultVp9Settings();
|
|
vp9.frameDroppingOn = conf.frame_dropping;
|
|
vp9.keyFrameInterval = conf.key_frame_interval.value_or(0);
|
|
vp9.numberOfTemporalLayers = static_cast<uint8_t>(conf.layers.temporal);
|
|
vp9.numberOfSpatialLayers = static_cast<uint8_t>(conf.layers.spatial);
|
|
vp9.interLayerPred = ToInterLayerPredMode(conf.layers.prediction);
|
|
|
|
if (conf.content_type == kScreen &&
|
|
(video_config.source.framerate > 5 || conf.layers.spatial >= 3)) {
|
|
vp9.flexibleMode = true;
|
|
}
|
|
|
|
if (conf.content_type == kScreen ||
|
|
conf.layers.temporal * conf.layers.spatial) {
|
|
vp9.automaticResizeOn = false;
|
|
vp9.denoisingOn = false;
|
|
} else {
|
|
vp9.automaticResizeOn = conf.single.automatic_scaling;
|
|
vp9.denoisingOn = conf.single.denoising;
|
|
}
|
|
return new rtc::RefCountedObject<
|
|
VideoEncoderConfig::Vp9EncoderSpecificSettings>(vp9);
|
|
}
|
|
|
|
rtc::scoped_refptr<VideoEncoderConfig::EncoderSpecificSettings>
|
|
CreateVp8SpecificSettings(VideoStreamConfig config) {
|
|
RTC_DCHECK_EQ(config.encoder.layers.temporal, 1);
|
|
RTC_DCHECK_EQ(config.encoder.layers.spatial, 1);
|
|
VideoCodecVP8 vp8_settings = VideoEncoder::GetDefaultVp8Settings();
|
|
vp8_settings.frameDroppingOn = config.encoder.frame_dropping;
|
|
vp8_settings.keyFrameInterval = config.encoder.key_frame_interval.value_or(0);
|
|
vp8_settings.automaticResizeOn = config.encoder.single.automatic_scaling;
|
|
vp8_settings.denoisingOn = config.encoder.single.denoising;
|
|
return new rtc::RefCountedObject<
|
|
VideoEncoderConfig::Vp8EncoderSpecificSettings>(vp8_settings);
|
|
}
|
|
|
|
rtc::scoped_refptr<VideoEncoderConfig::EncoderSpecificSettings>
|
|
CreateH264SpecificSettings(VideoStreamConfig config) {
|
|
RTC_DCHECK_EQ(config.encoder.layers.temporal, 1);
|
|
RTC_DCHECK_EQ(config.encoder.layers.spatial, 1);
|
|
|
|
VideoCodecH264 h264_settings = VideoEncoder::GetDefaultH264Settings();
|
|
h264_settings.frameDroppingOn = config.encoder.frame_dropping;
|
|
h264_settings.keyFrameInterval =
|
|
config.encoder.key_frame_interval.value_or(0);
|
|
return new rtc::RefCountedObject<
|
|
VideoEncoderConfig::H264EncoderSpecificSettings>(h264_settings);
|
|
}
|
|
|
|
rtc::scoped_refptr<VideoEncoderConfig::EncoderSpecificSettings>
|
|
CreateEncoderSpecificSettings(VideoStreamConfig config) {
|
|
using Codec = VideoStreamConfig::Encoder::Codec;
|
|
switch (config.encoder.codec) {
|
|
case Codec::kVideoCodecH264:
|
|
return CreateH264SpecificSettings(config);
|
|
case Codec::kVideoCodecVP8:
|
|
return CreateVp8SpecificSettings(config);
|
|
case Codec::kVideoCodecVP9:
|
|
return CreateVp9SpecificSettings(config);
|
|
case Codec::kVideoCodecGeneric:
|
|
return nullptr;
|
|
case Codec::kVideoCodecMultiplex:
|
|
RTC_NOTREACHED();
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
VideoEncoderConfig CreateVideoEncoderConfig(VideoStreamConfig config) {
|
|
VideoEncoderConfig encoder_config;
|
|
encoder_config.codec_type = config.encoder.codec;
|
|
encoder_config.content_type = ConvertContentType(config.encoder.content_type);
|
|
encoder_config.video_format =
|
|
SdpVideoFormat(CodecTypeToPayloadString(config.encoder.codec), {});
|
|
|
|
// TODO(srte): Replace with actual value when supported.
|
|
size_t num_streams = 1;
|
|
encoder_config.number_of_streams = num_streams;
|
|
encoder_config.simulcast_layers = std::vector<VideoStream>(num_streams);
|
|
encoder_config.min_transmit_bitrate_bps = config.stream.pad_to_rate.bps();
|
|
|
|
std::string cricket_codec = CodecTypeToCodecName(config.encoder.codec);
|
|
if (!cricket_codec.empty()) {
|
|
encoder_config.video_stream_factory =
|
|
new rtc::RefCountedObject<cricket::EncoderStreamFactory>(
|
|
cricket_codec, kDefaultMaxQp, false, false);
|
|
} else {
|
|
encoder_config.video_stream_factory =
|
|
new rtc::RefCountedObject<DefaultVideoStreamFactory>();
|
|
}
|
|
|
|
// TODO(srte): Base this on encoder capabilities.
|
|
encoder_config.max_bitrate_bps =
|
|
config.encoder.max_data_rate.value_or(DataRate::kbps(10000)).bps();
|
|
|
|
encoder_config.encoder_specific_settings =
|
|
CreateEncoderSpecificSettings(config);
|
|
if (config.encoder.max_framerate) {
|
|
for (auto& layer : encoder_config.simulcast_layers) {
|
|
layer.max_framerate = *config.encoder.max_framerate;
|
|
}
|
|
}
|
|
|
|
return encoder_config;
|
|
}
|
|
|
|
std::unique_ptr<FrameGenerator> CreateImageSlideGenerator(
|
|
Clock* clock,
|
|
VideoStreamConfig::Source::Slides slides,
|
|
int framerate) {
|
|
std::vector<std::string> paths = slides.images.paths;
|
|
for (std::string& path : paths)
|
|
path = TransformFilePath(path);
|
|
if (slides.images.crop.width || slides.images.crop.height) {
|
|
TimeDelta pause_duration =
|
|
slides.change_interval - slides.images.crop.scroll_duration;
|
|
RTC_CHECK(pause_duration >= TimeDelta::Zero());
|
|
int crop_width = slides.images.crop.width.value_or(slides.images.width);
|
|
int crop_height = slides.images.crop.height.value_or(slides.images.height);
|
|
RTC_CHECK_LE(crop_width, slides.images.width);
|
|
RTC_CHECK_LE(crop_height, slides.images.height);
|
|
return FrameGenerator::CreateScrollingInputFromYuvFiles(
|
|
clock, paths, slides.images.width, slides.images.height, crop_width,
|
|
crop_height, slides.images.crop.scroll_duration.ms(),
|
|
pause_duration.ms());
|
|
} else {
|
|
return FrameGenerator::CreateFromYuvFile(
|
|
paths, slides.images.width, slides.images.height,
|
|
slides.change_interval.seconds<double>() * framerate);
|
|
}
|
|
}
|
|
|
|
std::unique_ptr<FrameGenerator> CreateFrameGenerator(
|
|
Clock* clock,
|
|
VideoStreamConfig::Source source) {
|
|
using Capture = VideoStreamConfig::Source::Capture;
|
|
switch (source.capture) {
|
|
case Capture::kGenerator:
|
|
return FrameGenerator::CreateSquareGenerator(
|
|
source.generator.width, source.generator.height,
|
|
source.generator.pixel_format, /*num_squares*/ absl::nullopt);
|
|
case Capture::kVideoFile:
|
|
RTC_CHECK(source.video_file.width && source.video_file.height);
|
|
return FrameGenerator::CreateFromYuvFile(
|
|
{source.video_file.name}, source.video_file.width,
|
|
source.video_file.height, /*frame_repeat_count*/ 1);
|
|
case Capture::kGenerateSlides:
|
|
return FrameGenerator::CreateSlideGenerator(
|
|
source.slides.generator.width, source.slides.generator.height,
|
|
source.slides.change_interval.seconds<double>() * source.framerate);
|
|
case Capture::kImageSlides:
|
|
return CreateImageSlideGenerator(clock, source.slides, source.framerate);
|
|
}
|
|
}
|
|
|
|
} // namespace
|
|
|
|
SendVideoStream::SendVideoStream(CallClient* sender,
|
|
VideoStreamConfig config,
|
|
Transport* send_transport,
|
|
VideoQualityAnalyzer* analyzer)
|
|
: sender_(sender), config_(config) {
|
|
video_capturer_ = absl::make_unique<FrameGeneratorCapturer>(
|
|
sender_->clock_, CreateFrameGenerator(sender_->clock_, config.source),
|
|
config.source.framerate);
|
|
video_capturer_->Init();
|
|
|
|
using Encoder = VideoStreamConfig::Encoder;
|
|
using Codec = VideoStreamConfig::Encoder::Codec;
|
|
switch (config.encoder.implementation) {
|
|
case Encoder::Implementation::kFake:
|
|
if (config.encoder.codec == Codec::kVideoCodecGeneric) {
|
|
encoder_factory_ =
|
|
absl::make_unique<FunctionVideoEncoderFactory>([this]() {
|
|
rtc::CritScope cs(&crit_);
|
|
auto encoder =
|
|
absl::make_unique<test::FakeEncoder>(sender_->clock_);
|
|
fake_encoders_.push_back(encoder.get());
|
|
if (config_.encoder.fake.max_rate.IsFinite())
|
|
encoder->SetMaxBitrate(config_.encoder.fake.max_rate.kbps());
|
|
return encoder;
|
|
});
|
|
} else {
|
|
RTC_NOTREACHED();
|
|
}
|
|
break;
|
|
case VideoStreamConfig::Encoder::Implementation::kSoftware:
|
|
encoder_factory_.reset(new InternalEncoderFactory());
|
|
break;
|
|
case VideoStreamConfig::Encoder::Implementation::kHardware:
|
|
encoder_factory_ = CreateHardwareEncoderFactory();
|
|
break;
|
|
}
|
|
RTC_CHECK(encoder_factory_);
|
|
|
|
bitrate_allocator_factory_ = CreateBuiltinVideoBitrateAllocatorFactory();
|
|
RTC_CHECK(bitrate_allocator_factory_);
|
|
|
|
VideoEncoderConfig encoder_config = CreateVideoEncoderConfig(config);
|
|
for (size_t i = 0; i < encoder_config.number_of_streams; ++i) {
|
|
ssrcs_.push_back(sender->GetNextVideoSsrc());
|
|
rtx_ssrcs_.push_back(sender->GetNextRtxSsrc());
|
|
}
|
|
VideoSendStream::Config send_config =
|
|
CreateVideoSendStreamConfig(config, ssrcs_, send_transport);
|
|
send_config.encoder_settings.encoder_factory = encoder_factory_.get();
|
|
send_config.encoder_settings.bitrate_allocator_factory =
|
|
bitrate_allocator_factory_.get();
|
|
|
|
send_stream_ = sender_->call_->CreateVideoSendStream(
|
|
std::move(send_config), std::move(encoder_config));
|
|
std::vector<std::function<void(const VideoFrameQualityInfo&)> >
|
|
frame_info_handlers;
|
|
if (config.analyzer.frame_quality_handler)
|
|
frame_info_handlers.push_back(config.analyzer.frame_quality_handler);
|
|
|
|
if (analyzer->Active()) {
|
|
frame_tap_.reset(new ForwardingCapturedFrameTap(sender_->clock_, analyzer,
|
|
video_capturer_.get()));
|
|
send_stream_->SetSource(frame_tap_.get(),
|
|
config.encoder.degradation_preference);
|
|
} else {
|
|
send_stream_->SetSource(video_capturer_.get(),
|
|
config.encoder.degradation_preference);
|
|
}
|
|
}
|
|
|
|
SendVideoStream::~SendVideoStream() {
|
|
sender_->call_->DestroyVideoSendStream(send_stream_);
|
|
}
|
|
|
|
void SendVideoStream::Start() {
|
|
send_stream_->Start();
|
|
sender_->call_->SignalChannelNetworkState(MediaType::VIDEO, kNetworkUp);
|
|
}
|
|
|
|
void SendVideoStream::Stop() {
|
|
send_stream_->Stop();
|
|
}
|
|
|
|
void SendVideoStream::UpdateConfig(
|
|
std::function<void(VideoStreamConfig*)> modifier) {
|
|
rtc::CritScope cs(&crit_);
|
|
VideoStreamConfig prior_config = config_;
|
|
modifier(&config_);
|
|
if (prior_config.encoder.fake.max_rate != config_.encoder.fake.max_rate) {
|
|
for (auto* encoder : fake_encoders_) {
|
|
encoder->SetMaxBitrate(config_.encoder.fake.max_rate.kbps());
|
|
}
|
|
}
|
|
// TODO(srte): Add more conditions that should cause reconfiguration.
|
|
if (prior_config.encoder.max_framerate != config_.encoder.max_framerate) {
|
|
VideoEncoderConfig encoder_config = CreateVideoEncoderConfig(config_);
|
|
send_stream_->ReconfigureVideoEncoder(std::move(encoder_config));
|
|
}
|
|
if (prior_config.source.framerate != config_.source.framerate) {
|
|
SetCaptureFramerate(config_.source.framerate);
|
|
}
|
|
}
|
|
|
|
void SendVideoStream::SetCaptureFramerate(int framerate) {
|
|
video_capturer_->ChangeFramerate(framerate);
|
|
}
|
|
|
|
VideoSendStream::Stats SendVideoStream::GetStats() const {
|
|
return send_stream_->GetStats();
|
|
}
|
|
|
|
ColumnPrinter SendVideoStream::StatsPrinter() {
|
|
return ColumnPrinter::Lambda(
|
|
"video_target_rate video_sent_rate width height",
|
|
[this](rtc::SimpleStringBuilder& sb) {
|
|
VideoSendStream::Stats video_stats = send_stream_->GetStats();
|
|
int width = 0;
|
|
int height = 0;
|
|
for (const auto& stream_stat : video_stats.substreams) {
|
|
width = std::max(width, stream_stat.second.width);
|
|
height = std::max(height, stream_stat.second.height);
|
|
}
|
|
sb.AppendFormat("%.0lf %.0lf %i %i",
|
|
video_stats.target_media_bitrate_bps / 8.0,
|
|
video_stats.media_bitrate_bps / 8.0, width, height);
|
|
},
|
|
64);
|
|
}
|
|
|
|
ReceiveVideoStream::ReceiveVideoStream(CallClient* receiver,
|
|
VideoStreamConfig config,
|
|
SendVideoStream* send_stream,
|
|
size_t chosen_stream,
|
|
Transport* feedback_transport,
|
|
VideoQualityAnalyzer* analyzer)
|
|
: receiver_(receiver), config_(config) {
|
|
if (analyzer->Active()) {
|
|
renderer_ = absl::make_unique<DecodedFrameTap>(analyzer);
|
|
} else {
|
|
renderer_ = absl::make_unique<FakeVideoRenderer>();
|
|
}
|
|
VideoReceiveStream::Config recv_config(feedback_transport);
|
|
recv_config.rtp.remb = !config.stream.packet_feedback;
|
|
recv_config.rtp.transport_cc = config.stream.packet_feedback;
|
|
recv_config.rtp.local_ssrc = CallTest::kReceiverLocalVideoSsrc;
|
|
recv_config.rtp.extensions = GetVideoRtpExtensions(config);
|
|
receiver_->AddExtensions(recv_config.rtp.extensions);
|
|
RTC_DCHECK(!config.stream.use_rtx ||
|
|
config.stream.nack_history_time > TimeDelta::Zero());
|
|
recv_config.rtp.nack.rtp_history_ms = config.stream.nack_history_time.ms();
|
|
recv_config.rtp.protected_by_flexfec = config.stream.use_flexfec;
|
|
recv_config.renderer = renderer_.get();
|
|
if (config.stream.use_rtx) {
|
|
recv_config.rtp.rtx_ssrc = send_stream->rtx_ssrcs_[chosen_stream];
|
|
receiver->ssrc_media_types_[recv_config.rtp.rtx_ssrc] = MediaType::VIDEO;
|
|
recv_config.rtp
|
|
.rtx_associated_payload_types[CallTest::kSendRtxPayloadType] =
|
|
CodecTypeToPayloadType(config.encoder.codec);
|
|
}
|
|
recv_config.rtp.remote_ssrc = send_stream->ssrcs_[chosen_stream];
|
|
receiver->ssrc_media_types_[recv_config.rtp.remote_ssrc] = MediaType::VIDEO;
|
|
|
|
VideoReceiveStream::Decoder decoder =
|
|
CreateMatchingDecoder(CodecTypeToPayloadType(config.encoder.codec),
|
|
CodecTypeToPayloadString(config.encoder.codec));
|
|
if (config.encoder.codec ==
|
|
VideoStreamConfig::Encoder::Codec::kVideoCodecGeneric) {
|
|
decoder_factory_ = absl::make_unique<FunctionVideoDecoderFactory>(
|
|
[]() { return absl::make_unique<FakeDecoder>(); });
|
|
} else {
|
|
decoder_factory_ = absl::make_unique<InternalDecoderFactory>();
|
|
}
|
|
decoder.decoder_factory = decoder_factory_.get();
|
|
recv_config.decoders.push_back(decoder);
|
|
|
|
if (config.stream.use_flexfec) {
|
|
FlexfecReceiveStream::Config flexfec_config(feedback_transport);
|
|
flexfec_config.payload_type = CallTest::kFlexfecPayloadType;
|
|
flexfec_config.remote_ssrc = CallTest::kFlexfecSendSsrc;
|
|
receiver->ssrc_media_types_[flexfec_config.remote_ssrc] = MediaType::VIDEO;
|
|
flexfec_config.protected_media_ssrcs = send_stream->rtx_ssrcs_;
|
|
flexfec_config.local_ssrc = recv_config.rtp.local_ssrc;
|
|
flecfec_stream_ =
|
|
receiver_->call_->CreateFlexfecReceiveStream(flexfec_config);
|
|
}
|
|
if (config.stream.use_ulpfec) {
|
|
recv_config.rtp.red_payload_type = CallTest::kRedPayloadType;
|
|
recv_config.rtp.ulpfec_payload_type = CallTest::kUlpfecPayloadType;
|
|
recv_config.rtp.rtx_associated_payload_types[CallTest::kRtxRedPayloadType] =
|
|
CallTest::kRedPayloadType;
|
|
}
|
|
receive_stream_ =
|
|
receiver_->call_->CreateVideoReceiveStream(std::move(recv_config));
|
|
}
|
|
|
|
ReceiveVideoStream::~ReceiveVideoStream() {
|
|
receiver_->call_->DestroyVideoReceiveStream(receive_stream_);
|
|
if (flecfec_stream_)
|
|
receiver_->call_->DestroyFlexfecReceiveStream(flecfec_stream_);
|
|
}
|
|
|
|
void ReceiveVideoStream::Start() {
|
|
receive_stream_->Start();
|
|
receiver_->call_->SignalChannelNetworkState(MediaType::VIDEO, kNetworkUp);
|
|
}
|
|
|
|
void ReceiveVideoStream::Stop() {
|
|
receive_stream_->Stop();
|
|
}
|
|
|
|
VideoStreamPair::~VideoStreamPair() = default;
|
|
|
|
VideoStreamPair::VideoStreamPair(
|
|
CallClient* sender,
|
|
CallClient* receiver,
|
|
VideoStreamConfig config,
|
|
std::unique_ptr<RtcEventLogOutput> quality_writer)
|
|
: config_(config),
|
|
analyzer_(std::move(quality_writer),
|
|
config.analyzer.frame_quality_handler),
|
|
send_stream_(sender, config, &sender->transport_, &analyzer_),
|
|
receive_stream_(receiver,
|
|
config,
|
|
&send_stream_,
|
|
/*chosen_stream=*/0,
|
|
&receiver->transport_,
|
|
&analyzer_) {}
|
|
|
|
} // namespace test
|
|
} // namespace webrtc
|