niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 1 | /* |
pwestin@webrtc.org | 52fd98d | 2012-02-13 09:03:53 +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 | #ifndef MODULES_VIDEO_CODING_VIDEO_CODING_IMPL_H_ |
| 12 | #define MODULES_VIDEO_CODING_VIDEO_CODING_IMPL_H_ |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 13 | |
kwiberg | 3f55dea | 2016-02-29 05:51:59 -0800 | [diff] [blame] | 14 | #include <memory> |
perkj | 376b192 | 2016-05-02 11:35:24 -0700 | [diff] [blame] | 15 | #include <string> |
stefan@webrtc.org | c530043 | 2012-10-08 07:06:53 +0000 | [diff] [blame] | 16 | #include <vector> |
| 17 | |
Erik Språng | eee3920 | 2018-11-15 17:52:43 +0100 | [diff] [blame] | 18 | #include "absl/types/optional.h" |
Niels Möller | f906378 | 2018-02-20 16:09:48 +0100 | [diff] [blame] | 19 | #include "modules/video_coding/decoder_database.h" |
Mirko Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 20 | #include "modules/video_coding/frame_buffer.h" |
| 21 | #include "modules/video_coding/generic_decoder.h" |
Jonas Olsson | a4d8737 | 2019-07-05 19:08:33 +0200 | [diff] [blame] | 22 | #include "modules/video_coding/include/video_coding.h" |
Mirko Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 23 | #include "modules/video_coding/jitter_buffer.h" |
Mirko Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 24 | #include "modules/video_coding/receiver.h" |
| 25 | #include "modules/video_coding/timing.h" |
Steve Anton | 10542f2 | 2019-01-11 09:11:00 -0800 | [diff] [blame] | 26 | #include "rtc_base/one_time_event.h" |
Markus Handell | 6deec38 | 2020-07-07 12:17:12 +0200 | [diff] [blame] | 27 | #include "rtc_base/synchronization/mutex.h" |
Sebastian Jansson | b55015e | 2019-04-09 13:44:04 +0200 | [diff] [blame] | 28 | #include "rtc_base/synchronization/sequence_checker.h" |
Mirko Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 29 | #include "rtc_base/thread_annotations.h" |
| 30 | #include "rtc_base/thread_checker.h" |
| 31 | #include "system_wrappers/include/clock.h" |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 32 | |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 33 | namespace webrtc { |
sprang@webrtc.org | 4070935 | 2013-11-26 11:41:59 +0000 | [diff] [blame] | 34 | |
Erik Språng | 08127a9 | 2016-11-16 16:41:30 +0100 | [diff] [blame] | 35 | class VideoBitrateAllocator; |
sprang | 1a646ee | 2016-12-01 06:34:11 -0800 | [diff] [blame] | 36 | class VideoBitrateAllocationObserver; |
Erik Språng | 08127a9 | 2016-11-16 16:41:30 +0100 | [diff] [blame] | 37 | |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 38 | namespace vcm { |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 39 | |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 40 | class VCMProcessTimer { |
| 41 | public: |
sprang | 40217c3 | 2016-11-21 05:41:52 -0800 | [diff] [blame] | 42 | static const int64_t kDefaultProcessIntervalMs = 1000; |
| 43 | |
pkasting@chromium.org | 0b1534c | 2014-12-15 22:09:40 +0000 | [diff] [blame] | 44 | VCMProcessTimer(int64_t periodMs, Clock* clock) |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 45 | : _clock(clock), |
| 46 | _periodMs(periodMs), |
| 47 | _latestMs(_clock->TimeInMilliseconds()) {} |
pkasting@chromium.org | 0b1534c | 2014-12-15 22:09:40 +0000 | [diff] [blame] | 48 | int64_t Period() const; |
| 49 | int64_t TimeUntilProcess() const; |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 50 | void Processed(); |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 51 | |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 52 | private: |
| 53 | Clock* _clock; |
pkasting@chromium.org | 0b1534c | 2014-12-15 22:09:40 +0000 | [diff] [blame] | 54 | int64_t _periodMs; |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 55 | int64_t _latestMs; |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 56 | }; |
| 57 | |
Peter Boström | 0b25072 | 2016-04-22 18:23:15 +0200 | [diff] [blame] | 58 | class VideoReceiver : public Module { |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 59 | public: |
Niels Möller | db64d99 | 2019-03-29 14:30:53 +0100 | [diff] [blame] | 60 | VideoReceiver(Clock* clock, VCMTiming* timing); |
Tommi | fbf3bce | 2018-02-21 15:56:05 +0100 | [diff] [blame] | 61 | ~VideoReceiver() override; |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 62 | |
Niels Möller | 582102c | 2020-08-07 16:19:56 +0200 | [diff] [blame] | 63 | int32_t RegisterReceiveCodec(uint8_t payload_type, |
| 64 | const VideoCodec* receiveCodec, |
Niels Möller | 18c83d3 | 2020-08-07 14:14:49 +0200 | [diff] [blame] | 65 | int32_t numberOfCores); |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 66 | |
Peter Boström | 795dbe4 | 2015-11-27 14:09:07 +0100 | [diff] [blame] | 67 | void RegisterExternalDecoder(VideoDecoder* externalDecoder, |
perkj | 796cfaf | 2015-12-10 09:27:38 -0800 | [diff] [blame] | 68 | uint8_t payloadType); |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 69 | int32_t RegisterReceiveCallback(VCMReceiveCallback* receiveCallback); |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 70 | int32_t RegisterFrameTypeCallback(VCMFrameTypeCallback* frameTypeCallback); |
| 71 | int32_t RegisterPacketRequestCallback(VCMPacketRequestCallback* callback); |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 72 | |
| 73 | int32_t Decode(uint16_t maxWaitTimeMs); |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 74 | |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 75 | int32_t IncomingPacket(const uint8_t* incomingPayload, |
pkasting@chromium.org | 4591fbd | 2014-11-20 22:28:14 +0000 | [diff] [blame] | 76 | size_t payloadLength, |
Niels Möller | be7a0ec | 2019-04-25 10:02:52 +0200 | [diff] [blame] | 77 | const RTPHeader& rtp_header, |
| 78 | const RTPVideoHeader& video_header); |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 79 | |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 80 | void SetNackSettings(size_t max_nack_list_size, |
| 81 | int max_packet_age_to_nack, |
| 82 | int max_incomplete_time_ms); |
| 83 | |
Peter Boström | 0b25072 | 2016-04-22 18:23:15 +0200 | [diff] [blame] | 84 | int64_t TimeUntilNextProcess() override; |
| 85 | void Process() override; |
Tommi | fbf3bce | 2018-02-21 15:56:05 +0100 | [diff] [blame] | 86 | void ProcessThreadAttached(ProcessThread* process_thread) override; |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 87 | |
| 88 | protected: |
Tommi | fbf3bce | 2018-02-21 15:56:05 +0100 | [diff] [blame] | 89 | int32_t Decode(const webrtc::VCMEncodedFrame& frame); |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 90 | int32_t RequestKeyFrame(); |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 91 | |
| 92 | private: |
Tommi | fbf3bce | 2018-02-21 15:56:05 +0100 | [diff] [blame] | 93 | // Used for DCHECKing thread correctness. |
| 94 | // In build where DCHECKs are enabled, will return false before |
| 95 | // DecoderThreadStarting is called, then true until DecoderThreadStopped |
| 96 | // is called. |
| 97 | // In builds where DCHECKs aren't enabled, it will return true. |
| 98 | bool IsDecoderThreadRunning(); |
| 99 | |
| 100 | rtc::ThreadChecker construction_thread_checker_; |
| 101 | rtc::ThreadChecker decoder_thread_checker_; |
| 102 | rtc::ThreadChecker module_thread_checker_; |
pbos@webrtc.org | 20c1f56 | 2014-07-04 10:58:12 +0000 | [diff] [blame] | 103 | Clock* const clock_; |
Markus Handell | 6deec38 | 2020-07-07 12:17:12 +0200 | [diff] [blame] | 104 | Mutex process_mutex_; |
philipel | 721d402 | 2016-12-15 07:10:57 -0800 | [diff] [blame] | 105 | VCMTiming* _timing; |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 106 | VCMReceiver _receiver; |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 107 | VCMDecodedFrameCallback _decodedFrameCallback; |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 108 | |
Tommi | fbf3bce | 2018-02-21 15:56:05 +0100 | [diff] [blame] | 109 | // These callbacks are set on the construction thread before being attached |
| 110 | // to the module thread or decoding started, so a lock is not required. |
| 111 | VCMFrameTypeCallback* _frameTypeCallback; |
Tommi | fbf3bce | 2018-02-21 15:56:05 +0100 | [diff] [blame] | 112 | VCMPacketRequestCallback* _packetRequestCallback; |
| 113 | |
| 114 | // Used on both the module and decoder thread. |
Markus Handell | 6deec38 | 2020-07-07 12:17:12 +0200 | [diff] [blame] | 115 | bool _scheduleKeyRequest RTC_GUARDED_BY(process_mutex_); |
| 116 | bool drop_frames_until_keyframe_ RTC_GUARDED_BY(process_mutex_); |
sprang | 3911c26 | 2016-04-15 01:24:14 -0700 | [diff] [blame] | 117 | |
Tommi | fbf3bce | 2018-02-21 15:56:05 +0100 | [diff] [blame] | 118 | // Modified on the construction thread while not attached to the process |
| 119 | // thread. Once attached to the process thread, its value is only read |
| 120 | // so a lock is not required. |
| 121 | size_t max_nack_list_size_; |
Peter Boström | ed3277b | 2016-02-02 15:40:04 +0100 | [diff] [blame] | 122 | |
Tommi | fbf3bce | 2018-02-21 15:56:05 +0100 | [diff] [blame] | 123 | // Callbacks are set before the decoder thread starts. |
| 124 | // Once the decoder thread has been started, usage of |_codecDataBase| moves |
| 125 | // over to the decoder thread. |
| 126 | VCMDecoderDataBase _codecDataBase; |
Tommi | fbf3bce | 2018-02-21 15:56:05 +0100 | [diff] [blame] | 127 | |
Tommi | fbf3bce | 2018-02-21 15:56:05 +0100 | [diff] [blame] | 128 | VCMProcessTimer _retransmissionTimer RTC_GUARDED_BY(module_thread_checker_); |
| 129 | VCMProcessTimer _keyRequestTimer RTC_GUARDED_BY(module_thread_checker_); |
Tommi | fbf3bce | 2018-02-21 15:56:05 +0100 | [diff] [blame] | 130 | ThreadUnsafeOneTimeEvent first_frame_received_ |
| 131 | RTC_GUARDED_BY(decoder_thread_checker_); |
| 132 | // Modified on the construction thread. Can be read without a lock and assumed |
| 133 | // to be non-null on the module and decoder threads. |
| 134 | ProcessThread* process_thread_ = nullptr; |
| 135 | bool is_attached_to_process_thread_ |
| 136 | RTC_GUARDED_BY(construction_thread_checker_) = false; |
niklase@google.com | 470e71d | 2011-07-07 08:21:25 +0000 | [diff] [blame] | 137 | }; |
andresp@webrtc.org | f7eb75b | 2013-09-14 00:25:28 +0000 | [diff] [blame] | 138 | |
| 139 | } // namespace vcm |
pbos@webrtc.org | d900e8b | 2013-07-03 15:12:26 +0000 | [diff] [blame] | 140 | } // namespace webrtc |
Mirko Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 141 | #endif // MODULES_VIDEO_CODING_VIDEO_CODING_IMPL_H_ |