niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 1 | /* |
mflodman@webrtc.org | c80d9d9 | 2012-02-06 10:11:25 +0000 | [diff] [blame] | 2 | * Copyright (c) 2012 The WebRTC project authors. All Rights Reserved. |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 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 | |
Mirko Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 11 | #include "modules/video_coding/video_coding_impl.h" |
philipel | 5908c71 | 2015-12-21 08:23:20 -0800 | [diff] [blame] | 12 | |
| 13 | #include <algorithm> |
Erik Språng | 08127a9 | 2016-11-16 16:41:30 +0100 | [diff] [blame] | 14 | #include <utility> |
philipel | 5908c71 | 2015-12-21 08:23:20 -0800 | [diff] [blame] | 15 | |
Jiawei Ou | 4206a0a | 2018-07-20 15:49:43 -0700 | [diff] [blame] | 16 | #include "api/video/video_bitrate_allocator.h" |
Mirko Bonadei | 7120742 | 2017-09-15 13:58:09 +0200 | [diff] [blame] | 17 | #include "common_types.h" // NOLINT(build/include) |
Mirko Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 18 | #include "common_video/libyuv/include/webrtc_libyuv.h" |
Mirko Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 19 | #include "modules/video_coding/encoded_frame.h" |
| 20 | #include "modules/video_coding/include/video_codec_initializer.h" |
| 21 | #include "modules/video_coding/include/video_codec_interface.h" |
| 22 | #include "modules/video_coding/jitter_buffer.h" |
| 23 | #include "modules/video_coding/packet.h" |
| 24 | #include "modules/video_coding/timing.h" |
| 25 | #include "rtc_base/criticalsection.h" |
| 26 | #include "rtc_base/thread_checker.h" |
| 27 | #include "system_wrappers/include/clock.h" |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 28 | |
agalusza@google.com | a7e360e | 2013-08-01 03:15:08 +0000 | [diff] [blame] | 29 | namespace webrtc { |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 30 | namespace vcm { |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 31 | |
philipel | 5908c71 | 2015-12-21 08:23:20 -0800 | [diff] [blame] | 32 | int64_t VCMProcessTimer::Period() const { |
| 33 | return _periodMs; |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 34 | } |
| 35 | |
philipel | 5908c71 | 2015-12-21 08:23:20 -0800 | [diff] [blame] | 36 | int64_t VCMProcessTimer::TimeUntilProcess() const { |
| 37 | const int64_t time_since_process = _clock->TimeInMilliseconds() - _latestMs; |
| 38 | const int64_t time_until_process = _periodMs - time_since_process; |
| 39 | return std::max<int64_t>(time_until_process, 0); |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 40 | } |
| 41 | |
philipel | 5908c71 | 2015-12-21 08:23:20 -0800 | [diff] [blame] | 42 | void VCMProcessTimer::Processed() { |
| 43 | _latestMs = _clock->TimeInMilliseconds(); |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 44 | } |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 45 | } // namespace vcm |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 46 | |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 47 | namespace { |
andresp@webrtc.org | 1df9dc3 | 2014-01-09 08:01:57 +0000 | [diff] [blame] | 48 | // This wrapper provides a way to modify the callback without the need to expose |
| 49 | // a register method all the way down to the function calling it. |
| 50 | class EncodedImageCallbackWrapper : public EncodedImageCallback { |
| 51 | public: |
kthelgason | d701dfd | 2017-03-27 07:24:57 -0700 | [diff] [blame] | 52 | EncodedImageCallbackWrapper() : callback_(nullptr) {} |
andresp@webrtc.org | 1df9dc3 | 2014-01-09 08:01:57 +0000 | [diff] [blame] | 53 | |
| 54 | virtual ~EncodedImageCallbackWrapper() {} |
| 55 | |
| 56 | void Register(EncodedImageCallback* callback) { |
kthelgason | d701dfd | 2017-03-27 07:24:57 -0700 | [diff] [blame] | 57 | rtc::CritScope lock(&cs_); |
andresp@webrtc.org | 1df9dc3 | 2014-01-09 08:01:57 +0000 | [diff] [blame] | 58 | callback_ = callback; |
| 59 | } |
| 60 | |
Sergey Ulanov | 525df3f | 2016-08-02 17:46:41 -0700 | [diff] [blame] | 61 | virtual Result OnEncodedImage(const EncodedImage& encoded_image, |
| 62 | const CodecSpecificInfo* codec_specific_info, |
| 63 | const RTPFragmentationHeader* fragmentation) { |
kthelgason | d701dfd | 2017-03-27 07:24:57 -0700 | [diff] [blame] | 64 | rtc::CritScope lock(&cs_); |
Sergey Ulanov | 525df3f | 2016-08-02 17:46:41 -0700 | [diff] [blame] | 65 | if (callback_) { |
| 66 | return callback_->OnEncodedImage(encoded_image, codec_specific_info, |
| 67 | fragmentation); |
| 68 | } |
| 69 | return Result(Result::ERROR_SEND_FAILED); |
andresp@webrtc.org | 1df9dc3 | 2014-01-09 08:01:57 +0000 | [diff] [blame] | 70 | } |
| 71 | |
| 72 | private: |
kthelgason | d701dfd | 2017-03-27 07:24:57 -0700 | [diff] [blame] | 73 | rtc::CriticalSection cs_; |
danilchap | 56359be | 2017-09-07 07:53:45 -0700 | [diff] [blame] | 74 | EncodedImageCallback* callback_ RTC_GUARDED_BY(cs_); |
andresp@webrtc.org | 1df9dc3 | 2014-01-09 08:01:57 +0000 | [diff] [blame] | 75 | }; |
| 76 | |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 77 | class VideoCodingModuleImpl : public VideoCodingModule { |
| 78 | public: |
stefan@webrtc.org | 34c5da6 | 2014-04-11 14:08:35 +0000 | [diff] [blame] | 79 | VideoCodingModuleImpl(Clock* clock, |
philipel | 83f831a | 2016-03-12 03:30:23 -0800 | [diff] [blame] | 80 | NackSender* nack_sender, |
Niels Möller | 9eb44ac | 2018-10-02 12:47:47 +0200 | [diff] [blame] | 81 | KeyFrameRequestSender* keyframe_request_sender) |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 82 | : VideoCodingModule(), |
Niels Möller | a056599 | 2017-10-24 11:37:08 +0200 | [diff] [blame] | 83 | sender_(clock, &post_encode_callback_), |
philipel | 721d402 | 2016-12-15 07:10:57 -0800 | [diff] [blame] | 84 | timing_(new VCMTiming(clock)), |
Niels Möller | 689983f | 2018-11-07 16:36:22 +0100 | [diff] [blame] | 85 | receiver_(clock, timing_.get(), nack_sender, keyframe_request_sender) {} |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 86 | |
Peter Boström | 0b25072 | 2016-04-22 18:23:15 +0200 | [diff] [blame] | 87 | virtual ~VideoCodingModuleImpl() {} |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 88 | |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 +0000 | [diff] [blame] | 89 | int64_t TimeUntilNextProcess() override { |
pbos | 0fa9b22 | 2015-11-13 05:59:57 -0800 | [diff] [blame] | 90 | int64_t receiver_time = receiver_.TimeUntilNextProcess(); |
tommi | d0a71ba | 2017-03-14 04:16:20 -0700 | [diff] [blame] | 91 | RTC_DCHECK_GE(receiver_time, 0); |
Niels Möller | a056599 | 2017-10-24 11:37:08 +0200 | [diff] [blame] | 92 | return receiver_time; |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 93 | } |
| 94 | |
Yves Gerey | 665174f | 2018-06-19 15:03:05 +0200 | [diff] [blame] | 95 | void Process() override { receiver_.Process(); } |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 96 | |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 +0000 | [diff] [blame] | 97 | int32_t RegisterSendCodec(const VideoCodec* sendCodec, |
| 98 | uint32_t numberOfCores, |
| 99 | uint32_t maxPayloadSize) override { |
Sergio Garcia Murillo | 43800f9 | 2018-06-21 16:16:38 +0200 | [diff] [blame] | 100 | if (sendCodec != nullptr && ((sendCodec->codecType == kVideoCodecVP8) || |
| 101 | (sendCodec->codecType == kVideoCodecH264))) { |
| 102 | // Set up a rate allocator and temporal layers factory for this codec |
Erik Språng | 08127a9 | 2016-11-16 16:41:30 +0100 | [diff] [blame] | 103 | // instance. The codec impl will have a raw pointer to the TL factory, |
| 104 | // and will call it when initializing. Since this can happen |
| 105 | // asynchronously keep the instance alive until destruction or until a |
| 106 | // new send codec is registered. |
Sergio Garcia Murillo | 43800f9 | 2018-06-21 16:16:38 +0200 | [diff] [blame] | 107 | VideoCodec codec = *sendCodec; |
Qingsi Wang | 59844ce | 2018-11-01 04:45:53 +0000 | [diff] [blame] | 108 | rate_allocator_ = VideoCodecInitializer::CreateBitrateAllocator(codec); |
Sergio Garcia Murillo | 43800f9 | 2018-06-21 16:16:38 +0200 | [diff] [blame] | 109 | return sender_.RegisterSendCodec(&codec, numberOfCores, maxPayloadSize); |
Erik Språng | 08127a9 | 2016-11-16 16:41:30 +0100 | [diff] [blame] | 110 | } |
pbos | 0fa9b22 | 2015-11-13 05:59:57 -0800 | [diff] [blame] | 111 | return sender_.RegisterSendCodec(sendCodec, numberOfCores, maxPayloadSize); |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 112 | } |
| 113 | |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 +0000 | [diff] [blame] | 114 | int32_t RegisterExternalEncoder(VideoEncoder* externalEncoder, |
Niels Möller | bf3dbb4 | 2018-03-16 13:38:46 +0100 | [diff] [blame] | 115 | uint8_t /* payloadType */, |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 +0000 | [diff] [blame] | 116 | bool internalSource) override { |
Yves Gerey | 665174f | 2018-06-19 15:03:05 +0200 | [diff] [blame] | 117 | sender_.RegisterExternalEncoder(externalEncoder, internalSource); |
Peter Boström | 795dbe4 | 2015-11-27 14:09:07 +0100 | [diff] [blame] | 118 | return 0; |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 119 | } |
| 120 | |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 +0000 | [diff] [blame] | 121 | int32_t SetChannelParameters(uint32_t target_bitrate, // bits/s. |
| 122 | uint8_t lossRate, |
| 123 | int64_t rtt) override { |
Erik Språng | 08127a9 | 2016-11-16 16:41:30 +0100 | [diff] [blame] | 124 | return sender_.SetChannelParameters(target_bitrate, lossRate, rtt, |
sprang | 1a646ee | 2016-12-01 06:34:11 -0800 | [diff] [blame] | 125 | rate_allocator_.get(), nullptr); |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 126 | } |
| 127 | |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 +0000 | [diff] [blame] | 128 | int32_t SetVideoProtection(VCMVideoProtection videoProtection, |
| 129 | bool enable) override { |
pbos | ba8c15b | 2015-07-14 09:36:34 -0700 | [diff] [blame] | 130 | // TODO(pbos): Remove enable from receive-side protection modes as well. |
pbos | 0fa9b22 | 2015-11-13 05:59:57 -0800 | [diff] [blame] | 131 | return receiver_.SetVideoProtection(videoProtection, enable); |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 132 | } |
| 133 | |
Miguel Casas-Sanchez | 4765070 | 2015-05-29 17:21:40 -0700 | [diff] [blame] | 134 | int32_t AddVideoFrame(const VideoFrame& videoFrame, |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 +0000 | [diff] [blame] | 135 | const CodecSpecificInfo* codecSpecificInfo) override { |
Peter Boström | ad6fc5a | 2016-05-12 03:01:31 +0200 | [diff] [blame] | 136 | return sender_.AddVideoFrame(videoFrame, codecSpecificInfo); |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 137 | } |
| 138 | |
perkj | 600246e | 2016-05-04 11:26:51 -0700 | [diff] [blame] | 139 | int32_t IntraFrameRequest(size_t stream_index) override { |
pbos | 0fa9b22 | 2015-11-13 05:59:57 -0800 | [diff] [blame] | 140 | return sender_.IntraFrameRequest(stream_index); |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 141 | } |
| 142 | |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 +0000 | [diff] [blame] | 143 | int32_t EnableFrameDropper(bool enable) override { |
pbos | 0fa9b22 | 2015-11-13 05:59:57 -0800 | [diff] [blame] | 144 | return sender_.EnableFrameDropper(enable); |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 145 | } |
| 146 | |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 +0000 | [diff] [blame] | 147 | int32_t RegisterReceiveCodec(const VideoCodec* receiveCodec, |
| 148 | int32_t numberOfCores, |
| 149 | bool requireKeyFrame) override { |
pbos | 0fa9b22 | 2015-11-13 05:59:57 -0800 | [diff] [blame] | 150 | return receiver_.RegisterReceiveCodec(receiveCodec, numberOfCores, |
| 151 | requireKeyFrame); |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 152 | } |
| 153 | |
Anders Carlsson | 7eb8e9f | 2018-05-18 10:33:04 +0200 | [diff] [blame] | 154 | void RegisterExternalDecoder(VideoDecoder* externalDecoder, |
| 155 | uint8_t payloadType) override { |
| 156 | receiver_.RegisterExternalDecoder(externalDecoder, payloadType); |
| 157 | } |
| 158 | |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 +0000 | [diff] [blame] | 159 | int32_t RegisterReceiveCallback( |
| 160 | VCMReceiveCallback* receiveCallback) override { |
tommi | d0a71ba | 2017-03-14 04:16:20 -0700 | [diff] [blame] | 161 | RTC_DCHECK(construction_thread_.CalledOnValidThread()); |
pbos | 0fa9b22 | 2015-11-13 05:59:57 -0800 | [diff] [blame] | 162 | return receiver_.RegisterReceiveCallback(receiveCallback); |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 163 | } |
| 164 | |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 +0000 | [diff] [blame] | 165 | int32_t RegisterFrameTypeCallback( |
| 166 | VCMFrameTypeCallback* frameTypeCallback) override { |
pbos | 0fa9b22 | 2015-11-13 05:59:57 -0800 | [diff] [blame] | 167 | return receiver_.RegisterFrameTypeCallback(frameTypeCallback); |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 168 | } |
| 169 | |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 +0000 | [diff] [blame] | 170 | int32_t RegisterPacketRequestCallback( |
| 171 | VCMPacketRequestCallback* callback) override { |
tommi | d0a71ba | 2017-03-14 04:16:20 -0700 | [diff] [blame] | 172 | RTC_DCHECK(construction_thread_.CalledOnValidThread()); |
pbos | 0fa9b22 | 2015-11-13 05:59:57 -0800 | [diff] [blame] | 173 | return receiver_.RegisterPacketRequestCallback(callback); |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 174 | } |
| 175 | |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 +0000 | [diff] [blame] | 176 | int32_t Decode(uint16_t maxWaitTimeMs) override { |
pbos | 0fa9b22 | 2015-11-13 05:59:57 -0800 | [diff] [blame] | 177 | return receiver_.Decode(maxWaitTimeMs); |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 178 | } |
| 179 | |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 +0000 | [diff] [blame] | 180 | int32_t IncomingPacket(const uint8_t* incomingPayload, |
| 181 | size_t payloadLength, |
| 182 | const WebRtcRTPHeader& rtpInfo) override { |
pbos | 0fa9b22 | 2015-11-13 05:59:57 -0800 | [diff] [blame] | 183 | return receiver_.IncomingPacket(incomingPayload, payloadLength, rtpInfo); |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 184 | } |
| 185 | |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 +0000 | [diff] [blame] | 186 | int SetReceiverRobustnessMode(ReceiverRobustness robustnessMode, |
| 187 | VCMDecodeErrorMode errorMode) override { |
pbos | 0fa9b22 | 2015-11-13 05:59:57 -0800 | [diff] [blame] | 188 | return receiver_.SetReceiverRobustnessMode(robustnessMode, errorMode); |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 189 | } |
| 190 | |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 +0000 | [diff] [blame] | 191 | void SetNackSettings(size_t max_nack_list_size, |
| 192 | int max_packet_age_to_nack, |
| 193 | int max_incomplete_time_ms) override { |
pbos | 0fa9b22 | 2015-11-13 05:59:57 -0800 | [diff] [blame] | 194 | return receiver_.SetNackSettings(max_nack_list_size, max_packet_age_to_nack, |
| 195 | max_incomplete_time_ms); |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 196 | } |
| 197 | |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 +0000 | [diff] [blame] | 198 | void RegisterPostEncodeImageCallback( |
| 199 | EncodedImageCallback* observer) override { |
andresp@webrtc.org | 1df9dc3 | 2014-01-09 08:01:57 +0000 | [diff] [blame] | 200 | post_encode_callback_.Register(observer); |
sprang@webrtc.org | 4070935 | 2013-11-26 11:41:59 +0000 | [diff] [blame] | 201 | } |
| 202 | |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 203 | private: |
tommi | d0a71ba | 2017-03-14 04:16:20 -0700 | [diff] [blame] | 204 | rtc::ThreadChecker construction_thread_; |
andresp@webrtc.org | 1df9dc3 | 2014-01-09 08:01:57 +0000 | [diff] [blame] | 205 | EncodedImageCallbackWrapper post_encode_callback_; |
pbos | 0fa9b22 | 2015-11-13 05:59:57 -0800 | [diff] [blame] | 206 | vcm::VideoSender sender_; |
Erik Språng | 08127a9 | 2016-11-16 16:41:30 +0100 | [diff] [blame] | 207 | std::unique_ptr<VideoBitrateAllocator> rate_allocator_; |
Qingsi Wang | 59844ce | 2018-11-01 04:45:53 +0000 | [diff] [blame] | 208 | std::unique_ptr<VCMTiming> timing_; |
pbos | 0fa9b22 | 2015-11-13 05:59:57 -0800 | [diff] [blame] | 209 | vcm::VideoReceiver receiver_; |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 210 | }; |
| 211 | } // namespace |
| 212 | |
tommi | a5c18d7 | 2017-03-20 10:43:23 -0700 | [diff] [blame] | 213 | // DEPRECATED. Create method for current interface, will be removed when the |
philipel | 83f831a | 2016-03-12 03:30:23 -0800 | [diff] [blame] | 214 | // new jitter buffer is in place. |
Niels Möller | 689983f | 2018-11-07 16:36:22 +0100 | [diff] [blame] | 215 | VideoCodingModule* VideoCodingModule::Create(Clock* clock) { |
tommi | d0a71ba | 2017-03-14 04:16:20 -0700 | [diff] [blame] | 216 | RTC_DCHECK(clock); |
Niels Möller | c4e9825 | 2018-11-08 13:08:26 +0100 | [diff] [blame] | 217 | return new VideoCodingModuleImpl(clock, nullptr, nullptr); |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 218 | } |
| 219 | |
stefan@webrtc.org | 791eec7 | 2011-10-11 07:53:43 +0000 | [diff] [blame] | 220 | } // namespace webrtc |