blob: fcd715839174652361945a6086aec3aa9861779f [file] [log] [blame]
philipel2fee4d62018-03-21 16:52:13 +01001/*
2 * Copyright (c) 2018 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 */
10
11#ifndef VIDEO_VIDEO_STREAM_DECODER_IMPL_H_
12#define VIDEO_VIDEO_STREAM_DECODER_IMPL_H_
13
philipel2fee4d62018-03-21 16:52:13 +010014#include <map>
15#include <memory>
16#include <utility>
17
Danil Chapovalovb9b146c2018-06-15 12:28:07 +020018#include "absl/types/optional.h"
Jonas Orelande62c2f22022-03-29 11:04:48 +020019#include "api/field_trials_view.h"
Artem Titovd15a5752021-02-10 14:31:24 +010020#include "api/sequence_checker.h"
Jonas Orelande02f9ee2022-03-25 12:43:14 +010021#include "api/transport/field_trial_based_config.h"
philipel2fee4d62018-03-21 16:52:13 +010022#include "api/video/video_stream_decoder.h"
philipel97187112018-03-23 10:43:21 +010023#include "modules/video_coding/frame_buffer2.h"
Rasmus Brandtc4d253c2022-05-25 12:03:35 +020024#include "modules/video_coding/timing/timing.h"
Jonas Orelande02f9ee2022-03-25 12:43:14 +010025#include "rtc_base/memory/always_valid_pointer.h"
philipel844876d2018-04-05 11:02:54 +020026#include "rtc_base/platform_thread.h"
Markus Handell265931e2020-07-10 12:23:48 +020027#include "rtc_base/synchronization/mutex.h"
philipel97187112018-03-23 10:43:21 +010028#include "rtc_base/task_queue.h"
philipel97187112018-03-23 10:43:21 +010029#include "system_wrappers/include/clock.h"
philipel2fee4d62018-03-21 16:52:13 +010030
31namespace webrtc {
32
philipel539f9b32020-01-09 16:12:25 +010033class VideoStreamDecoderImpl : public VideoStreamDecoderInterface {
philipel2fee4d62018-03-21 16:52:13 +010034 public:
35 VideoStreamDecoderImpl(
Danil Chapovalovb703db92019-04-08 16:59:28 +020036 VideoStreamDecoderInterface::Callbacks* callbacks,
philipel2fee4d62018-03-21 16:52:13 +010037 VideoDecoderFactory* decoder_factory,
Danil Chapovalovb703db92019-04-08 16:59:28 +020038 TaskQueueFactory* task_queue_factory,
Jonas Orelande02f9ee2022-03-25 12:43:14 +010039 std::map<int, std::pair<SdpVideoFormat, int>> decoder_settings,
Jonas Orelande62c2f22022-03-29 11:04:48 +020040 const FieldTrialsView* field_trials);
philipel2fee4d62018-03-21 16:52:13 +010041
42 ~VideoStreamDecoderImpl() override;
43
philipelca188092021-03-23 12:00:49 +010044 void OnFrame(std::unique_ptr<EncodedFrame> frame) override;
philipel2fee4d62018-03-21 16:52:13 +010045
philipel781653c2019-06-04 17:10:37 +020046 void SetMinPlayoutDelay(TimeDelta min_delay) override;
47 void SetMaxPlayoutDelay(TimeDelta max_delay) override;
48
philipel2fee4d62018-03-21 16:52:13 +010049 private:
philipel539f9b32020-01-09 16:12:25 +010050 class DecodeCallbacks : public DecodedImageCallback {
51 public:
52 explicit DecodeCallbacks(VideoStreamDecoderImpl* video_stream_decoder_impl);
53 int32_t Decoded(VideoFrame& decodedImage) override;
54 int32_t Decoded(VideoFrame& decodedImage, int64_t decode_time_ms) override;
55 void Decoded(VideoFrame& decodedImage,
56 absl::optional<int32_t> decode_time_ms,
57 absl::optional<uint8_t> qp) override;
58
59 private:
60 VideoStreamDecoderImpl* const video_stream_decoder_impl_;
61 };
62
philipel844876d2018-04-05 11:02:54 +020063 enum DecodeResult {
64 kOk,
philipel539f9b32020-01-09 16:12:25 +010065 kOkRequestKeyframe,
philipel844876d2018-04-05 11:02:54 +020066 kDecodeFailure,
philipel844876d2018-04-05 11:02:54 +020067 };
68
philipel3dc47802020-09-10 15:30:02 +020069 struct FrameInfo {
70 int64_t timestamp = -1;
philipel6847f9b2018-04-20 15:05:37 +020071 int64_t decode_start_time_ms;
72 int64_t render_time_us;
philipel3dc47802020-09-10 15:30:02 +020073 VideoContentType content_type;
philipel6847f9b2018-04-20 15:05:37 +020074 };
75
philipelca188092021-03-23 12:00:49 +010076 void SaveFrameInfo(const EncodedFrame& frame) RTC_RUN_ON(bookkeeping_queue_);
philipel3dc47802020-09-10 15:30:02 +020077 FrameInfo* GetFrameInfo(int64_t timestamp) RTC_RUN_ON(bookkeeping_queue_);
philipel539f9b32020-01-09 16:12:25 +010078 void StartNextDecode() RTC_RUN_ON(bookkeeping_queue_);
Evan Shrubsole3d29efd2021-12-07 14:11:45 +010079 void OnNextFrameCallback(std::unique_ptr<EncodedFrame> frame)
philipel539f9b32020-01-09 16:12:25 +010080 RTC_RUN_ON(bookkeeping_queue_);
81 void OnDecodedFrameCallback(VideoFrame& decodedImage, // NOLINT
82 absl::optional<int32_t> decode_time_ms,
83 absl::optional<uint8_t> qp);
philipel79aab3f2018-03-26 14:31:23 +020084
philipel539f9b32020-01-09 16:12:25 +010085 VideoDecoder* GetDecoder(int payload_type) RTC_RUN_ON(decode_queue_);
86 VideoStreamDecoderImpl::DecodeResult DecodeFrame(
philipelca188092021-03-23 12:00:49 +010087 std::unique_ptr<EncodedFrame> frame) RTC_RUN_ON(decode_queue_);
philipel6847f9b2018-04-20 15:05:37 +020088
Jonas Orelande62c2f22022-03-29 11:04:48 +020089 AlwaysValidPointer<const FieldTrialsView, FieldTrialBasedConfig>
Jonas Orelande02f9ee2022-03-25 12:43:14 +010090 field_trials_;
philipel97187112018-03-23 10:43:21 +010091 VCMTiming timing_;
philipel539f9b32020-01-09 16:12:25 +010092 DecodeCallbacks decode_callbacks_;
philipel844876d2018-04-05 11:02:54 +020093
philipel6847f9b2018-04-20 15:05:37 +020094 // Some decoders are pipelined so it is not sufficient to save frame info
95 // for the last frame only.
philipel3dc47802020-09-10 15:30:02 +020096 static constexpr int kFrameInfoMemory = 8;
97 std::array<FrameInfo, kFrameInfoMemory> frame_info_
philipel844876d2018-04-05 11:02:54 +020098 RTC_GUARDED_BY(bookkeeping_queue_);
philipel3dc47802020-09-10 15:30:02 +020099 int next_frame_info_index_ RTC_GUARDED_BY(bookkeeping_queue_);
philipel539f9b32020-01-09 16:12:25 +0100100 VideoStreamDecoderInterface::Callbacks* const callbacks_
101 RTC_PT_GUARDED_BY(bookkeeping_queue_);
philipel9aa9b8d2021-02-15 13:31:29 +0100102 int64_t last_continuous_frame_id_ RTC_GUARDED_BY(bookkeeping_queue_) = -1;
philipel539f9b32020-01-09 16:12:25 +0100103 bool keyframe_required_ RTC_GUARDED_BY(bookkeeping_queue_);
104
105 absl::optional<int> current_payload_type_ RTC_GUARDED_BY(decode_queue_);
106 VideoDecoderFactory* const decoder_factory_ RTC_PT_GUARDED_BY(decode_queue_);
107 std::map<int, std::pair<SdpVideoFormat, int>> decoder_settings_
108 RTC_GUARDED_BY(decode_queue_);
109
Artem Titovab30d722021-07-27 16:22:11 +0200110 // The `bookkeeping_queue_` use the `frame_buffer_` and also posts tasks to
111 // the `decode_queue_`. The `decode_queue_` in turn use the `decoder_` to
112 // decode frames. When the `decoder_` is done it will post back to the
113 // `bookkeeping_queue_` with the decoded frame. During shutdown we start by
114 // isolating the `bookkeeping_queue_` from the `decode_queue_`, so now it's
115 // safe for the `decode_queue_` to be destructed. After that the `decoder_`
116 // can be destructed, and then the `bookkeeping_queue_`. Finally the
117 // `frame_buffer_` can be destructed.
Markus Handell265931e2020-07-10 12:23:48 +0200118 Mutex shut_down_mutex_;
119 bool shut_down_ RTC_GUARDED_BY(shut_down_mutex_);
philipel539f9b32020-01-09 16:12:25 +0100120 video_coding::FrameBuffer frame_buffer_ RTC_GUARDED_BY(bookkeeping_queue_);
121 rtc::TaskQueue bookkeeping_queue_;
122 std::unique_ptr<VideoDecoder> decoder_ RTC_GUARDED_BY(decode_queue_);
123 rtc::TaskQueue decode_queue_;
philipel2fee4d62018-03-21 16:52:13 +0100124};
125
126} // namespace webrtc
127
128#endif // VIDEO_VIDEO_STREAM_DECODER_IMPL_H_