xref: /aosp_15_r20/external/webrtc/video/video_stream_decoder_impl.h (revision d9f758449e529ab9291ac668be2861e7a55c2422)
1*d9f75844SAndroid Build Coastguard Worker /*
2*d9f75844SAndroid Build Coastguard Worker  *  Copyright (c) 2018 The WebRTC project authors. All Rights Reserved.
3*d9f75844SAndroid Build Coastguard Worker  *
4*d9f75844SAndroid Build Coastguard Worker  *  Use of this source code is governed by a BSD-style license
5*d9f75844SAndroid Build Coastguard Worker  *  that can be found in the LICENSE file in the root of the source
6*d9f75844SAndroid Build Coastguard Worker  *  tree. An additional intellectual property rights grant can be found
7*d9f75844SAndroid Build Coastguard Worker  *  in the file PATENTS.  All contributing project authors may
8*d9f75844SAndroid Build Coastguard Worker  *  be found in the AUTHORS file in the root of the source tree.
9*d9f75844SAndroid Build Coastguard Worker  */
10*d9f75844SAndroid Build Coastguard Worker 
11*d9f75844SAndroid Build Coastguard Worker #ifndef VIDEO_VIDEO_STREAM_DECODER_IMPL_H_
12*d9f75844SAndroid Build Coastguard Worker #define VIDEO_VIDEO_STREAM_DECODER_IMPL_H_
13*d9f75844SAndroid Build Coastguard Worker 
14*d9f75844SAndroid Build Coastguard Worker #include <map>
15*d9f75844SAndroid Build Coastguard Worker #include <memory>
16*d9f75844SAndroid Build Coastguard Worker #include <utility>
17*d9f75844SAndroid Build Coastguard Worker 
18*d9f75844SAndroid Build Coastguard Worker #include "absl/types/optional.h"
19*d9f75844SAndroid Build Coastguard Worker #include "api/field_trials_view.h"
20*d9f75844SAndroid Build Coastguard Worker #include "api/sequence_checker.h"
21*d9f75844SAndroid Build Coastguard Worker #include "api/transport/field_trial_based_config.h"
22*d9f75844SAndroid Build Coastguard Worker #include "api/video/video_stream_decoder.h"
23*d9f75844SAndroid Build Coastguard Worker #include "modules/video_coding/frame_buffer2.h"
24*d9f75844SAndroid Build Coastguard Worker #include "modules/video_coding/timing/timing.h"
25*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/memory/always_valid_pointer.h"
26*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/platform_thread.h"
27*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/synchronization/mutex.h"
28*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/task_queue.h"
29*d9f75844SAndroid Build Coastguard Worker #include "system_wrappers/include/clock.h"
30*d9f75844SAndroid Build Coastguard Worker 
31*d9f75844SAndroid Build Coastguard Worker namespace webrtc {
32*d9f75844SAndroid Build Coastguard Worker 
33*d9f75844SAndroid Build Coastguard Worker class VideoStreamDecoderImpl : public VideoStreamDecoderInterface {
34*d9f75844SAndroid Build Coastguard Worker  public:
35*d9f75844SAndroid Build Coastguard Worker   VideoStreamDecoderImpl(
36*d9f75844SAndroid Build Coastguard Worker       VideoStreamDecoderInterface::Callbacks* callbacks,
37*d9f75844SAndroid Build Coastguard Worker       VideoDecoderFactory* decoder_factory,
38*d9f75844SAndroid Build Coastguard Worker       TaskQueueFactory* task_queue_factory,
39*d9f75844SAndroid Build Coastguard Worker       std::map<int, std::pair<SdpVideoFormat, int>> decoder_settings,
40*d9f75844SAndroid Build Coastguard Worker       const FieldTrialsView* field_trials);
41*d9f75844SAndroid Build Coastguard Worker 
42*d9f75844SAndroid Build Coastguard Worker   ~VideoStreamDecoderImpl() override;
43*d9f75844SAndroid Build Coastguard Worker 
44*d9f75844SAndroid Build Coastguard Worker   void OnFrame(std::unique_ptr<EncodedFrame> frame) override;
45*d9f75844SAndroid Build Coastguard Worker 
46*d9f75844SAndroid Build Coastguard Worker   void SetMinPlayoutDelay(TimeDelta min_delay) override;
47*d9f75844SAndroid Build Coastguard Worker   void SetMaxPlayoutDelay(TimeDelta max_delay) override;
48*d9f75844SAndroid Build Coastguard Worker 
49*d9f75844SAndroid Build Coastguard Worker  private:
50*d9f75844SAndroid Build Coastguard Worker   class DecodeCallbacks : public DecodedImageCallback {
51*d9f75844SAndroid Build Coastguard Worker    public:
52*d9f75844SAndroid Build Coastguard Worker     explicit DecodeCallbacks(VideoStreamDecoderImpl* video_stream_decoder_impl);
53*d9f75844SAndroid Build Coastguard Worker     int32_t Decoded(VideoFrame& decodedImage) override;
54*d9f75844SAndroid Build Coastguard Worker     int32_t Decoded(VideoFrame& decodedImage, int64_t decode_time_ms) override;
55*d9f75844SAndroid Build Coastguard Worker     void Decoded(VideoFrame& decodedImage,
56*d9f75844SAndroid Build Coastguard Worker                  absl::optional<int32_t> decode_time_ms,
57*d9f75844SAndroid Build Coastguard Worker                  absl::optional<uint8_t> qp) override;
58*d9f75844SAndroid Build Coastguard Worker 
59*d9f75844SAndroid Build Coastguard Worker    private:
60*d9f75844SAndroid Build Coastguard Worker     VideoStreamDecoderImpl* const video_stream_decoder_impl_;
61*d9f75844SAndroid Build Coastguard Worker   };
62*d9f75844SAndroid Build Coastguard Worker 
63*d9f75844SAndroid Build Coastguard Worker   enum DecodeResult {
64*d9f75844SAndroid Build Coastguard Worker     kOk,
65*d9f75844SAndroid Build Coastguard Worker     kOkRequestKeyframe,
66*d9f75844SAndroid Build Coastguard Worker     kDecodeFailure,
67*d9f75844SAndroid Build Coastguard Worker   };
68*d9f75844SAndroid Build Coastguard Worker 
69*d9f75844SAndroid Build Coastguard Worker   struct FrameInfo {
70*d9f75844SAndroid Build Coastguard Worker     int64_t timestamp = -1;
71*d9f75844SAndroid Build Coastguard Worker     int64_t decode_start_time_ms;
72*d9f75844SAndroid Build Coastguard Worker     int64_t render_time_us;
73*d9f75844SAndroid Build Coastguard Worker     VideoContentType content_type;
74*d9f75844SAndroid Build Coastguard Worker   };
75*d9f75844SAndroid Build Coastguard Worker 
76*d9f75844SAndroid Build Coastguard Worker   void SaveFrameInfo(const EncodedFrame& frame) RTC_RUN_ON(bookkeeping_queue_);
77*d9f75844SAndroid Build Coastguard Worker   FrameInfo* GetFrameInfo(int64_t timestamp) RTC_RUN_ON(bookkeeping_queue_);
78*d9f75844SAndroid Build Coastguard Worker   void StartNextDecode() RTC_RUN_ON(bookkeeping_queue_);
79*d9f75844SAndroid Build Coastguard Worker   void OnNextFrameCallback(std::unique_ptr<EncodedFrame> frame)
80*d9f75844SAndroid Build Coastguard Worker       RTC_RUN_ON(bookkeeping_queue_);
81*d9f75844SAndroid Build Coastguard Worker   void OnDecodedFrameCallback(VideoFrame& decodedImage,  // NOLINT
82*d9f75844SAndroid Build Coastguard Worker                               absl::optional<int32_t> decode_time_ms,
83*d9f75844SAndroid Build Coastguard Worker                               absl::optional<uint8_t> qp);
84*d9f75844SAndroid Build Coastguard Worker 
85*d9f75844SAndroid Build Coastguard Worker   VideoDecoder* GetDecoder(int payload_type) RTC_RUN_ON(decode_queue_);
86*d9f75844SAndroid Build Coastguard Worker   VideoStreamDecoderImpl::DecodeResult DecodeFrame(
87*d9f75844SAndroid Build Coastguard Worker       std::unique_ptr<EncodedFrame> frame) RTC_RUN_ON(decode_queue_);
88*d9f75844SAndroid Build Coastguard Worker 
89*d9f75844SAndroid Build Coastguard Worker   AlwaysValidPointer<const FieldTrialsView, FieldTrialBasedConfig>
90*d9f75844SAndroid Build Coastguard Worker       field_trials_;
91*d9f75844SAndroid Build Coastguard Worker   VCMTiming timing_;
92*d9f75844SAndroid Build Coastguard Worker   DecodeCallbacks decode_callbacks_;
93*d9f75844SAndroid Build Coastguard Worker 
94*d9f75844SAndroid Build Coastguard Worker   // Some decoders are pipelined so it is not sufficient to save frame info
95*d9f75844SAndroid Build Coastguard Worker   // for the last frame only.
96*d9f75844SAndroid Build Coastguard Worker   static constexpr int kFrameInfoMemory = 8;
97*d9f75844SAndroid Build Coastguard Worker   std::array<FrameInfo, kFrameInfoMemory> frame_info_
98*d9f75844SAndroid Build Coastguard Worker       RTC_GUARDED_BY(bookkeeping_queue_);
99*d9f75844SAndroid Build Coastguard Worker   int next_frame_info_index_ RTC_GUARDED_BY(bookkeeping_queue_);
100*d9f75844SAndroid Build Coastguard Worker   VideoStreamDecoderInterface::Callbacks* const callbacks_
101*d9f75844SAndroid Build Coastguard Worker       RTC_PT_GUARDED_BY(bookkeeping_queue_);
102*d9f75844SAndroid Build Coastguard Worker   int64_t last_continuous_frame_id_ RTC_GUARDED_BY(bookkeeping_queue_) = -1;
103*d9f75844SAndroid Build Coastguard Worker   bool keyframe_required_ RTC_GUARDED_BY(bookkeeping_queue_);
104*d9f75844SAndroid Build Coastguard Worker 
105*d9f75844SAndroid Build Coastguard Worker   absl::optional<int> current_payload_type_ RTC_GUARDED_BY(decode_queue_);
106*d9f75844SAndroid Build Coastguard Worker   VideoDecoderFactory* const decoder_factory_ RTC_PT_GUARDED_BY(decode_queue_);
107*d9f75844SAndroid Build Coastguard Worker   std::map<int, std::pair<SdpVideoFormat, int>> decoder_settings_
108*d9f75844SAndroid Build Coastguard Worker       RTC_GUARDED_BY(decode_queue_);
109*d9f75844SAndroid Build Coastguard Worker 
110*d9f75844SAndroid Build Coastguard Worker   // The `bookkeeping_queue_` use the `frame_buffer_` and also posts tasks to
111*d9f75844SAndroid Build Coastguard Worker   // the `decode_queue_`. The `decode_queue_` in turn use the `decoder_` to
112*d9f75844SAndroid Build Coastguard Worker   // decode frames. When the `decoder_` is done it will post back to the
113*d9f75844SAndroid Build Coastguard Worker   // `bookkeeping_queue_` with the decoded frame. During shutdown we start by
114*d9f75844SAndroid Build Coastguard Worker   // isolating the `bookkeeping_queue_` from the `decode_queue_`, so now it's
115*d9f75844SAndroid Build Coastguard Worker   // safe for the `decode_queue_` to be destructed. After that the `decoder_`
116*d9f75844SAndroid Build Coastguard Worker   // can be destructed, and then the `bookkeeping_queue_`. Finally the
117*d9f75844SAndroid Build Coastguard Worker   // `frame_buffer_` can be destructed.
118*d9f75844SAndroid Build Coastguard Worker   Mutex shut_down_mutex_;
119*d9f75844SAndroid Build Coastguard Worker   bool shut_down_ RTC_GUARDED_BY(shut_down_mutex_);
120*d9f75844SAndroid Build Coastguard Worker   video_coding::FrameBuffer frame_buffer_ RTC_GUARDED_BY(bookkeeping_queue_);
121*d9f75844SAndroid Build Coastguard Worker   rtc::TaskQueue bookkeeping_queue_;
122*d9f75844SAndroid Build Coastguard Worker   std::unique_ptr<VideoDecoder> decoder_ RTC_GUARDED_BY(decode_queue_);
123*d9f75844SAndroid Build Coastguard Worker   rtc::TaskQueue decode_queue_;
124*d9f75844SAndroid Build Coastguard Worker };
125*d9f75844SAndroid Build Coastguard Worker 
126*d9f75844SAndroid Build Coastguard Worker }  // namespace webrtc
127*d9f75844SAndroid Build Coastguard Worker 
128*d9f75844SAndroid Build Coastguard Worker #endif  // VIDEO_VIDEO_STREAM_DECODER_IMPL_H_
129