niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 1 | /* |
sprang | 3911c26 | 2016-04-15 01:24:14 -0700 | [diff] [blame] | 2 | * Copyright (c) 2012 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 | */ |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 10 | |
philipel | 9d3ab61 | 2015-12-21 04:12:39 -0800 | [diff] [blame] | 11 | #include "webrtc/modules/video_coding/generic_encoder.h" |
| 12 | |
| 13 | #include <vector> |
| 14 | |
Guo-wei Shieh | 2c37078 | 2015-04-08 13:00:10 -0700 | [diff] [blame] | 15 | #include "webrtc/base/checks.h" |
pbos | 854e84c | 2015-11-16 16:39:06 -0800 | [diff] [blame] | 16 | #include "webrtc/base/logging.h" |
pbos | d9eec76 | 2015-11-17 06:03:43 -0800 | [diff] [blame] | 17 | #include "webrtc/base/trace_event.h" |
pbos@webrtc.org | a440732 | 2013-07-16 12:32:05 +0000 | [diff] [blame] | 18 | #include "webrtc/engine_configurations.h" |
Henrik Kjellander | 2557b86 | 2015-11-18 22:00:21 +0100 | [diff] [blame] | 19 | #include "webrtc/modules/video_coding/encoded_frame.h" |
Henrik Kjellander | 2557b86 | 2015-11-18 22:00:21 +0100 | [diff] [blame] | 20 | #include "webrtc/modules/video_coding/media_optimization.h" |
Henrik Kjellander | 98f5351 | 2015-10-28 18:17:40 +0100 | [diff] [blame] | 21 | #include "webrtc/system_wrappers/include/critical_section_wrapper.h" |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 22 | |
| 23 | namespace webrtc { |
Peter Boström | 4f5db11 | 2015-10-29 16:53:59 +0100 | [diff] [blame] | 24 | VCMGenericEncoder::VCMGenericEncoder( |
| 25 | VideoEncoder* encoder, |
| 26 | VideoEncoderRateObserver* rate_observer, |
| 27 | VCMEncodedFrameCallback* encoded_frame_callback, |
sprang | 3911c26 | 2016-04-15 01:24:14 -0700 | [diff] [blame] | 28 | bool internal_source) |
pbos@webrtc.org | 891d483 | 2015-02-26 13:15:22 +0000 | [diff] [blame] | 29 | : encoder_(encoder), |
| 30 | rate_observer_(rate_observer), |
Peter Boström | 4f5db11 | 2015-10-29 16:53:59 +0100 | [diff] [blame] | 31 | vcm_encoded_frame_callback_(encoded_frame_callback), |
sprang | 3911c26 | 2016-04-15 01:24:14 -0700 | [diff] [blame] | 32 | internal_source_(internal_source), |
Peter Boström | 4f5db11 | 2015-10-29 16:53:59 +0100 | [diff] [blame] | 33 | encoder_params_({0, 0, 0, 0}), |
Peter Boström | 69ccb33 | 2015-10-29 16:30:23 +0100 | [diff] [blame] | 34 | is_screenshare_(false) {} |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 35 | |
Peter Boström | 4f5db11 | 2015-10-29 16:53:59 +0100 | [diff] [blame] | 36 | VCMGenericEncoder::~VCMGenericEncoder() {} |
| 37 | |
| 38 | int32_t VCMGenericEncoder::Release() { |
Peter Boström | 4fd6cda | 2016-01-26 10:19:53 +0100 | [diff] [blame] | 39 | TRACE_EVENT0("webrtc", "VCMGenericEncoder::Release"); |
Peter Boström | 4f5db11 | 2015-10-29 16:53:59 +0100 | [diff] [blame] | 40 | return encoder_->Release(); |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 41 | } |
| 42 | |
Peter Boström | 4f5db11 | 2015-10-29 16:53:59 +0100 | [diff] [blame] | 43 | int32_t VCMGenericEncoder::InitEncode(const VideoCodec* settings, |
sprang | 3911c26 | 2016-04-15 01:24:14 -0700 | [diff] [blame] | 44 | int32_t number_of_cores, |
| 45 | size_t max_payload_size) { |
pbos | d9eec76 | 2015-11-17 06:03:43 -0800 | [diff] [blame] | 46 | TRACE_EVENT0("webrtc", "VCMGenericEncoder::InitEncode"); |
Peter Boström | 4f5db11 | 2015-10-29 16:53:59 +0100 | [diff] [blame] | 47 | { |
| 48 | rtc::CritScope lock(¶ms_lock_); |
| 49 | encoder_params_.target_bitrate = settings->startBitrate * 1000; |
| 50 | encoder_params_.input_frame_rate = settings->maxFramerate; |
| 51 | } |
tommi@webrtc.org | 558dc40 | 2015-03-07 20:55:56 +0000 | [diff] [blame] | 52 | |
Peter Boström | 4f5db11 | 2015-10-29 16:53:59 +0100 | [diff] [blame] | 53 | is_screenshare_ = settings->mode == VideoCodecMode::kScreensharing; |
sprang | 3911c26 | 2016-04-15 01:24:14 -0700 | [diff] [blame] | 54 | if (encoder_->InitEncode(settings, number_of_cores, max_payload_size) != 0) { |
Peter Boström | 4f5db11 | 2015-10-29 16:53:59 +0100 | [diff] [blame] | 55 | LOG(LS_ERROR) << "Failed to initialize the encoder associated with " |
| 56 | "payload name: " |
| 57 | << settings->plName; |
| 58 | return -1; |
| 59 | } |
| 60 | encoder_->RegisterEncodeCompleteCallback(vcm_encoded_frame_callback_); |
| 61 | return 0; |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 62 | } |
| 63 | |
sprang | 3911c26 | 2016-04-15 01:24:14 -0700 | [diff] [blame] | 64 | int32_t VCMGenericEncoder::Encode(const VideoFrame& frame, |
| 65 | const CodecSpecificInfo* codec_specific, |
| 66 | const std::vector<FrameType>& frame_types) { |
pbos | d9eec76 | 2015-11-17 06:03:43 -0800 | [diff] [blame] | 67 | TRACE_EVENT1("webrtc", "VCMGenericEncoder::Encode", "timestamp", |
sprang | 3911c26 | 2016-04-15 01:24:14 -0700 | [diff] [blame] | 68 | frame.timestamp()); |
pbos | d9eec76 | 2015-11-17 06:03:43 -0800 | [diff] [blame] | 69 | |
sprang | 3911c26 | 2016-04-15 01:24:14 -0700 | [diff] [blame] | 70 | for (FrameType frame_type : frame_types) |
pbos | 22993e1 | 2015-10-19 02:39:06 -0700 | [diff] [blame] | 71 | RTC_DCHECK(frame_type == kVideoFrameKey || frame_type == kVideoFrameDelta); |
guoweis@webrtc.org | 54d072e | 2015-03-17 21:54:50 +0000 | [diff] [blame] | 72 | |
sprang | 3911c26 | 2016-04-15 01:24:14 -0700 | [diff] [blame] | 73 | int32_t result = encoder_->Encode(frame, codec_specific, &frame_types); |
Peter Boström | b7d9a97 | 2015-12-18 16:01:11 +0100 | [diff] [blame] | 74 | |
| 75 | if (vcm_encoded_frame_callback_) { |
| 76 | vcm_encoded_frame_callback_->SignalLastEncoderImplementationUsed( |
| 77 | encoder_->ImplementationName()); |
| 78 | } |
| 79 | |
Erik Språng | 2c4c914 | 2015-06-24 11:24:44 +0200 | [diff] [blame] | 80 | if (is_screenshare_ && |
| 81 | result == WEBRTC_VIDEO_CODEC_TARGET_BITRATE_OVERSHOOT) { |
| 82 | // Target bitrate exceeded, encoder state has been reset - try again. |
sprang | 3911c26 | 2016-04-15 01:24:14 -0700 | [diff] [blame] | 83 | return encoder_->Encode(frame, codec_specific, &frame_types); |
Erik Språng | 2c4c914 | 2015-06-24 11:24:44 +0200 | [diff] [blame] | 84 | } |
| 85 | |
| 86 | return result; |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 87 | } |
| 88 | |
Peter Boström | 69ccb33 | 2015-10-29 16:30:23 +0100 | [diff] [blame] | 89 | void VCMGenericEncoder::SetEncoderParameters(const EncoderParameters& params) { |
| 90 | bool channel_parameters_have_changed; |
| 91 | bool rates_have_changed; |
| 92 | { |
| 93 | rtc::CritScope lock(¶ms_lock_); |
| 94 | channel_parameters_have_changed = |
| 95 | params.loss_rate != encoder_params_.loss_rate || |
| 96 | params.rtt != encoder_params_.rtt; |
| 97 | rates_have_changed = |
| 98 | params.target_bitrate != encoder_params_.target_bitrate || |
| 99 | params.input_frame_rate != encoder_params_.input_frame_rate; |
| 100 | encoder_params_ = params; |
| 101 | } |
| 102 | if (channel_parameters_have_changed) |
| 103 | encoder_->SetChannelParameters(params.loss_rate, params.rtt); |
| 104 | if (rates_have_changed) { |
| 105 | uint32_t target_bitrate_kbps = (params.target_bitrate + 500) / 1000; |
| 106 | encoder_->SetRates(target_bitrate_kbps, params.input_frame_rate); |
sprang | 3911c26 | 2016-04-15 01:24:14 -0700 | [diff] [blame] | 107 | if (rate_observer_) { |
Peter Boström | 69ccb33 | 2015-10-29 16:30:23 +0100 | [diff] [blame] | 108 | rate_observer_->OnSetRates(params.target_bitrate, |
| 109 | params.input_frame_rate); |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 110 | } |
Peter Boström | 69ccb33 | 2015-10-29 16:30:23 +0100 | [diff] [blame] | 111 | } |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 112 | } |
| 113 | |
Peter Boström | 69ccb33 | 2015-10-29 16:30:23 +0100 | [diff] [blame] | 114 | EncoderParameters VCMGenericEncoder::GetEncoderParameters() const { |
| 115 | rtc::CritScope lock(¶ms_lock_); |
| 116 | return encoder_params_; |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 117 | } |
| 118 | |
philipel | 9d3ab61 | 2015-12-21 04:12:39 -0800 | [diff] [blame] | 119 | int32_t VCMGenericEncoder::SetPeriodicKeyFrames(bool enable) { |
| 120 | return encoder_->SetPeriodicKeyFrames(enable); |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 121 | } |
| 122 | |
pbos@webrtc.org | 7b859cc | 2013-04-02 15:54:38 +0000 | [diff] [blame] | 123 | int32_t VCMGenericEncoder::RequestFrame( |
stefan@webrtc.org | cf21686 | 2012-10-25 11:29:51 +0000 | [diff] [blame] | 124 | const std::vector<FrameType>& frame_types) { |
Miguel Casas-Sanchez | 4765070 | 2015-05-29 17:21:40 -0700 | [diff] [blame] | 125 | VideoFrame image; |
pbos | 22993e1 | 2015-10-19 02:39:06 -0700 | [diff] [blame] | 126 | return encoder_->Encode(image, NULL, &frame_types); |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 127 | } |
| 128 | |
philipel | 9d3ab61 | 2015-12-21 04:12:39 -0800 | [diff] [blame] | 129 | bool VCMGenericEncoder::InternalSource() const { |
| 130 | return internal_source_; |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 131 | } |
| 132 | |
jackychen | 61b4d51 | 2015-04-21 15:30:11 -0700 | [diff] [blame] | 133 | void VCMGenericEncoder::OnDroppedFrame() { |
| 134 | encoder_->OnDroppedFrame(); |
| 135 | } |
| 136 | |
Peter Boström | eb66e80 | 2015-06-05 11:08:03 +0200 | [diff] [blame] | 137 | bool VCMGenericEncoder::SupportsNativeHandle() const { |
| 138 | return encoder_->SupportsNativeHandle(); |
| 139 | } |
| 140 | |
jackychen | 6e2ce6e | 2015-07-13 16:26:33 -0700 | [diff] [blame] | 141 | int VCMGenericEncoder::GetTargetFramerate() { |
| 142 | return encoder_->GetTargetFramerate(); |
| 143 | } |
| 144 | |
andresp@webrtc.org | 1df9dc3 | 2014-01-09 08:01:57 +0000 | [diff] [blame] | 145 | VCMEncodedFrameCallback::VCMEncodedFrameCallback( |
guoweis@webrtc.org | 54d072e | 2015-03-17 21:54:50 +0000 | [diff] [blame] | 146 | EncodedImageCallback* post_encode_callback) |
Peter Boström | b7d9a97 | 2015-12-18 16:01:11 +0100 | [diff] [blame] | 147 | : send_callback_(), |
sprang | 3911c26 | 2016-04-15 01:24:14 -0700 | [diff] [blame] | 148 | media_opt_(nullptr), |
sprang | 3911c26 | 2016-04-15 01:24:14 -0700 | [diff] [blame] | 149 | internal_source_(false), |
sprang | 3911c26 | 2016-04-15 01:24:14 -0700 | [diff] [blame] | 150 | post_encode_callback_(post_encode_callback) {} |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 151 | |
sprang | 3911c26 | 2016-04-15 01:24:14 -0700 | [diff] [blame] | 152 | VCMEncodedFrameCallback::~VCMEncodedFrameCallback() {} |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 153 | |
philipel | 9d3ab61 | 2015-12-21 04:12:39 -0800 | [diff] [blame] | 154 | int32_t VCMEncodedFrameCallback::SetTransportCallback( |
| 155 | VCMPacketizationCallback* transport) { |
| 156 | send_callback_ = transport; |
| 157 | return VCM_OK; |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 158 | } |
| 159 | |
changbin.shao@webrtc.org | f31f56d | 2015-02-09 09:14:03 +0000 | [diff] [blame] | 160 | int32_t VCMEncodedFrameCallback::Encoded( |
Peter Boström | b7d9a97 | 2015-12-18 16:01:11 +0100 | [diff] [blame] | 161 | const EncodedImage& encoded_image, |
sprang | 3911c26 | 2016-04-15 01:24:14 -0700 | [diff] [blame] | 162 | const CodecSpecificInfo* codec_specific, |
| 163 | const RTPFragmentationHeader* fragmentation_header) { |
pbos | d9eec76 | 2015-11-17 06:03:43 -0800 | [diff] [blame] | 164 | TRACE_EVENT_INSTANT1("webrtc", "VCMEncodedFrameCallback::Encoded", |
Peter Boström | b7d9a97 | 2015-12-18 16:01:11 +0100 | [diff] [blame] | 165 | "timestamp", encoded_image._timeStamp); |
perkj | f5d55aa | 2016-04-20 01:17:02 -0700 | [diff] [blame] | 166 | int ret_val = post_encode_callback_->Encoded(encoded_image, codec_specific, |
| 167 | fragmentation_header); |
sprang | 3911c26 | 2016-04-15 01:24:14 -0700 | [diff] [blame] | 168 | if (ret_val < 0) |
| 169 | return ret_val; |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 170 | |
sprang | 3911c26 | 2016-04-15 01:24:14 -0700 | [diff] [blame] | 171 | if (media_opt_) { |
| 172 | media_opt_->UpdateWithEncodedData(encoded_image); |
| 173 | if (internal_source_) |
| 174 | return media_opt_->DropFrame(); // Signal to encoder to drop next frame. |
changbin.shao@webrtc.org | f31f56d | 2015-02-09 09:14:03 +0000 | [diff] [blame] | 175 | } |
| 176 | return VCM_OK; |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 177 | } |
| 178 | |
Peter Boström | b7d9a97 | 2015-12-18 16:01:11 +0100 | [diff] [blame] | 179 | void VCMEncodedFrameCallback::SetMediaOpt( |
| 180 | media_optimization::MediaOptimization* mediaOpt) { |
sprang | 3911c26 | 2016-04-15 01:24:14 -0700 | [diff] [blame] | 181 | media_opt_ = mediaOpt; |
Peter Boström | b7d9a97 | 2015-12-18 16:01:11 +0100 | [diff] [blame] | 182 | } |
| 183 | |
| 184 | void VCMEncodedFrameCallback::SignalLastEncoderImplementationUsed( |
| 185 | const char* implementation_name) { |
| 186 | if (send_callback_) |
| 187 | send_callback_->OnEncoderImplementationName(implementation_name); |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 188 | } |
pbos@webrtc.org | d900e8b | 2013-07-03 15:12:26 +0000 | [diff] [blame] | 189 | } // namespace webrtc |