blob: 31d84601946df7ad093d658a0b7a85d8196dc535 [file] [log] [blame]
niklase@google.com470e71d2011-07-07 08:21:25 +00001/*
mikhal@webrtc.orga2031d52012-07-31 15:53:44 +00002 * Copyright (c) 2012 The WebRTC project authors. All Rights Reserved.
niklase@google.com470e71d2011-07-07 08:21:25 +00003 *
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 Bonadei92ea95e2017-09-15 06:47:31 +020011#ifndef MODULES_VIDEO_CODING_GENERIC_DECODER_H_
12#define MODULES_VIDEO_CODING_GENERIC_DECODER_H_
niklase@google.com470e71d2011-07-07 08:21:25 +000013
Ilya Nikolaevskiy43c108b2020-05-15 12:24:29 +020014#include <string>
tommi5b7fc8c2017-07-05 16:45:57 -070015
Artem Titovd15a5752021-02-10 14:31:24 +010016#include "api/sequence_checker.h"
Johannes Kron7ddea572019-09-11 12:00:22 +020017#include "api/units/time_delta.h"
Danil Chapovalov355b8d22021-08-13 16:50:37 +020018#include "api/video_codecs/video_decoder.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020019#include "modules/video_coding/encoded_frame.h"
20#include "modules/video_coding/include/video_codec_interface.h"
21#include "modules/video_coding/timestamp_map.h"
22#include "modules/video_coding/timing.h"
Johannes Kron7ddea572019-09-11 12:00:22 +020023#include "rtc_base/experiments/field_trial_parser.h"
Markus Handell6deec382020-07-07 12:17:12 +020024#include "rtc_base/synchronization/mutex.h"
niklase@google.com470e71d2011-07-07 08:21:25 +000025
philipel9d3ab612015-12-21 04:12:39 -080026namespace webrtc {
niklase@google.com470e71d2011-07-07 08:21:25 +000027
28class VCMReceiveCallback;
29
30enum { kDecoderFrameMemoryLength = 10 };
31
philipel9d3ab612015-12-21 04:12:39 -080032class VCMDecodedFrameCallback : public DecodedImageCallback {
33 public:
34 VCMDecodedFrameCallback(VCMTiming* timing, Clock* clock);
tommid0a71ba2017-03-14 04:16:20 -070035 ~VCMDecodedFrameCallback() override;
36 void SetUserReceiveCallback(VCMReceiveCallback* receiveCallback);
37 VCMReceiveCallback* UserReceiveCallback();
niklase@google.com470e71d2011-07-07 08:21:25 +000038
tommid0a71ba2017-03-14 04:16:20 -070039 int32_t Decoded(VideoFrame& decodedImage) override;
40 int32_t Decoded(VideoFrame& decodedImage, int64_t decode_time_ms) override;
41 void Decoded(VideoFrame& decodedImage,
Danil Chapovalov0040b662018-06-18 10:48:16 +020042 absl::optional<int32_t> decode_time_ms,
43 absl::optional<uint8_t> qp) override;
niklase@google.com470e71d2011-07-07 08:21:25 +000044
tommid0a71ba2017-03-14 04:16:20 -070045 void OnDecoderImplementationName(const char* implementation_name);
niklase@google.com470e71d2011-07-07 08:21:25 +000046
Johannes Kronb6b782d2021-03-03 14:39:44 +010047 void Map(uint32_t timestamp, const VCMFrameInformation& frameInfo);
Johannes Kronfc5d2762021-04-09 16:03:22 +020048 void ClearTimestampMap();
niklase@google.com470e71d2011-07-07 08:21:25 +000049
philipel9d3ab612015-12-21 04:12:39 -080050 private:
Artem Titovc8421c42021-02-02 10:57:19 +010051 SequenceChecker construction_thread_;
Artem Titovdcd7fc72021-08-09 13:02:57 +020052 // Protect `_timestampMap`.
tommid0a71ba2017-03-14 04:16:20 -070053 Clock* const _clock;
54 // This callback must be set before the decoder thread starts running
55 // and must only be unset when external threads (e.g decoder thread)
56 // have been stopped. Due to that, the variable should regarded as const
57 // while there are more than one threads involved, it must be set
58 // from the same thread, and therfore a lock is not required to access it.
59 VCMReceiveCallback* _receiveCallback = nullptr;
Lu Liu352314a2018-02-21 19:38:59 +000060 VCMTiming* _timing;
Markus Handell6deec382020-07-07 12:17:12 +020061 Mutex lock_;
Lu Liu352314a2018-02-21 19:38:59 +000062 VCMTimestampMap _timestampMap RTC_GUARDED_BY(lock_);
ilnik04f4d122017-06-19 07:18:55 -070063 int64_t ntp_offset_;
Johannes Kron7ddea572019-09-11 12:00:22 +020064 // Set by the field trial WebRTC-SlowDownDecoder to simulate a slow decoder.
65 FieldTrialOptional<TimeDelta> _extra_decode_time;
Johannes Kron111e9812020-10-26 13:54:40 +010066
Artem Titovdcd7fc72021-08-09 13:02:57 +020067 // Set by the field trial WebRTC-LowLatencyRenderer. The parameter `enabled`
Johannes Kron111e9812020-10-26 13:54:40 +010068 // determines if the low-latency renderer algorithm should be used for the
69 // case min playout delay=0 and max playout delay>0.
70 FieldTrialParameter<bool> low_latency_renderer_enabled_;
71 // Set by the field trial WebRTC-LowLatencyRenderer. The parameter
Artem Titovdcd7fc72021-08-09 13:02:57 +020072 // `include_predecode_buffer` determines if the predecode buffer should be
Johannes Kron111e9812020-10-26 13:54:40 +010073 // taken into account when calculating maximum number of frames in composition
74 // queue.
75 FieldTrialParameter<bool> low_latency_renderer_include_predecode_buffer_;
niklase@google.com470e71d2011-07-07 08:21:25 +000076};
77
philipel9d3ab612015-12-21 04:12:39 -080078class VCMGenericDecoder {
philipel9d3ab612015-12-21 04:12:39 -080079 public:
Danil Chapovalov7b78a312021-08-06 12:30:02 +020080 explicit VCMGenericDecoder(VideoDecoder* decoder);
philipel9d3ab612015-12-21 04:12:39 -080081 ~VCMGenericDecoder();
niklase@google.com470e71d2011-07-07 08:21:25 +000082
philipel9d3ab612015-12-21 04:12:39 -080083 /**
Danil Chapovalov355b8d22021-08-13 16:50:37 +020084 * Initialize the decoder with the information from the `settings`
Yves Gerey665174f2018-06-19 15:03:05 +020085 */
Danil Chapovalov355b8d22021-08-13 16:50:37 +020086 bool Configure(const VideoDecoder::Settings& settings);
niklase@google.com470e71d2011-07-07 08:21:25 +000087
philipel9d3ab612015-12-21 04:12:39 -080088 /**
Yves Gerey665174f2018-06-19 15:03:05 +020089 * Decode to a raw I420 frame,
90 *
91 * inputVideoBuffer reference to encoded video frame
92 */
Johannes Kron05f84872020-01-16 14:09:33 +010093 int32_t Decode(const VCMEncodedFrame& inputFrame, Timestamp now);
niklase@google.com470e71d2011-07-07 08:21:25 +000094
philipel9d3ab612015-12-21 04:12:39 -080095 /**
Yves Gerey665174f2018-06-19 15:03:05 +020096 * Set decode callback. Deregistering while decoding is illegal.
97 */
philipel9d3ab612015-12-21 04:12:39 -080098 int32_t RegisterDecodeCompleteCallback(VCMDecodedFrameCallback* callback);
niklase@google.com470e71d2011-07-07 08:21:25 +000099
tommi5b7fc8c2017-07-05 16:45:57 -0700100 bool IsSameDecoder(VideoDecoder* decoder) const {
Danil Chapovalov7b78a312021-08-06 12:30:02 +0200101 return decoder_ == decoder;
tommi5b7fc8c2017-07-05 16:45:57 -0700102 }
niklase@google.com470e71d2011-07-07 08:21:25 +0000103
philipel9d3ab612015-12-21 04:12:39 -0800104 private:
Danil Chapovalov7b78a312021-08-06 12:30:02 +0200105 VCMDecodedFrameCallback* _callback = nullptr;
106 VideoDecoder* const decoder_;
ilnik00d802b2017-04-11 10:34:31 -0700107 VideoContentType _last_keyframe_content_type;
Erik Språngc12f6252021-01-13 21:49:59 +0100108 VideoDecoder::DecoderInfo decoder_info_;
niklase@google.com470e71d2011-07-07 08:21:25 +0000109};
110
pbos@webrtc.orgd900e8b2013-07-03 15:12:26 +0000111} // namespace webrtc
niklase@google.com470e71d2011-07-07 08:21:25 +0000112
Mirko Bonadei92ea95e2017-09-15 06:47:31 +0200113#endif // MODULES_VIDEO_CODING_GENERIC_DECODER_H_