2020-08-14 19:58:22 +03:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2014 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 MEDIA_ENGINE_SIMULCAST_ENCODER_ADAPTER_H_
|
|
|
|
#define MEDIA_ENGINE_SIMULCAST_ENCODER_ADAPTER_H_
|
|
|
|
|
2021-06-25 03:43:10 +03:00
|
|
|
#include <list>
|
2020-08-14 19:58:22 +03:00
|
|
|
#include <memory>
|
|
|
|
#include <stack>
|
|
|
|
#include <string>
|
|
|
|
#include <utility>
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
#include "absl/types/optional.h"
|
|
|
|
#include "api/fec_controller_override.h"
|
2021-06-25 03:43:10 +03:00
|
|
|
#include "api/sequence_checker.h"
|
2020-08-14 19:58:22 +03:00
|
|
|
#include "api/video_codecs/sdp_video_format.h"
|
|
|
|
#include "api/video_codecs/video_encoder.h"
|
2021-06-25 03:43:10 +03:00
|
|
|
#include "api/video_codecs/video_encoder_factory.h"
|
2022-03-11 19:49:54 +03:00
|
|
|
#include "common_video/framerate_controller.h"
|
2020-08-14 19:58:22 +03:00
|
|
|
#include "modules/video_coding/include/video_codec_interface.h"
|
|
|
|
#include "rtc_base/atomic_ops.h"
|
2021-06-25 03:43:10 +03:00
|
|
|
#include "rtc_base/experiments/encoder_info_settings.h"
|
|
|
|
#include "rtc_base/system/no_unique_address.h"
|
2020-08-14 19:58:22 +03:00
|
|
|
#include "rtc_base/system/rtc_export.h"
|
|
|
|
|
|
|
|
namespace webrtc {
|
|
|
|
|
|
|
|
// SimulcastEncoderAdapter implements simulcast support by creating multiple
|
|
|
|
// webrtc::VideoEncoder instances with the given VideoEncoderFactory.
|
|
|
|
// The object is created and destroyed on the worker thread, but all public
|
|
|
|
// interfaces should be called from the encoder task queue.
|
|
|
|
class RTC_EXPORT SimulcastEncoderAdapter : public VideoEncoder {
|
|
|
|
public:
|
|
|
|
// TODO(bugs.webrtc.org/11000): Remove when downstream usage is gone.
|
|
|
|
SimulcastEncoderAdapter(VideoEncoderFactory* primarty_factory,
|
|
|
|
const SdpVideoFormat& format);
|
2022-03-11 19:49:54 +03:00
|
|
|
// `primary_factory` produces the first-choice encoders to use.
|
|
|
|
// `fallback_factory`, if non-null, is used to create fallback encoder that
|
2020-08-14 19:58:22 +03:00
|
|
|
// will be used if InitEncode() fails for the primary encoder.
|
|
|
|
SimulcastEncoderAdapter(VideoEncoderFactory* primary_factory,
|
|
|
|
VideoEncoderFactory* fallback_factory,
|
|
|
|
const SdpVideoFormat& format);
|
|
|
|
~SimulcastEncoderAdapter() override;
|
|
|
|
|
|
|
|
// Implements VideoEncoder.
|
|
|
|
void SetFecControllerOverride(
|
|
|
|
FecControllerOverride* fec_controller_override) override;
|
|
|
|
int Release() override;
|
|
|
|
int InitEncode(const VideoCodec* codec_settings,
|
|
|
|
const VideoEncoder::Settings& settings) override;
|
|
|
|
int Encode(const VideoFrame& input_image,
|
|
|
|
const std::vector<VideoFrameType>* frame_types) override;
|
|
|
|
int RegisterEncodeCompleteCallback(EncodedImageCallback* callback) override;
|
|
|
|
void SetRates(const RateControlParameters& parameters) override;
|
|
|
|
void OnPacketLossRateUpdate(float packet_loss_rate) override;
|
|
|
|
void OnRttUpdate(int64_t rtt_ms) override;
|
|
|
|
void OnLossNotification(const LossNotification& loss_notification) override;
|
|
|
|
|
|
|
|
EncoderInfo GetEncoderInfo() const override;
|
|
|
|
|
|
|
|
private:
|
2021-06-25 03:43:10 +03:00
|
|
|
class EncoderContext {
|
|
|
|
public:
|
|
|
|
EncoderContext(std::unique_ptr<VideoEncoder> encoder,
|
2022-03-11 19:49:54 +03:00
|
|
|
bool prefer_temporal_support,
|
|
|
|
VideoEncoder::EncoderInfo primary_info,
|
|
|
|
VideoEncoder::EncoderInfo fallback_info);
|
2021-06-25 03:43:10 +03:00
|
|
|
EncoderContext& operator=(EncoderContext&&) = delete;
|
|
|
|
|
|
|
|
VideoEncoder& encoder() { return *encoder_; }
|
|
|
|
bool prefer_temporal_support() { return prefer_temporal_support_; }
|
|
|
|
void Release();
|
|
|
|
|
2022-03-11 19:49:54 +03:00
|
|
|
const VideoEncoder::EncoderInfo& PrimaryInfo() { return primary_info_; }
|
|
|
|
|
|
|
|
const VideoEncoder::EncoderInfo& FallbackInfo() { return fallback_info_; }
|
|
|
|
|
2021-06-25 03:43:10 +03:00
|
|
|
private:
|
|
|
|
std::unique_ptr<VideoEncoder> encoder_;
|
|
|
|
bool prefer_temporal_support_;
|
2022-03-11 19:49:54 +03:00
|
|
|
const VideoEncoder::EncoderInfo primary_info_;
|
|
|
|
const VideoEncoder::EncoderInfo fallback_info_;
|
2020-08-14 19:58:22 +03:00
|
|
|
};
|
|
|
|
|
2021-06-25 03:43:10 +03:00
|
|
|
class StreamContext : public EncodedImageCallback {
|
|
|
|
public:
|
|
|
|
StreamContext(SimulcastEncoderAdapter* parent,
|
|
|
|
std::unique_ptr<EncoderContext> encoder_context,
|
|
|
|
std::unique_ptr<FramerateController> framerate_controller,
|
|
|
|
int stream_idx,
|
|
|
|
uint16_t width,
|
|
|
|
uint16_t height,
|
|
|
|
bool send_stream);
|
|
|
|
StreamContext(StreamContext&& rhs);
|
|
|
|
StreamContext& operator=(StreamContext&&) = delete;
|
|
|
|
~StreamContext() override;
|
|
|
|
|
|
|
|
Result OnEncodedImage(
|
|
|
|
const EncodedImage& encoded_image,
|
|
|
|
const CodecSpecificInfo* codec_specific_info) override;
|
|
|
|
void OnDroppedFrame(DropReason reason) override;
|
|
|
|
|
|
|
|
VideoEncoder& encoder() { return encoder_context_->encoder(); }
|
|
|
|
const VideoEncoder& encoder() const { return encoder_context_->encoder(); }
|
|
|
|
int stream_idx() const { return stream_idx_; }
|
|
|
|
uint16_t width() const { return width_; }
|
|
|
|
uint16_t height() const { return height_; }
|
|
|
|
bool is_keyframe_needed() const {
|
|
|
|
return !is_paused_ && is_keyframe_needed_;
|
|
|
|
}
|
|
|
|
void set_is_keyframe_needed() { is_keyframe_needed_ = true; }
|
|
|
|
bool is_paused() const { return is_paused_; }
|
|
|
|
void set_is_paused(bool is_paused) { is_paused_ = is_paused; }
|
2022-03-11 19:49:54 +03:00
|
|
|
absl::optional<double> target_fps() const {
|
2021-06-25 03:43:10 +03:00
|
|
|
return framerate_controller_ == nullptr
|
|
|
|
? absl::nullopt
|
2022-03-11 19:49:54 +03:00
|
|
|
: absl::optional<double>(
|
|
|
|
framerate_controller_->GetMaxFramerate());
|
2021-06-25 03:43:10 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
std::unique_ptr<EncoderContext> ReleaseEncoderContext() &&;
|
|
|
|
void OnKeyframe(Timestamp timestamp);
|
|
|
|
bool ShouldDropFrame(Timestamp timestamp);
|
|
|
|
|
|
|
|
private:
|
|
|
|
SimulcastEncoderAdapter* const parent_;
|
|
|
|
std::unique_ptr<EncoderContext> encoder_context_;
|
|
|
|
std::unique_ptr<FramerateController> framerate_controller_;
|
|
|
|
const int stream_idx_;
|
|
|
|
const uint16_t width_;
|
|
|
|
const uint16_t height_;
|
|
|
|
bool is_keyframe_needed_;
|
|
|
|
bool is_paused_;
|
2020-08-14 19:58:22 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
bool Initialized() const;
|
|
|
|
|
|
|
|
void DestroyStoredEncoders();
|
|
|
|
|
2022-03-11 19:49:54 +03:00
|
|
|
// This method creates encoder. May reuse previously created encoders from
|
|
|
|
// `cached_encoder_contexts_`. It's const because it's used from
|
|
|
|
// const GetEncoderInfo().
|
2021-06-25 03:43:10 +03:00
|
|
|
std::unique_ptr<EncoderContext> FetchOrCreateEncoderContext(
|
2022-03-11 19:49:54 +03:00
|
|
|
bool is_lowest_quality_stream) const;
|
2021-06-25 03:43:10 +03:00
|
|
|
|
|
|
|
webrtc::VideoCodec MakeStreamCodec(const webrtc::VideoCodec& codec,
|
|
|
|
int stream_idx,
|
|
|
|
uint32_t start_bitrate_kbps,
|
|
|
|
bool is_lowest_quality_stream,
|
|
|
|
bool is_highest_quality_stream);
|
|
|
|
|
|
|
|
EncodedImageCallback::Result OnEncodedImage(
|
|
|
|
size_t stream_idx,
|
|
|
|
const EncodedImage& encoded_image,
|
|
|
|
const CodecSpecificInfo* codec_specific_info);
|
|
|
|
|
|
|
|
void OnDroppedFrame(size_t stream_idx);
|
|
|
|
|
|
|
|
void OverrideFromFieldTrial(VideoEncoder::EncoderInfo* info) const;
|
|
|
|
|
2020-08-14 19:58:22 +03:00
|
|
|
volatile int inited_; // Accessed atomically.
|
|
|
|
VideoEncoderFactory* const primary_encoder_factory_;
|
|
|
|
VideoEncoderFactory* const fallback_encoder_factory_;
|
|
|
|
const SdpVideoFormat video_format_;
|
|
|
|
VideoCodec codec_;
|
2021-06-25 03:43:10 +03:00
|
|
|
int total_streams_count_;
|
|
|
|
bool bypass_mode_;
|
|
|
|
std::vector<StreamContext> stream_contexts_;
|
2020-08-14 19:58:22 +03:00
|
|
|
EncodedImageCallback* encoded_complete_callback_;
|
|
|
|
|
|
|
|
// Used for checking the single-threaded access of the encoder interface.
|
2021-06-25 03:43:10 +03:00
|
|
|
RTC_NO_UNIQUE_ADDRESS SequenceChecker encoder_queue_;
|
2020-08-14 19:58:22 +03:00
|
|
|
|
2022-03-11 19:49:54 +03:00
|
|
|
// Store previously created and released encoders , so they don't have to be
|
|
|
|
// recreated. Remaining encoders are destroyed by the destructor.
|
|
|
|
// Marked as `mutable` becuase we may need to temporarily create encoder in
|
|
|
|
// GetEncoderInfo(), which is const.
|
|
|
|
mutable std::list<std::unique_ptr<EncoderContext>> cached_encoder_contexts_;
|
2020-08-14 19:58:22 +03:00
|
|
|
|
|
|
|
const absl::optional<unsigned int> experimental_boosted_screenshare_qp_;
|
|
|
|
const bool boost_base_layer_quality_;
|
|
|
|
const bool prefer_temporal_support_on_base_layer_;
|
2021-06-25 03:43:10 +03:00
|
|
|
|
|
|
|
const SimulcastEncoderAdapterEncoderInfoSettings encoder_info_override_;
|
2020-08-14 19:58:22 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace webrtc
|
|
|
|
|
|
|
|
#endif // MEDIA_ENGINE_SIMULCAST_ENCODER_ADAPTER_H_
|