2014-11-01 06:10:48 +00: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.
|
|
|
|
|
*
|
|
|
|
|
*/
|
|
|
|
|
|
2017-09-15 06:47:31 +02:00
|
|
|
#ifndef MODULES_VIDEO_CODING_CODECS_VP9_VP9_IMPL_H_
|
|
|
|
|
#define MODULES_VIDEO_CODING_CODECS_VP9_VP9_IMPL_H_
|
2015-12-21 03:04:49 -08:00
|
|
|
|
2018-11-19 09:52:37 +01:00
|
|
|
#ifdef RTC_ENABLE_VP9
|
|
|
|
|
|
2018-05-28 12:26:36 +02:00
|
|
|
#include <map>
|
2016-02-29 05:51:59 -08:00
|
|
|
#include <memory>
|
2019-03-06 16:04:32 +01:00
|
|
|
#include <string>
|
2015-12-21 03:04:49 -08:00
|
|
|
#include <vector>
|
2014-11-01 06:10:48 +00:00
|
|
|
|
2019-06-28 15:19:43 +02:00
|
|
|
#include "api/fec_controller_override.h"
|
2019-06-11 14:57:57 +02:00
|
|
|
#include "api/video_codecs/video_encoder.h"
|
2018-06-28 10:59:02 -07:00
|
|
|
#include "media/base/vp9_profile.h"
|
2017-09-15 06:47:31 +02:00
|
|
|
#include "modules/video_coding/codecs/vp9/include/vp9.h"
|
|
|
|
|
#include "modules/video_coding/codecs/vp9/vp9_frame_buffer_pool.h"
|
2018-08-29 01:05:26 +02:00
|
|
|
#include "modules/video_coding/utility/framerate_controller.h"
|
2017-01-26 13:23:44 -08:00
|
|
|
#include "vpx/vp8cx.h"
|
2014-12-17 13:43:55 +00:00
|
|
|
#include "vpx/vpx_decoder.h"
|
|
|
|
|
#include "vpx/vpx_encoder.h"
|
2014-11-01 06:10:48 +00:00
|
|
|
|
|
|
|
|
namespace webrtc {
|
|
|
|
|
|
|
|
|
|
class VP9EncoderImpl : public VP9Encoder {
|
|
|
|
|
public:
|
2018-06-28 10:59:02 -07:00
|
|
|
explicit VP9EncoderImpl(const cricket::VideoCodec& codec);
|
2014-11-01 06:10:48 +00:00
|
|
|
|
2019-06-28 15:19:43 +02:00
|
|
|
~VP9EncoderImpl() override;
|
|
|
|
|
|
|
|
|
|
void SetFecControllerOverride(
|
|
|
|
|
FecControllerOverride* fec_controller_override) override;
|
2014-11-01 06:10:48 +00:00
|
|
|
|
2015-03-04 12:58:35 +00:00
|
|
|
int Release() override;
|
2014-11-01 06:10:48 +00:00
|
|
|
|
2015-03-04 12:58:35 +00:00
|
|
|
int InitEncode(const VideoCodec* codec_settings,
|
2019-06-11 14:57:57 +02:00
|
|
|
const Settings& settings) override;
|
2014-11-01 06:10:48 +00:00
|
|
|
|
2015-05-29 17:21:40 -07:00
|
|
|
int Encode(const VideoFrame& input_image,
|
2019-03-07 10:18:23 +01:00
|
|
|
const std::vector<VideoFrameType>* frame_types) override;
|
2014-11-01 06:10:48 +00:00
|
|
|
|
2015-03-04 12:58:35 +00:00
|
|
|
int RegisterEncodeCompleteCallback(EncodedImageCallback* callback) override;
|
2014-11-01 06:10:48 +00:00
|
|
|
|
2019-04-12 13:59:09 +02:00
|
|
|
void SetRates(const RateControlParameters& parameters) override;
|
2014-11-01 06:10:48 +00:00
|
|
|
|
2018-11-07 16:54:15 +01:00
|
|
|
EncoderInfo GetEncoderInfo() const override;
|
2015-12-18 16:01:11 +01:00
|
|
|
|
2014-11-01 06:10:48 +00:00
|
|
|
private:
|
2015-01-26 15:21:36 +00:00
|
|
|
// Determine number of encoder threads to use.
|
|
|
|
|
int NumberOfThreads(int width, int height, int number_of_cores);
|
|
|
|
|
|
2014-11-01 06:10:48 +00:00
|
|
|
// Call encoder initialize function and set control settings.
|
|
|
|
|
int InitAndSetControlSettings(const VideoCodec* inst);
|
|
|
|
|
|
|
|
|
|
void PopulateCodecSpecific(CodecSpecificInfo* codec_specific,
|
2018-08-27 15:33:42 +02:00
|
|
|
absl::optional<int>* spatial_idx,
|
2014-11-01 06:10:48 +00:00
|
|
|
const vpx_codec_cx_pkt& pkt,
|
2018-11-23 13:59:24 +01:00
|
|
|
uint32_t timestamp);
|
2018-05-28 12:26:36 +02:00
|
|
|
void FillReferenceIndices(const vpx_codec_cx_pkt& pkt,
|
|
|
|
|
const size_t pic_num,
|
|
|
|
|
const bool inter_layer_predicted,
|
|
|
|
|
CodecSpecificInfoVP9* vp9_info);
|
|
|
|
|
void UpdateReferenceBuffers(const vpx_codec_cx_pkt& pkt,
|
|
|
|
|
const size_t pic_num);
|
2018-12-04 15:54:52 +01:00
|
|
|
vpx_svc_ref_frame_config_t SetReferences(
|
|
|
|
|
bool is_key_pic,
|
|
|
|
|
size_t first_active_spatial_layer_id);
|
2014-11-01 06:10:48 +00:00
|
|
|
|
2015-11-02 07:23:20 -08:00
|
|
|
bool ExplicitlyConfiguredSpatialLayers() const;
|
2018-04-23 12:32:22 +02:00
|
|
|
bool SetSvcRates(const VideoBitrateAllocation& bitrate_allocation);
|
2015-07-31 06:10:09 -07:00
|
|
|
|
|
|
|
|
virtual int GetEncodedLayerFrame(const vpx_codec_cx_pkt* pkt);
|
|
|
|
|
|
|
|
|
|
// Callback function for outputting packets per spatial layer.
|
|
|
|
|
static void EncoderOutputCodedPacketCallback(vpx_codec_cx_pkt* pkt,
|
|
|
|
|
void* user_data);
|
2014-11-01 06:10:48 +00:00
|
|
|
|
2018-04-24 21:29:14 +02:00
|
|
|
void DeliverBufferedFrame(bool end_of_picture);
|
2018-04-04 11:45:41 +02:00
|
|
|
|
2018-08-29 01:05:26 +02:00
|
|
|
bool DropFrame(uint8_t spatial_idx, uint32_t rtp_timestamp);
|
2018-05-17 16:46:43 +02:00
|
|
|
|
2014-11-01 06:10:48 +00:00
|
|
|
// Determine maximum target for Intra frames
|
|
|
|
|
//
|
|
|
|
|
// Input:
|
|
|
|
|
// - optimal_buffer_size : Optimal buffer size
|
|
|
|
|
// Return Value : Max target size for Intra frames represented as
|
|
|
|
|
// percentage of the per frame bandwidth
|
|
|
|
|
uint32_t MaxIntraTarget(uint32_t optimal_buffer_size);
|
|
|
|
|
|
2019-03-06 16:04:32 +01:00
|
|
|
size_t SteadyStateSize(int sid, int tid);
|
|
|
|
|
|
2014-11-01 06:10:48 +00:00
|
|
|
EncodedImage encoded_image_;
|
2018-04-04 11:45:41 +02:00
|
|
|
CodecSpecificInfo codec_specific_;
|
2014-11-01 06:10:48 +00:00
|
|
|
EncodedImageCallback* encoded_complete_callback_;
|
|
|
|
|
VideoCodec codec_;
|
2018-06-28 10:59:02 -07:00
|
|
|
const VP9Profile profile_;
|
2014-11-01 06:10:48 +00:00
|
|
|
bool inited_;
|
|
|
|
|
int64_t timestamp_;
|
|
|
|
|
int cpu_speed_;
|
|
|
|
|
uint32_t rc_max_intra_target_;
|
|
|
|
|
vpx_codec_ctx_t* encoder_;
|
|
|
|
|
vpx_codec_enc_cfg_t* config_;
|
|
|
|
|
vpx_image_t* raw_;
|
2017-01-26 13:23:44 -08:00
|
|
|
vpx_svc_extra_cfg_t svc_params_;
|
2015-07-31 06:10:09 -07:00
|
|
|
const VideoFrame* input_image_;
|
2018-05-28 12:26:36 +02:00
|
|
|
GofInfoVP9 gof_; // Contains each frame's temporal information for
|
|
|
|
|
// non-flexible mode.
|
2018-05-03 14:14:09 +02:00
|
|
|
bool force_key_frame_;
|
2018-04-26 11:03:49 +02:00
|
|
|
size_t pics_since_key_;
|
2015-07-31 06:10:09 -07:00
|
|
|
uint8_t num_temporal_layers_;
|
2018-06-21 11:07:21 -05:00
|
|
|
uint8_t num_spatial_layers_; // Number of configured SLs
|
|
|
|
|
uint8_t num_active_spatial_layers_; // Number of actively encoded SLs
|
2019-09-25 09:10:33 +00:00
|
|
|
uint8_t first_active_layer_;
|
2018-10-03 18:04:57 +02:00
|
|
|
bool layer_deactivation_requires_key_frame_;
|
2018-05-28 12:26:36 +02:00
|
|
|
bool is_svc_;
|
2018-04-26 11:03:49 +02:00
|
|
|
InterLayerPredMode inter_layer_pred_;
|
2018-10-01 17:13:50 +02:00
|
|
|
bool external_ref_control_;
|
2018-11-08 16:56:43 +01:00
|
|
|
const bool trusted_rate_controller_;
|
2019-04-15 12:22:55 +02:00
|
|
|
const bool dynamic_rate_settings_;
|
2019-09-09 10:16:18 +02:00
|
|
|
bool layer_buffering_;
|
2018-11-23 13:59:24 +01:00
|
|
|
const bool full_superframe_drop_;
|
2019-05-24 16:50:00 +02:00
|
|
|
vpx_svc_frame_drop_t svc_drop_frame_;
|
2018-11-23 13:59:24 +01:00
|
|
|
bool first_frame_in_picture_;
|
2018-12-04 15:54:52 +01:00
|
|
|
VideoBitrateAllocation current_bitrate_allocation_;
|
|
|
|
|
bool ss_info_needed_;
|
2019-09-25 09:10:33 +00:00
|
|
|
bool force_all_active_layers_;
|
Reland of Work on flexible mode and screen sharing. (patchset #1 id:1 of https://codereview.webrtc.org/1438543002/ )
Reason for revert:
Failed test not related to this CL (test fails on
master at an earlier date), re-landing original CL..
(This time from my @webrtc account.)
Original issue's description:
> Revert of Work on flexible mode and screen sharing. (patchset #28 id:520001 of https://codereview.webrtc.org/1328113004/ )
>
> Reason for revert:
> Seems to break VideoSendStreamTest.ReconfigureBitratesSetsEncoderBitratesCorrectly on Linux Memcheck buildbot.
>
> Original issue's description:
> > Work on flexible mode and screen sharing.
> >
> > Implement VP8 style screensharing but with spatial layers.
> > Implement flexible mode.
> >
> > Files from other patches:
> > generic_encoder.cc
> > layer_filtering_transport.cc
> >
> > BUG=webrtc:4914
> >
> > Committed: https://crrev.com/77ccfb4d16c148e61a316746bb5d9705e8b39f4a
> > Cr-Commit-Position: refs/heads/master@{#10572}
>
> TBR=sprang@webrtc.org,stefan@webrtc.org,philipel@google.com,asapersson@webrtc.org,mflodman@webrtc.org,philipel@webrtc.org
> NOPRESUBMIT=true
> NOTREECHECKS=true
> NOTRY=true
> BUG=webrtc:4914
>
> Committed: https://crrev.com/0be8f1d347bdb171462df89c2a4c69b3f3eb7519
> Cr-Commit-Position: refs/heads/master@{#10578}
TBR=sprang@webrtc.org,stefan@webrtc.org,philipel@google.com,asapersson@webrtc.org,mflodman@webrtc.org,terelius@webrtc.org
NOPRESUBMIT=true
NOTREECHECKS=true
NOTRY=true
BUG=webrtc:4914
Review URL: https://codereview.webrtc.org/1431283002
Cr-Commit-Position: refs/heads/master@{#10581}
2015-11-10 07:17:23 -08:00
|
|
|
|
2018-09-05 21:07:17 +02:00
|
|
|
std::vector<FramerateController> framerate_controller_;
|
2018-05-18 17:31:19 +02:00
|
|
|
|
Reland of Work on flexible mode and screen sharing. (patchset #1 id:1 of https://codereview.webrtc.org/1438543002/ )
Reason for revert:
Failed test not related to this CL (test fails on
master at an earlier date), re-landing original CL..
(This time from my @webrtc account.)
Original issue's description:
> Revert of Work on flexible mode and screen sharing. (patchset #28 id:520001 of https://codereview.webrtc.org/1328113004/ )
>
> Reason for revert:
> Seems to break VideoSendStreamTest.ReconfigureBitratesSetsEncoderBitratesCorrectly on Linux Memcheck buildbot.
>
> Original issue's description:
> > Work on flexible mode and screen sharing.
> >
> > Implement VP8 style screensharing but with spatial layers.
> > Implement flexible mode.
> >
> > Files from other patches:
> > generic_encoder.cc
> > layer_filtering_transport.cc
> >
> > BUG=webrtc:4914
> >
> > Committed: https://crrev.com/77ccfb4d16c148e61a316746bb5d9705e8b39f4a
> > Cr-Commit-Position: refs/heads/master@{#10572}
>
> TBR=sprang@webrtc.org,stefan@webrtc.org,philipel@google.com,asapersson@webrtc.org,mflodman@webrtc.org,philipel@webrtc.org
> NOPRESUBMIT=true
> NOTREECHECKS=true
> NOTRY=true
> BUG=webrtc:4914
>
> Committed: https://crrev.com/0be8f1d347bdb171462df89c2a4c69b3f3eb7519
> Cr-Commit-Position: refs/heads/master@{#10578}
TBR=sprang@webrtc.org,stefan@webrtc.org,philipel@google.com,asapersson@webrtc.org,mflodman@webrtc.org,terelius@webrtc.org
NOPRESUBMIT=true
NOTREECHECKS=true
NOTRY=true
BUG=webrtc:4914
Review URL: https://codereview.webrtc.org/1431283002
Cr-Commit-Position: refs/heads/master@{#10581}
2015-11-10 07:17:23 -08:00
|
|
|
// Used for flexible mode.
|
|
|
|
|
bool is_flexible_mode_;
|
2018-05-28 12:26:36 +02:00
|
|
|
struct RefFrameBuffer {
|
|
|
|
|
RefFrameBuffer(size_t pic_num,
|
|
|
|
|
size_t spatial_layer_id,
|
|
|
|
|
size_t temporal_layer_id)
|
|
|
|
|
: pic_num(pic_num),
|
|
|
|
|
spatial_layer_id(spatial_layer_id),
|
|
|
|
|
temporal_layer_id(temporal_layer_id) {}
|
|
|
|
|
RefFrameBuffer() {}
|
2018-08-27 14:54:20 +02:00
|
|
|
|
|
|
|
|
bool operator==(const RefFrameBuffer& o) {
|
|
|
|
|
return pic_num == o.pic_num && spatial_layer_id == o.spatial_layer_id &&
|
|
|
|
|
temporal_layer_id == o.temporal_layer_id;
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-28 12:26:36 +02:00
|
|
|
size_t pic_num = 0;
|
|
|
|
|
size_t spatial_layer_id = 0;
|
|
|
|
|
size_t temporal_layer_id = 0;
|
|
|
|
|
};
|
|
|
|
|
std::map<size_t, RefFrameBuffer> ref_buf_;
|
2019-03-06 16:04:32 +01:00
|
|
|
|
|
|
|
|
// Variable frame-rate related fields and methods.
|
|
|
|
|
const struct VariableFramerateExperiment {
|
|
|
|
|
bool enabled;
|
|
|
|
|
// Framerate is limited to this value in steady state.
|
|
|
|
|
float framerate_limit;
|
|
|
|
|
// This qp or below is considered a steady state.
|
|
|
|
|
int steady_state_qp;
|
|
|
|
|
// Frames of at least this percentage below ideal for configured bitrate are
|
|
|
|
|
// considered in a steady state.
|
|
|
|
|
int steady_state_undershoot_percentage;
|
|
|
|
|
// Number of consecutive frames with good QP and size required to detect
|
|
|
|
|
// the steady state.
|
|
|
|
|
int frames_before_steady_state;
|
|
|
|
|
} variable_framerate_experiment_;
|
|
|
|
|
static VariableFramerateExperiment ParseVariableFramerateConfig(
|
|
|
|
|
std::string group_name);
|
|
|
|
|
FramerateController variable_framerate_controller_;
|
2020-06-18 19:16:53 +02:00
|
|
|
|
|
|
|
|
const struct QualityScalerExperiment {
|
|
|
|
|
int low_qp;
|
|
|
|
|
int high_qp;
|
|
|
|
|
bool enabled;
|
|
|
|
|
} quality_scaler_experiment_;
|
|
|
|
|
static QualityScalerExperiment ParseQualityScalerConfig(
|
|
|
|
|
std::string group_name);
|
|
|
|
|
|
2019-03-06 16:04:32 +01:00
|
|
|
int num_steady_state_frames_;
|
2020-02-21 10:47:10 -08:00
|
|
|
// Only set config when this flag is set.
|
|
|
|
|
bool config_changed_;
|
2014-11-01 06:10:48 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
class VP9DecoderImpl : public VP9Decoder {
|
|
|
|
|
public:
|
|
|
|
|
VP9DecoderImpl();
|
|
|
|
|
|
|
|
|
|
virtual ~VP9DecoderImpl();
|
|
|
|
|
|
2015-03-04 12:58:35 +00:00
|
|
|
int InitDecode(const VideoCodec* inst, int number_of_cores) override;
|
2014-11-01 06:10:48 +00:00
|
|
|
|
2015-03-04 12:58:35 +00:00
|
|
|
int Decode(const EncodedImage& input_image,
|
|
|
|
|
bool missing_frames,
|
|
|
|
|
int64_t /*render_time_ms*/) override;
|
2014-11-01 06:10:48 +00:00
|
|
|
|
2015-03-04 12:58:35 +00:00
|
|
|
int RegisterDecodeCompleteCallback(DecodedImageCallback* callback) override;
|
2014-11-01 06:10:48 +00:00
|
|
|
|
2015-03-04 12:58:35 +00:00
|
|
|
int Release() override;
|
2014-11-01 06:10:48 +00:00
|
|
|
|
2015-12-18 16:01:11 +01:00
|
|
|
const char* ImplementationName() const override;
|
|
|
|
|
|
2014-11-01 06:10:48 +00:00
|
|
|
private:
|
2019-07-12 11:20:47 +02:00
|
|
|
int ReturnFrame(const vpx_image_t* img,
|
|
|
|
|
uint32_t timestamp,
|
|
|
|
|
int qp,
|
|
|
|
|
const webrtc::ColorSpace* explicit_color_space);
|
2014-11-01 06:10:48 +00:00
|
|
|
|
2015-05-06 10:42:15 +02:00
|
|
|
// Memory pool used to share buffers between libvpx and webrtc.
|
|
|
|
|
Vp9FrameBufferPool frame_buffer_pool_;
|
2014-11-01 06:10:48 +00:00
|
|
|
DecodedImageCallback* decode_complete_callback_;
|
|
|
|
|
bool inited_;
|
2014-12-17 13:43:55 +00:00
|
|
|
vpx_codec_ctx_t* decoder_;
|
2014-11-01 06:10:48 +00:00
|
|
|
bool key_frame_required_;
|
Reland "VP9 decoder: Sets thread count based on resolution, reinit on change."
This is a reland of d5925756980f6e82a55f57532c8d855e954459fb
Patchset 2 is a reland of
https://webrtc-review.googlesource.com/c/src/+/177012
Patchset 3 is a fix for a potential crash when InitDecode()is called from
VideoStreamDecoderImpl::GetDecoder(), where the decoder_settings
parameter is a but surprisingly set to nullptr.
Original change's description:
> VP9 decoder: Sets thread count based on resolution, reinit on change.
>
> Previously, number of decoder threads for VP9 were always set to 8 but
> with a cap at number of cores. This was done since we "can't know" the
> resolution that will be used.
>
> With this change, we now intialize the number of threads based on
> resolution given in InitDecode(). If a resolution change happens in
> flight, it requires a keyframe. We therefore parse the header from
> any key frame and if it has a new resolution, we re-initialize the
> decoder.
>
> The number of threads used is based on pixel count. We set one thread
> as target for 1280x720, and scale up lineraly from there. The 8-thread
> cap is gone, but still limit it core count.
>
> This means for instance: 1 <= 720p, 2 for 1080p, 4 for 1440p, 9 for 4K.
>
> Bug: webrtc:11551
> Change-Id: I14c169a6c651c50bd1b870c4b22bc4495c8448fd
> Reviewed-on: https://webrtc-review.googlesource.com/c/src/+/174460
> Commit-Queue: Erik Språng <sprang@webrtc.org>
> Reviewed-by: Ilya Nikolaevskiy <ilnik@webrtc.org>
> Cr-Commit-Position: refs/heads/master@{#31507}
Bug: webrtc:11551
Change-Id: I2b4b146d0b8319f07ce1660202d6aa4b374eb015
Reviewed-on: https://webrtc-review.googlesource.com/c/src/+/177246
Reviewed-by: Johannes Kron <kron@webrtc.org>
Commit-Queue: Erik Språng <sprang@webrtc.org>
Cr-Commit-Position: refs/heads/master@{#31527}
2020-06-15 16:52:13 +02:00
|
|
|
VideoCodec current_codec_;
|
|
|
|
|
int num_cores_;
|
2014-11-01 06:10:48 +00:00
|
|
|
};
|
|
|
|
|
} // namespace webrtc
|
|
|
|
|
|
2018-11-19 09:52:37 +01:00
|
|
|
#endif // RTC_ENABLE_VP9
|
|
|
|
|
|
2017-09-15 06:47:31 +02:00
|
|
|
#endif // MODULES_VIDEO_CODING_CODECS_VP9_VP9_IMPL_H_
|