Some of the state that's managed in VideoStreamEncoder, is updated and accessed on the encoder queue, but a method that's used for testing only (GetAdaptationResources()), represents a race between PostTask operations that update state and checking for said state on the worker thread. This CL removes Wait() operations related to adaptation resources from the common path and puts one in the test path instead. Bug: webrtc:13612 Change-Id: Ie3e018e815e24951bc0634ed70de17eaf336a508 Reviewed-on: https://webrtc-review.googlesource.com/c/src/+/249220 Reviewed-by: Henrik Boström <hbos@webrtc.org> Commit-Queue: Tomas Gunnarsson <tommi@webrtc.org> Cr-Commit-Position: refs/heads/main@{#35797}
166 lines
7.1 KiB
C++
166 lines
7.1 KiB
C++
/*
|
|
* 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.
|
|
*/
|
|
|
|
#ifndef CALL_ADAPTATION_RESOURCE_ADAPTATION_PROCESSOR_H_
|
|
#define CALL_ADAPTATION_RESOURCE_ADAPTATION_PROCESSOR_H_
|
|
|
|
#include <map>
|
|
#include <memory>
|
|
#include <string>
|
|
#include <utility>
|
|
#include <vector>
|
|
|
|
#include "absl/types/optional.h"
|
|
#include "api/adaptation/resource.h"
|
|
#include "api/rtp_parameters.h"
|
|
#include "api/scoped_refptr.h"
|
|
#include "api/task_queue/task_queue_base.h"
|
|
#include "api/video/video_adaptation_counters.h"
|
|
#include "api/video/video_frame.h"
|
|
#include "api/video/video_stream_encoder_observer.h"
|
|
#include "call/adaptation/resource_adaptation_processor_interface.h"
|
|
#include "call/adaptation/video_source_restrictions.h"
|
|
#include "call/adaptation/video_stream_adapter.h"
|
|
#include "call/adaptation/video_stream_input_state.h"
|
|
#include "call/adaptation/video_stream_input_state_provider.h"
|
|
|
|
namespace webrtc {
|
|
|
|
// The Resource Adaptation Processor is responsible for reacting to resource
|
|
// usage measurements (e.g. overusing or underusing CPU). When a resource is
|
|
// overused the Processor is responsible for performing mitigations in order to
|
|
// consume less resources.
|
|
//
|
|
// Today we have one Processor per VideoStreamEncoder and the Processor is only
|
|
// capable of restricting resolution or frame rate of the encoded stream. In the
|
|
// future we should have a single Processor responsible for all encoded streams,
|
|
// and it should be capable of reconfiguring other things than just
|
|
// VideoSourceRestrictions (e.g. reduce render frame rate).
|
|
// See Resource-Adaptation hotlist:
|
|
// https://bugs.chromium.org/u/590058293/hotlists/Resource-Adaptation
|
|
//
|
|
// The ResourceAdaptationProcessor is single-threaded. It may be constructed on
|
|
// any thread but MUST subsequently be used and destroyed on a single sequence,
|
|
// i.e. the "resource adaptation task queue". Resources can be added and removed
|
|
// from any thread.
|
|
class ResourceAdaptationProcessor : public ResourceAdaptationProcessorInterface,
|
|
public VideoSourceRestrictionsListener,
|
|
public ResourceListener {
|
|
public:
|
|
explicit ResourceAdaptationProcessor(
|
|
VideoStreamAdapter* video_stream_adapter);
|
|
~ResourceAdaptationProcessor() override;
|
|
|
|
// ResourceAdaptationProcessorInterface implementation.
|
|
void AddResourceLimitationsListener(
|
|
ResourceLimitationsListener* limitations_listener) override;
|
|
void RemoveResourceLimitationsListener(
|
|
ResourceLimitationsListener* limitations_listener) override;
|
|
void AddResource(rtc::scoped_refptr<Resource> resource) override;
|
|
std::vector<rtc::scoped_refptr<Resource>> GetResources() const override;
|
|
void RemoveResource(rtc::scoped_refptr<Resource> resource) override;
|
|
|
|
// ResourceListener implementation.
|
|
// Triggers OnResourceUnderuse() or OnResourceOveruse().
|
|
void OnResourceUsageStateMeasured(rtc::scoped_refptr<Resource> resource,
|
|
ResourceUsageState usage_state) override;
|
|
|
|
// VideoSourceRestrictionsListener implementation.
|
|
void OnVideoSourceRestrictionsUpdated(
|
|
VideoSourceRestrictions restrictions,
|
|
const VideoAdaptationCounters& adaptation_counters,
|
|
rtc::scoped_refptr<Resource> reason,
|
|
const VideoSourceRestrictions& unfiltered_restrictions) override;
|
|
|
|
private:
|
|
// If resource usage measurements happens off the adaptation task queue, this
|
|
// class takes care of posting the measurement for the processor to handle it
|
|
// on the adaptation task queue.
|
|
class ResourceListenerDelegate : public rtc::RefCountInterface,
|
|
public ResourceListener {
|
|
public:
|
|
explicit ResourceListenerDelegate(ResourceAdaptationProcessor* processor);
|
|
|
|
void OnProcessorDestroyed();
|
|
|
|
// ResourceListener implementation.
|
|
void OnResourceUsageStateMeasured(rtc::scoped_refptr<Resource> resource,
|
|
ResourceUsageState usage_state) override;
|
|
|
|
private:
|
|
TaskQueueBase* task_queue_;
|
|
ResourceAdaptationProcessor* processor_ RTC_GUARDED_BY(task_queue_);
|
|
};
|
|
|
|
enum class MitigationResult {
|
|
kNotMostLimitedResource,
|
|
kSharedMostLimitedResource,
|
|
kRejectedByAdapter,
|
|
kAdaptationApplied,
|
|
};
|
|
|
|
struct MitigationResultAndLogMessage {
|
|
MitigationResultAndLogMessage();
|
|
MitigationResultAndLogMessage(MitigationResult result, std::string message);
|
|
MitigationResult result;
|
|
std::string message;
|
|
};
|
|
|
|
// Performs the adaptation by getting the next target, applying it and
|
|
// informing listeners of the new VideoSourceRestriction and adaptation
|
|
// counters.
|
|
MitigationResultAndLogMessage OnResourceUnderuse(
|
|
rtc::scoped_refptr<Resource> reason_resource);
|
|
MitigationResultAndLogMessage OnResourceOveruse(
|
|
rtc::scoped_refptr<Resource> reason_resource);
|
|
|
|
void UpdateResourceLimitations(rtc::scoped_refptr<Resource> reason_resource,
|
|
const VideoSourceRestrictions& restrictions,
|
|
const VideoAdaptationCounters& counters)
|
|
RTC_RUN_ON(task_queue_);
|
|
|
|
// Searches `adaptation_limits_by_resources_` for each resource with the
|
|
// highest total adaptation counts. Adaptation up may only occur if the
|
|
// resource performing the adaptation is the only most limited resource. This
|
|
// function returns the list of all most limited resources as well as the
|
|
// corresponding adaptation of that resource.
|
|
std::pair<std::vector<rtc::scoped_refptr<Resource>>,
|
|
VideoStreamAdapter::RestrictionsWithCounters>
|
|
FindMostLimitedResources() const RTC_RUN_ON(task_queue_);
|
|
|
|
void RemoveLimitationsImposedByResource(
|
|
rtc::scoped_refptr<Resource> resource);
|
|
|
|
TaskQueueBase* task_queue_;
|
|
rtc::scoped_refptr<ResourceListenerDelegate> resource_listener_delegate_;
|
|
// Input and output.
|
|
mutable Mutex resources_lock_;
|
|
std::vector<rtc::scoped_refptr<Resource>> resources_
|
|
RTC_GUARDED_BY(resources_lock_);
|
|
std::vector<ResourceLimitationsListener*> resource_limitations_listeners_
|
|
RTC_GUARDED_BY(task_queue_);
|
|
// Purely used for statistics, does not ensure mapped resources stay alive.
|
|
std::map<rtc::scoped_refptr<Resource>,
|
|
VideoStreamAdapter::RestrictionsWithCounters>
|
|
adaptation_limits_by_resources_ RTC_GUARDED_BY(task_queue_);
|
|
// Responsible for generating and applying possible adaptations.
|
|
VideoStreamAdapter* const stream_adapter_ RTC_GUARDED_BY(task_queue_);
|
|
VideoSourceRestrictions last_reported_source_restrictions_
|
|
RTC_GUARDED_BY(task_queue_);
|
|
// Keeps track of previous mitigation results per resource since the last
|
|
// successful adaptation. Used to avoid RTC_LOG spam.
|
|
std::map<Resource*, MitigationResult> previous_mitigation_results_
|
|
RTC_GUARDED_BY(task_queue_);
|
|
};
|
|
|
|
} // namespace webrtc
|
|
|
|
#endif // CALL_ADAPTATION_RESOURCE_ADAPTATION_PROCESSOR_H_
|