|  | /* | 
|  | *  Copyright (c) 2015 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 "video/video_loopback.h" | 
|  |  | 
|  | #include <stdio.h> | 
|  |  | 
|  | #include <memory> | 
|  | #include <string> | 
|  | #include <vector> | 
|  |  | 
|  | #include "absl/flags/flag.h" | 
|  | #include "absl/flags/parse.h" | 
|  | #include "absl/types/optional.h" | 
|  | #include "api/test/simulated_network.h" | 
|  | #include "api/test/video_quality_test_fixture.h" | 
|  | #include "api/transport/bitrate_settings.h" | 
|  | #include "api/video_codecs/video_codec.h" | 
|  | #include "rtc_base/checks.h" | 
|  | #include "rtc_base/logging.h" | 
|  | #include "system_wrappers/include/field_trial.h" | 
|  | #include "test/field_trial.h" | 
|  | #include "test/gtest.h" | 
|  | #include "test/run_test.h" | 
|  | #include "video/video_quality_test.h" | 
|  |  | 
|  | // Flags common with screenshare loopback, with different default values. | 
|  | ABSL_FLAG(int, width, 640, "Video width."); | 
|  |  | 
|  | ABSL_FLAG(int, height, 480, "Video height."); | 
|  |  | 
|  | ABSL_FLAG(int, fps, 30, "Frames per second."); | 
|  |  | 
|  | ABSL_FLAG(int, capture_device_index, 0, "Capture device to select"); | 
|  |  | 
|  | ABSL_FLAG(int, min_bitrate, 50, "Call and stream min bitrate in kbps."); | 
|  |  | 
|  | ABSL_FLAG(int, start_bitrate, 300, "Call start bitrate in kbps."); | 
|  |  | 
|  | ABSL_FLAG(int, target_bitrate, 800, "Stream target bitrate in kbps."); | 
|  |  | 
|  | ABSL_FLAG(int, max_bitrate, 800, "Call and stream max bitrate in kbps."); | 
|  |  | 
|  | ABSL_FLAG(bool, | 
|  | suspend_below_min_bitrate, | 
|  | false, | 
|  | "Suspends video below the configured min bitrate."); | 
|  |  | 
|  | ABSL_FLAG(int, | 
|  | num_temporal_layers, | 
|  | 1, | 
|  | "Number of temporal layers. Set to 1-4 to override."); | 
|  |  | 
|  | ABSL_FLAG(int, | 
|  | inter_layer_pred, | 
|  | 2, | 
|  | "Inter-layer prediction mode. " | 
|  | "0 - enabled, 1 - disabled, 2 - enabled only for key pictures."); | 
|  |  | 
|  | // Flags common with screenshare loopback, with equal default values. | 
|  | ABSL_FLAG(std::string, codec, "VP8", "Video codec to use."); | 
|  |  | 
|  | ABSL_FLAG(int, | 
|  | selected_tl, | 
|  | -1, | 
|  | "Temporal layer to show or analyze. -1 to disable filtering."); | 
|  |  | 
|  | ABSL_FLAG( | 
|  | int, | 
|  | duration, | 
|  | 0, | 
|  | "Duration of the test in seconds. If 0, rendered will be shown instead."); | 
|  |  | 
|  | ABSL_FLAG(std::string, output_filename, "", "Target graph data filename."); | 
|  |  | 
|  | ABSL_FLAG(std::string, | 
|  | graph_title, | 
|  | "", | 
|  | "If empty, title will be generated automatically."); | 
|  |  | 
|  | ABSL_FLAG(int, loss_percent, 0, "Percentage of packets randomly lost."); | 
|  |  | 
|  | ABSL_FLAG(int, | 
|  | avg_burst_loss_length, | 
|  | -1, | 
|  | "Average burst length of lost packets."); | 
|  |  | 
|  | ABSL_FLAG(int, | 
|  | link_capacity, | 
|  | 0, | 
|  | "Capacity (kbps) of the fake link. 0 means infinite."); | 
|  |  | 
|  | ABSL_FLAG(int, queue_size, 0, "Size of the bottleneck link queue in packets."); | 
|  |  | 
|  | ABSL_FLAG(int, | 
|  | avg_propagation_delay_ms, | 
|  | 0, | 
|  | "Average link propagation delay in ms."); | 
|  |  | 
|  | ABSL_FLAG(std::string, | 
|  | rtc_event_log_name, | 
|  | "", | 
|  | "Filename for rtc event log. Two files " | 
|  | "with \"_send\" and \"_recv\" suffixes will be created."); | 
|  |  | 
|  | ABSL_FLAG(std::string, | 
|  | rtp_dump_name, | 
|  | "", | 
|  | "Filename for dumped received RTP stream."); | 
|  |  | 
|  | ABSL_FLAG(int, | 
|  | std_propagation_delay_ms, | 
|  | 0, | 
|  | "Link propagation delay standard deviation in ms."); | 
|  |  | 
|  | ABSL_FLAG(int, num_streams, 0, "Number of streams to show or analyze."); | 
|  |  | 
|  | ABSL_FLAG(int, | 
|  | selected_stream, | 
|  | 0, | 
|  | "ID of the stream to show or analyze. " | 
|  | "Set to the number of streams to show them all."); | 
|  |  | 
|  | ABSL_FLAG(int, num_spatial_layers, 1, "Number of spatial layers to use."); | 
|  |  | 
|  | ABSL_FLAG(int, | 
|  | selected_sl, | 
|  | -1, | 
|  | "Spatial layer to show or analyze. -1 to disable filtering."); | 
|  |  | 
|  | ABSL_FLAG(std::string, | 
|  | stream0, | 
|  | "", | 
|  | "Comma separated values describing VideoStream for stream #0."); | 
|  |  | 
|  | ABSL_FLAG(std::string, | 
|  | stream1, | 
|  | "", | 
|  | "Comma separated values describing VideoStream for stream #1."); | 
|  |  | 
|  | ABSL_FLAG(std::string, | 
|  | sl0, | 
|  | "", | 
|  | "Comma separated values describing SpatialLayer for layer #0."); | 
|  |  | 
|  | ABSL_FLAG(std::string, | 
|  | sl1, | 
|  | "", | 
|  | "Comma separated values describing SpatialLayer for layer #1."); | 
|  |  | 
|  | ABSL_FLAG(std::string, | 
|  | sl2, | 
|  | "", | 
|  | "Comma separated values describing SpatialLayer for layer #2."); | 
|  |  | 
|  | ABSL_FLAG(std::string, | 
|  | encoded_frame_path, | 
|  | "", | 
|  | "The base path for encoded frame logs. Created files will have " | 
|  | "the form <encoded_frame_path>.<n>.(recv|send.<m>).ivf"); | 
|  |  | 
|  | ABSL_FLAG(bool, logs, false, "print logs to stderr"); | 
|  |  | 
|  | ABSL_FLAG(bool, send_side_bwe, true, "Use send-side bandwidth estimation"); | 
|  |  | 
|  | ABSL_FLAG(bool, generic_descriptor, false, "Use the generic frame descriptor."); | 
|  |  | 
|  | ABSL_FLAG(bool, allow_reordering, false, "Allow packet reordering to occur"); | 
|  |  | 
|  | ABSL_FLAG(bool, use_ulpfec, false, "Use RED+ULPFEC forward error correction."); | 
|  |  | 
|  | ABSL_FLAG(bool, use_flexfec, false, "Use FlexFEC forward error correction."); | 
|  |  | 
|  | ABSL_FLAG(bool, audio, false, "Add audio stream"); | 
|  |  | 
|  | ABSL_FLAG(bool, | 
|  | use_real_adm, | 
|  | false, | 
|  | "Use real ADM instead of fake (no effect if audio is false)"); | 
|  |  | 
|  | ABSL_FLAG(bool, | 
|  | audio_video_sync, | 
|  | false, | 
|  | "Sync audio and video stream (no effect if" | 
|  | " audio is false)"); | 
|  |  | 
|  | ABSL_FLAG(bool, | 
|  | audio_dtx, | 
|  | false, | 
|  | "Enable audio DTX (no effect if audio is false)"); | 
|  |  | 
|  | ABSL_FLAG(bool, video, true, "Add video stream"); | 
|  |  | 
|  | ABSL_FLAG( | 
|  | std::string, | 
|  | force_fieldtrials, | 
|  | "", | 
|  | "Field trials control experimental feature code which can be forced. " | 
|  | "E.g. running with --force_fieldtrials=WebRTC-FooFeature/Enabled/" | 
|  | " will assign the group Enable to field trial WebRTC-FooFeature. Multiple " | 
|  | "trials are separated by \"/\""); | 
|  |  | 
|  | // Video-specific flags. | 
|  | ABSL_FLAG(std::string, | 
|  | clip, | 
|  | "", | 
|  | "Name of the clip to show. If empty, using chroma generator."); | 
|  |  | 
|  | namespace webrtc { | 
|  | namespace { | 
|  |  | 
|  | size_t Width() { | 
|  | return static_cast<size_t>(absl::GetFlag(FLAGS_width)); | 
|  | } | 
|  |  | 
|  | size_t Height() { | 
|  | return static_cast<size_t>(absl::GetFlag(FLAGS_height)); | 
|  | } | 
|  |  | 
|  | int Fps() { | 
|  | return absl::GetFlag(FLAGS_fps); | 
|  | } | 
|  |  | 
|  | size_t GetCaptureDevice() { | 
|  | return static_cast<size_t>(absl::GetFlag(FLAGS_capture_device_index)); | 
|  | } | 
|  |  | 
|  | int MinBitrateKbps() { | 
|  | return absl::GetFlag(FLAGS_min_bitrate); | 
|  | } | 
|  |  | 
|  | int StartBitrateKbps() { | 
|  | return absl::GetFlag(FLAGS_start_bitrate); | 
|  | } | 
|  |  | 
|  | int TargetBitrateKbps() { | 
|  | return absl::GetFlag(FLAGS_target_bitrate); | 
|  | } | 
|  |  | 
|  | int MaxBitrateKbps() { | 
|  | return absl::GetFlag(FLAGS_max_bitrate); | 
|  | } | 
|  |  | 
|  | int NumTemporalLayers() { | 
|  | return absl::GetFlag(FLAGS_num_temporal_layers); | 
|  | } | 
|  |  | 
|  | InterLayerPredMode InterLayerPred() { | 
|  | if (absl::GetFlag(FLAGS_inter_layer_pred) == 0) { | 
|  | return InterLayerPredMode::kOn; | 
|  | } else if (absl::GetFlag(FLAGS_inter_layer_pred) == 1) { | 
|  | return InterLayerPredMode::kOff; | 
|  | } else { | 
|  | RTC_DCHECK_EQ(absl::GetFlag(FLAGS_inter_layer_pred), 2); | 
|  | return InterLayerPredMode::kOnKeyPic; | 
|  | } | 
|  | } | 
|  |  | 
|  | std::string Codec() { | 
|  | return absl::GetFlag(FLAGS_codec); | 
|  | } | 
|  |  | 
|  | int SelectedTL() { | 
|  | return absl::GetFlag(FLAGS_selected_tl); | 
|  | } | 
|  |  | 
|  | int DurationSecs() { | 
|  | return absl::GetFlag(FLAGS_duration); | 
|  | } | 
|  |  | 
|  | std::string OutputFilename() { | 
|  | return absl::GetFlag(FLAGS_output_filename); | 
|  | } | 
|  |  | 
|  | std::string GraphTitle() { | 
|  | return absl::GetFlag(FLAGS_graph_title); | 
|  | } | 
|  |  | 
|  | int LossPercent() { | 
|  | return static_cast<int>(absl::GetFlag(FLAGS_loss_percent)); | 
|  | } | 
|  |  | 
|  | int AvgBurstLossLength() { | 
|  | return static_cast<int>(absl::GetFlag(FLAGS_avg_burst_loss_length)); | 
|  | } | 
|  |  | 
|  | int LinkCapacityKbps() { | 
|  | return static_cast<int>(absl::GetFlag(FLAGS_link_capacity)); | 
|  | } | 
|  |  | 
|  | int QueueSize() { | 
|  | return static_cast<int>(absl::GetFlag(FLAGS_queue_size)); | 
|  | } | 
|  |  | 
|  | int AvgPropagationDelayMs() { | 
|  | return static_cast<int>(absl::GetFlag(FLAGS_avg_propagation_delay_ms)); | 
|  | } | 
|  |  | 
|  | std::string RtcEventLogName() { | 
|  | return absl::GetFlag(FLAGS_rtc_event_log_name); | 
|  | } | 
|  |  | 
|  | std::string RtpDumpName() { | 
|  | return absl::GetFlag(FLAGS_rtp_dump_name); | 
|  | } | 
|  |  | 
|  | int StdPropagationDelayMs() { | 
|  | return absl::GetFlag(FLAGS_std_propagation_delay_ms); | 
|  | } | 
|  |  | 
|  | int NumStreams() { | 
|  | return absl::GetFlag(FLAGS_num_streams); | 
|  | } | 
|  |  | 
|  | int SelectedStream() { | 
|  | return absl::GetFlag(FLAGS_selected_stream); | 
|  | } | 
|  |  | 
|  | int NumSpatialLayers() { | 
|  | return absl::GetFlag(FLAGS_num_spatial_layers); | 
|  | } | 
|  |  | 
|  | int SelectedSL() { | 
|  | return absl::GetFlag(FLAGS_selected_sl); | 
|  | } | 
|  |  | 
|  | std::string Stream0() { | 
|  | return absl::GetFlag(FLAGS_stream0); | 
|  | } | 
|  |  | 
|  | std::string Stream1() { | 
|  | return absl::GetFlag(FLAGS_stream1); | 
|  | } | 
|  |  | 
|  | std::string SL0() { | 
|  | return absl::GetFlag(FLAGS_sl0); | 
|  | } | 
|  |  | 
|  | std::string SL1() { | 
|  | return absl::GetFlag(FLAGS_sl1); | 
|  | } | 
|  |  | 
|  | std::string SL2() { | 
|  | return absl::GetFlag(FLAGS_sl2); | 
|  | } | 
|  |  | 
|  | std::string EncodedFramePath() { | 
|  | return absl::GetFlag(FLAGS_encoded_frame_path); | 
|  | } | 
|  |  | 
|  | std::string Clip() { | 
|  | return absl::GetFlag(FLAGS_clip); | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | void Loopback() { | 
|  | BuiltInNetworkBehaviorConfig pipe_config; | 
|  | pipe_config.loss_percent = LossPercent(); | 
|  | pipe_config.avg_burst_loss_length = AvgBurstLossLength(); | 
|  | pipe_config.link_capacity_kbps = LinkCapacityKbps(); | 
|  | pipe_config.queue_length_packets = QueueSize(); | 
|  | pipe_config.queue_delay_ms = AvgPropagationDelayMs(); | 
|  | pipe_config.delay_standard_deviation_ms = StdPropagationDelayMs(); | 
|  | pipe_config.allow_reordering = absl::GetFlag(FLAGS_allow_reordering); | 
|  |  | 
|  | BitrateConstraints call_bitrate_config; | 
|  | call_bitrate_config.min_bitrate_bps = MinBitrateKbps() * 1000; | 
|  | call_bitrate_config.start_bitrate_bps = StartBitrateKbps() * 1000; | 
|  | call_bitrate_config.max_bitrate_bps = -1;  // Don't cap bandwidth estimate. | 
|  |  | 
|  | VideoQualityTest::Params params; | 
|  | params.call.send_side_bwe = absl::GetFlag(FLAGS_send_side_bwe); | 
|  | params.call.generic_descriptor = absl::GetFlag(FLAGS_generic_descriptor); | 
|  | params.call.call_bitrate_config = call_bitrate_config; | 
|  |  | 
|  | params.video[0].enabled = absl::GetFlag(FLAGS_video); | 
|  | params.video[0].width = Width(); | 
|  | params.video[0].height = Height(); | 
|  | params.video[0].fps = Fps(); | 
|  | params.video[0].min_bitrate_bps = MinBitrateKbps() * 1000; | 
|  | params.video[0].target_bitrate_bps = TargetBitrateKbps() * 1000; | 
|  | params.video[0].max_bitrate_bps = MaxBitrateKbps() * 1000; | 
|  | params.video[0].suspend_below_min_bitrate = | 
|  | absl::GetFlag(FLAGS_suspend_below_min_bitrate); | 
|  | params.video[0].codec = Codec(); | 
|  | params.video[0].num_temporal_layers = NumTemporalLayers(); | 
|  | params.video[0].selected_tl = SelectedTL(); | 
|  | params.video[0].min_transmit_bps = 0; | 
|  | params.video[0].ulpfec = absl::GetFlag(FLAGS_use_ulpfec); | 
|  | params.video[0].flexfec = absl::GetFlag(FLAGS_use_flexfec); | 
|  | params.video[0].automatic_scaling = NumStreams() < 2; | 
|  | params.video[0].clip_path = Clip(); | 
|  | params.video[0].capture_device_index = GetCaptureDevice(); | 
|  | params.audio.enabled = absl::GetFlag(FLAGS_audio); | 
|  | params.audio.sync_video = absl::GetFlag(FLAGS_audio_video_sync); | 
|  | params.audio.dtx = absl::GetFlag(FLAGS_audio_dtx); | 
|  | params.audio.use_real_adm = absl::GetFlag(FLAGS_use_real_adm); | 
|  | params.logging.rtc_event_log_name = RtcEventLogName(); | 
|  | params.logging.rtp_dump_name = RtpDumpName(); | 
|  | params.logging.encoded_frame_base_path = EncodedFramePath(); | 
|  | params.screenshare[0].enabled = false; | 
|  | params.analyzer.test_label = "video"; | 
|  | params.analyzer.test_durations_secs = DurationSecs(); | 
|  | params.analyzer.graph_data_output_filename = OutputFilename(); | 
|  | params.analyzer.graph_title = GraphTitle(); | 
|  | params.config = pipe_config; | 
|  |  | 
|  | if (NumStreams() > 1 && Stream0().empty() && Stream1().empty()) { | 
|  | params.ss[0].infer_streams = true; | 
|  | } | 
|  |  | 
|  | std::vector<std::string> stream_descriptors; | 
|  | stream_descriptors.push_back(Stream0()); | 
|  | stream_descriptors.push_back(Stream1()); | 
|  | std::vector<std::string> SL_descriptors; | 
|  | SL_descriptors.push_back(SL0()); | 
|  | SL_descriptors.push_back(SL1()); | 
|  | SL_descriptors.push_back(SL2()); | 
|  | VideoQualityTest::FillScalabilitySettings( | 
|  | ¶ms, 0, stream_descriptors, NumStreams(), SelectedStream(), | 
|  | NumSpatialLayers(), SelectedSL(), InterLayerPred(), SL_descriptors); | 
|  |  | 
|  | auto fixture = std::make_unique<VideoQualityTest>(nullptr); | 
|  | if (DurationSecs()) { | 
|  | fixture->RunWithAnalyzer(params); | 
|  | } else { | 
|  | fixture->RunWithRenderers(params); | 
|  | } | 
|  | } | 
|  |  | 
|  | int RunLoopbackTest(int argc, char* argv[]) { | 
|  | ::testing::InitGoogleTest(&argc, argv); | 
|  | absl::ParseCommandLine(argc, argv); | 
|  |  | 
|  | rtc::LogMessage::SetLogToStderr(absl::GetFlag(FLAGS_logs)); | 
|  |  | 
|  | // InitFieldTrialsFromString stores the char*, so the char array must outlive | 
|  | // the application. | 
|  | const std::string field_trials = absl::GetFlag(FLAGS_force_fieldtrials); | 
|  | webrtc::field_trial::InitFieldTrialsFromString(field_trials.c_str()); | 
|  |  | 
|  | webrtc::test::RunTest(webrtc::Loopback); | 
|  | return 0; | 
|  | } | 
|  | }  // namespace webrtc |