blob: 7009311072fc0ae2cd1d88fb60f65c2bbf8e9393 [file] [log] [blame]
marpan@webrtc.org5b883172014-11-01 06:10:48 +00001/*
2 * Copyright (c) 2014 The WebRTC project authors. All Rights Reserved.
3 *
4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 *
10 */
11
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020012#ifndef MODULES_VIDEO_CODING_CODECS_VP9_VP9_IMPL_H_
13#define MODULES_VIDEO_CODING_CODECS_VP9_VP9_IMPL_H_
philipelcce46fc2015-12-21 03:04:49 -080014
Sergey Silkin4e6cd5e2018-05-28 12:26:36 +020015#include <map>
kwiberg3f55dea2016-02-29 05:51:59 -080016#include <memory>
philipelcce46fc2015-12-21 03:04:49 -080017#include <vector>
marpan@webrtc.org5b883172014-11-01 06:10:48 +000018
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020019#include "modules/video_coding/codecs/vp9/include/vp9.h"
Emircan Uysaler98badbc2018-06-28 10:59:02 -070020
21#include "media/base/vp9_profile.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020022#include "modules/video_coding/codecs/vp9/vp9_frame_buffer_pool.h"
Sergey Silkind902d582018-05-18 17:31:19 +020023#include "rtc_base/rate_statistics.h"
marpan@webrtc.org5b883172014-11-01 06:10:48 +000024
johannkoenig8225c402017-01-26 13:23:44 -080025#include "vpx/vp8cx.h"
pbos@webrtc.orge728ee02014-12-17 13:43:55 +000026#include "vpx/vpx_decoder.h"
27#include "vpx/vpx_encoder.h"
marpan@webrtc.org5b883172014-11-01 06:10:48 +000028
29namespace webrtc {
30
31class VP9EncoderImpl : public VP9Encoder {
32 public:
Emircan Uysaler98badbc2018-06-28 10:59:02 -070033 explicit VP9EncoderImpl(const cricket::VideoCodec& codec);
marpan@webrtc.org5b883172014-11-01 06:10:48 +000034
35 virtual ~VP9EncoderImpl();
36
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +000037 int Release() override;
marpan@webrtc.org5b883172014-11-01 06:10:48 +000038
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +000039 int InitEncode(const VideoCodec* codec_settings,
40 int number_of_cores,
41 size_t max_payload_size) override;
marpan@webrtc.org5b883172014-11-01 06:10:48 +000042
Miguel Casas-Sanchez47650702015-05-29 17:21:40 -070043 int Encode(const VideoFrame& input_image,
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +000044 const CodecSpecificInfo* codec_specific_info,
pbos22993e12015-10-19 02:39:06 -070045 const std::vector<FrameType>* frame_types) override;
marpan@webrtc.org5b883172014-11-01 06:10:48 +000046
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +000047 int RegisterEncodeCompleteCallback(EncodedImageCallback* callback) override;
marpan@webrtc.org5b883172014-11-01 06:10:48 +000048
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +000049 int SetChannelParameters(uint32_t packet_loss, int64_t rtt) override;
marpan@webrtc.org5b883172014-11-01 06:10:48 +000050
Erik Språng566124a2018-04-23 12:32:22 +020051 int SetRateAllocation(const VideoBitrateAllocation& bitrate_allocation,
Erik Språng08127a92016-11-16 16:41:30 +010052 uint32_t frame_rate) override;
marpan@webrtc.org5b883172014-11-01 06:10:48 +000053
Peter Boströmb7d9a972015-12-18 16:01:11 +010054 const char* ImplementationName() const override;
55
marpan@webrtc.org5b883172014-11-01 06:10:48 +000056 private:
marpan@webrtc.org38d11b82015-01-26 15:21:36 +000057 // Determine number of encoder threads to use.
58 int NumberOfThreads(int width, int height, int number_of_cores);
59
marpan@webrtc.org5b883172014-11-01 06:10:48 +000060 // Call encoder initialize function and set control settings.
61 int InitAndSetControlSettings(const VideoCodec* inst);
62
63 void PopulateCodecSpecific(CodecSpecificInfo* codec_specific,
64 const vpx_codec_cx_pkt& pkt,
Sergey Silkin07f80cc2018-04-09 13:11:59 +020065 uint32_t timestamp,
66 bool first_frame_in_picture);
Sergey Silkin4e6cd5e2018-05-28 12:26:36 +020067 void FillReferenceIndices(const vpx_codec_cx_pkt& pkt,
68 const size_t pic_num,
69 const bool inter_layer_predicted,
70 CodecSpecificInfoVP9* vp9_info);
71 void UpdateReferenceBuffers(const vpx_codec_cx_pkt& pkt,
72 const size_t pic_num);
marpan@webrtc.org5b883172014-11-01 06:10:48 +000073
sprangce4aef12015-11-02 07:23:20 -080074 bool ExplicitlyConfiguredSpatialLayers() const;
Erik Språng566124a2018-04-23 12:32:22 +020075 bool SetSvcRates(const VideoBitrateAllocation& bitrate_allocation);
asaperssona9455ab2015-07-31 06:10:09 -070076
77 virtual int GetEncodedLayerFrame(const vpx_codec_cx_pkt* pkt);
78
79 // Callback function for outputting packets per spatial layer.
80 static void EncoderOutputCodedPacketCallback(vpx_codec_cx_pkt* pkt,
81 void* user_data);
marpan@webrtc.org5b883172014-11-01 06:10:48 +000082
Sergey Silkinbc0f0d32018-04-24 21:29:14 +020083 void DeliverBufferedFrame(bool end_of_picture);
Sergey Silkin2a1f1832018-04-04 11:45:41 +020084
Sergey Silkinbe71a1e2018-05-17 16:46:43 +020085 bool DropFrame(uint32_t rtp_timestamp);
86
marpan@webrtc.org5b883172014-11-01 06:10:48 +000087 // Determine maximum target for Intra frames
88 //
89 // Input:
90 // - optimal_buffer_size : Optimal buffer size
91 // Return Value : Max target size for Intra frames represented as
92 // percentage of the per frame bandwidth
93 uint32_t MaxIntraTarget(uint32_t optimal_buffer_size);
94
95 EncodedImage encoded_image_;
Sergey Silkin2a1f1832018-04-04 11:45:41 +020096 CodecSpecificInfo codec_specific_;
marpan@webrtc.org5b883172014-11-01 06:10:48 +000097 EncodedImageCallback* encoded_complete_callback_;
98 VideoCodec codec_;
Emircan Uysaler98badbc2018-06-28 10:59:02 -070099 const VP9Profile profile_;
marpan@webrtc.org5b883172014-11-01 06:10:48 +0000100 bool inited_;
101 int64_t timestamp_;
marpan@webrtc.org5b883172014-11-01 06:10:48 +0000102 int cpu_speed_;
103 uint32_t rc_max_intra_target_;
104 vpx_codec_ctx_t* encoder_;
105 vpx_codec_enc_cfg_t* config_;
106 vpx_image_t* raw_;
johannkoenig8225c402017-01-26 13:23:44 -0800107 vpx_svc_extra_cfg_t svc_params_;
asaperssona9455ab2015-07-31 06:10:09 -0700108 const VideoFrame* input_image_;
Sergey Silkin4e6cd5e2018-05-28 12:26:36 +0200109 GofInfoVP9 gof_; // Contains each frame's temporal information for
110 // non-flexible mode.
Sergey Silkinbd0954e2018-05-03 14:14:09 +0200111 bool force_key_frame_;
Sergey Silkin6a8f30e2018-04-26 11:03:49 +0200112 size_t pics_since_key_;
asaperssona9455ab2015-07-31 06:10:09 -0700113 uint8_t num_temporal_layers_;
“Michael23c5a992018-06-21 11:07:21 -0500114 uint8_t num_spatial_layers_; // Number of configured SLs
115 uint8_t num_active_spatial_layers_; // Number of actively encoded SLs
Sergey Silkin4e6cd5e2018-05-28 12:26:36 +0200116 bool is_svc_;
Sergey Silkin6a8f30e2018-04-26 11:03:49 +0200117 InterLayerPredMode inter_layer_pred_;
philipelcfc319b2015-11-10 07:17:23 -0800118
Sergey Silkind902d582018-05-18 17:31:19 +0200119 // Framerate controller.
Danil Chapovalov0040b662018-06-18 10:48:16 +0200120 absl::optional<float> target_framerate_fps_;
Sergey Silkind902d582018-05-18 17:31:19 +0200121 RateStatistics output_framerate_;
122 uint32_t last_encoded_frame_rtp_timestamp_;
123
philipelcfc319b2015-11-10 07:17:23 -0800124 // Used for flexible mode.
125 bool is_flexible_mode_;
Sergey Silkin4e6cd5e2018-05-28 12:26:36 +0200126 struct RefFrameBuffer {
127 RefFrameBuffer(size_t pic_num,
128 size_t spatial_layer_id,
129 size_t temporal_layer_id)
130 : pic_num(pic_num),
131 spatial_layer_id(spatial_layer_id),
132 temporal_layer_id(temporal_layer_id) {}
133 RefFrameBuffer() {}
134 size_t pic_num = 0;
135 size_t spatial_layer_id = 0;
136 size_t temporal_layer_id = 0;
137 };
138 std::map<size_t, RefFrameBuffer> ref_buf_;
marpan@webrtc.org5b883172014-11-01 06:10:48 +0000139};
140
marpan@webrtc.org5b883172014-11-01 06:10:48 +0000141class VP9DecoderImpl : public VP9Decoder {
142 public:
143 VP9DecoderImpl();
144
145 virtual ~VP9DecoderImpl();
146
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000147 int InitDecode(const VideoCodec* inst, int number_of_cores) override;
marpan@webrtc.org5b883172014-11-01 06:10:48 +0000148
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000149 int Decode(const EncodedImage& input_image,
150 bool missing_frames,
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000151 const CodecSpecificInfo* codec_specific_info,
152 int64_t /*render_time_ms*/) override;
marpan@webrtc.org5b883172014-11-01 06:10:48 +0000153
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000154 int RegisterDecodeCompleteCallback(DecodedImageCallback* callback) override;
marpan@webrtc.org5b883172014-11-01 06:10:48 +0000155
kjellander@webrtc.org14665ff2015-03-04 12:58:35 +0000156 int Release() override;
marpan@webrtc.org5b883172014-11-01 06:10:48 +0000157
Peter Boströmb7d9a972015-12-18 16:01:11 +0100158 const char* ImplementationName() const override;
159
marpan@webrtc.org5b883172014-11-01 06:10:48 +0000160 private:
asapersson1490f7a2016-09-23 02:09:46 -0700161 int ReturnFrame(const vpx_image_t* img,
162 uint32_t timestamp,
sakal7adadb12017-02-23 02:54:57 -0800163 int64_t ntp_time_ms,
164 int qp);
marpan@webrtc.org5b883172014-11-01 06:10:48 +0000165
Henrik Boström9695d852015-05-06 10:42:15 +0200166 // Memory pool used to share buffers between libvpx and webrtc.
167 Vp9FrameBufferPool frame_buffer_pool_;
marpan@webrtc.org5b883172014-11-01 06:10:48 +0000168 DecodedImageCallback* decode_complete_callback_;
169 bool inited_;
pbos@webrtc.orge728ee02014-12-17 13:43:55 +0000170 vpx_codec_ctx_t* decoder_;
marpan@webrtc.org5b883172014-11-01 06:10:48 +0000171 bool key_frame_required_;
172};
173} // namespace webrtc
174
Mirko Bonadei92ea95e2017-09-15 06:47:31 +0200175#endif // MODULES_VIDEO_CODING_CODECS_VP9_VP9_IMPL_H_