/* * Copyright 2020 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_source_sink_controller.h" #include #include #include #include "rtc_base/logging.h" #include "rtc_base/numerics/safe_conversions.h" #include "rtc_base/strings/string_builder.h" namespace webrtc { namespace { std::string WantsToString(const rtc::VideoSinkWants& wants) { rtc::StringBuilder ss; ss << "max_fps=" << wants.max_framerate_fps << " max_pixel_count=" << wants.max_pixel_count << " target_pixel_count=" << (wants.target_pixel_count.has_value() ? std::to_string(wants.target_pixel_count.value()) : "null") << " resolutions={"; for (size_t i = 0; i < wants.resolutions.size(); ++i) { if (i != 0) ss << ","; ss << wants.resolutions[i].width << "x" << wants.resolutions[i].height; } ss << "}"; return ss.Release(); } } // namespace VideoSourceSinkController::VideoSourceSinkController( rtc::VideoSinkInterface* sink, rtc::VideoSourceInterface* source) : sink_(sink), source_(source) { RTC_DCHECK(sink_); } VideoSourceSinkController::~VideoSourceSinkController() { RTC_DCHECK_RUN_ON(&sequence_checker_); } void VideoSourceSinkController::SetSource( rtc::VideoSourceInterface* source) { RTC_DCHECK_RUN_ON(&sequence_checker_); rtc::VideoSourceInterface* old_source = source_; source_ = source; if (old_source != source && old_source) old_source->RemoveSink(sink_); if (!source) return; source->AddOrUpdateSink(sink_, CurrentSettingsToSinkWants()); } bool VideoSourceSinkController::HasSource() const { RTC_DCHECK_RUN_ON(&sequence_checker_); return source_ != nullptr; } void VideoSourceSinkController::RequestRefreshFrame() { RTC_DCHECK_RUN_ON(&sequence_checker_); if (source_) source_->RequestRefreshFrame(); } void VideoSourceSinkController::PushSourceSinkSettings() { RTC_DCHECK_RUN_ON(&sequence_checker_); if (!source_) return; rtc::VideoSinkWants wants = CurrentSettingsToSinkWants(); RTC_LOG(LS_INFO) << "Pushing SourceSink restrictions: " << WantsToString(wants); source_->AddOrUpdateSink(sink_, wants); } VideoSourceRestrictions VideoSourceSinkController::restrictions() const { RTC_DCHECK_RUN_ON(&sequence_checker_); return restrictions_; } absl::optional VideoSourceSinkController::pixels_per_frame_upper_limit() const { RTC_DCHECK_RUN_ON(&sequence_checker_); return pixels_per_frame_upper_limit_; } absl::optional VideoSourceSinkController::frame_rate_upper_limit() const { RTC_DCHECK_RUN_ON(&sequence_checker_); return frame_rate_upper_limit_; } bool VideoSourceSinkController::rotation_applied() const { RTC_DCHECK_RUN_ON(&sequence_checker_); return rotation_applied_; } int VideoSourceSinkController::resolution_alignment() const { RTC_DCHECK_RUN_ON(&sequence_checker_); return resolution_alignment_; } const std::vector& VideoSourceSinkController::resolutions() const { RTC_DCHECK_RUN_ON(&sequence_checker_); return resolutions_; } void VideoSourceSinkController::SetRestrictions( VideoSourceRestrictions restrictions) { RTC_DCHECK_RUN_ON(&sequence_checker_); restrictions_ = std::move(restrictions); } void VideoSourceSinkController::SetPixelsPerFrameUpperLimit( absl::optional pixels_per_frame_upper_limit) { RTC_DCHECK_RUN_ON(&sequence_checker_); pixels_per_frame_upper_limit_ = std::move(pixels_per_frame_upper_limit); } void VideoSourceSinkController::SetFrameRateUpperLimit( absl::optional frame_rate_upper_limit) { RTC_DCHECK_RUN_ON(&sequence_checker_); frame_rate_upper_limit_ = std::move(frame_rate_upper_limit); } void VideoSourceSinkController::SetRotationApplied(bool rotation_applied) { RTC_DCHECK_RUN_ON(&sequence_checker_); rotation_applied_ = rotation_applied; } void VideoSourceSinkController::SetResolutionAlignment( int resolution_alignment) { RTC_DCHECK_RUN_ON(&sequence_checker_); resolution_alignment_ = resolution_alignment; } void VideoSourceSinkController::SetResolutions( std::vector resolutions) { RTC_DCHECK_RUN_ON(&sequence_checker_); resolutions_ = std::move(resolutions); } // RTC_EXCLUSIVE_LOCKS_REQUIRED(sequence_checker_) rtc::VideoSinkWants VideoSourceSinkController::CurrentSettingsToSinkWants() const { rtc::VideoSinkWants wants; wants.rotation_applied = rotation_applied_; // `wants.black_frames` is not used, it always has its default value false. wants.max_pixel_count = rtc::dchecked_cast(restrictions_.max_pixels_per_frame().value_or( std::numeric_limits::max())); wants.target_pixel_count = restrictions_.target_pixels_per_frame().has_value() ? absl::optional(rtc::dchecked_cast( restrictions_.target_pixels_per_frame().value())) : absl::nullopt; wants.max_framerate_fps = restrictions_.max_frame_rate().has_value() ? static_cast(restrictions_.max_frame_rate().value()) : std::numeric_limits::max(); wants.resolution_alignment = resolution_alignment_; wants.max_pixel_count = std::min(wants.max_pixel_count, rtc::dchecked_cast(pixels_per_frame_upper_limit_.value_or( std::numeric_limits::max()))); wants.max_framerate_fps = std::min(wants.max_framerate_fps, frame_rate_upper_limit_.has_value() ? static_cast(frame_rate_upper_limit_.value()) : std::numeric_limits::max()); wants.resolutions = resolutions_; return wants; } } // namespace webrtc