blob: 3f4c8d6c79abc82f6b3a98f8f666189c386bb6b9 [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"
Artem Titovd15a5752021-02-10 14:31:24 +010019#include "api/sequence_checker.h"
philipel2fee4d62018-03-21 16:52:13 +010020#include "api/video/video_stream_decoder.h"
philipel97187112018-03-23 10:43:21 +010021#include "modules/video_coding/frame_buffer2.h"
philipel97187112018-03-23 10:43:21 +010022#include "modules/video_coding/timing.h"
philipel844876d2018-04-05 11:02:54 +020023#include "rtc_base/platform_thread.h"
Markus Handell265931e2020-07-10 12:23:48 +020024#include "rtc_base/synchronization/mutex.h"
philipel97187112018-03-23 10:43:21 +010025#include "rtc_base/task_queue.h"
philipel97187112018-03-23 10:43:21 +010026#include "system_wrappers/include/clock.h"
philipel2fee4d62018-03-21 16:52:13 +010027
28namespace webrtc {
29
philipel539f9b32020-01-09 16:12:25 +010030class VideoStreamDecoderImpl : public VideoStreamDecoderInterface {
philipel2fee4d62018-03-21 16:52:13 +010031 public:
32 VideoStreamDecoderImpl(
Danil Chapovalovb703db92019-04-08 16:59:28 +020033 VideoStreamDecoderInterface::Callbacks* callbacks,
philipel2fee4d62018-03-21 16:52:13 +010034 VideoDecoderFactory* decoder_factory,
Danil Chapovalovb703db92019-04-08 16:59:28 +020035 TaskQueueFactory* task_queue_factory,
philipel2fee4d62018-03-21 16:52:13 +010036 std::map<int, std::pair<SdpVideoFormat, int>> decoder_settings);
37
38 ~VideoStreamDecoderImpl() override;
39
40 void OnFrame(std::unique_ptr<video_coding::EncodedFrame> frame) override;
41
philipel781653c2019-06-04 17:10:37 +020042 void SetMinPlayoutDelay(TimeDelta min_delay) override;
43 void SetMaxPlayoutDelay(TimeDelta max_delay) override;
44
philipel2fee4d62018-03-21 16:52:13 +010045 private:
philipel539f9b32020-01-09 16:12:25 +010046 class DecodeCallbacks : public DecodedImageCallback {
47 public:
48 explicit DecodeCallbacks(VideoStreamDecoderImpl* video_stream_decoder_impl);
49 int32_t Decoded(VideoFrame& decodedImage) override;
50 int32_t Decoded(VideoFrame& decodedImage, int64_t decode_time_ms) override;
51 void Decoded(VideoFrame& decodedImage,
52 absl::optional<int32_t> decode_time_ms,
53 absl::optional<uint8_t> qp) override;
54
55 private:
56 VideoStreamDecoderImpl* const video_stream_decoder_impl_;
57 };
58
philipel844876d2018-04-05 11:02:54 +020059 enum DecodeResult {
60 kOk,
philipel539f9b32020-01-09 16:12:25 +010061 kOkRequestKeyframe,
philipel844876d2018-04-05 11:02:54 +020062 kDecodeFailure,
philipel844876d2018-04-05 11:02:54 +020063 };
64
philipel3dc47802020-09-10 15:30:02 +020065 struct FrameInfo {
66 int64_t timestamp = -1;
philipel6847f9b2018-04-20 15:05:37 +020067 int64_t decode_start_time_ms;
68 int64_t render_time_us;
philipel3dc47802020-09-10 15:30:02 +020069 VideoContentType content_type;
philipel6847f9b2018-04-20 15:05:37 +020070 };
71
philipel3dc47802020-09-10 15:30:02 +020072 void SaveFrameInfo(const video_coding::EncodedFrame& frame)
philipel539f9b32020-01-09 16:12:25 +010073 RTC_RUN_ON(bookkeeping_queue_);
philipel3dc47802020-09-10 15:30:02 +020074 FrameInfo* GetFrameInfo(int64_t timestamp) RTC_RUN_ON(bookkeeping_queue_);
philipel539f9b32020-01-09 16:12:25 +010075 void StartNextDecode() RTC_RUN_ON(bookkeeping_queue_);
76 void OnNextFrameCallback(std::unique_ptr<video_coding::EncodedFrame> frame,
77 video_coding::FrameBuffer::ReturnReason res)
78 RTC_RUN_ON(bookkeeping_queue_);
79 void OnDecodedFrameCallback(VideoFrame& decodedImage, // NOLINT
80 absl::optional<int32_t> decode_time_ms,
81 absl::optional<uint8_t> qp);
philipel79aab3f2018-03-26 14:31:23 +020082
philipel539f9b32020-01-09 16:12:25 +010083 VideoDecoder* GetDecoder(int payload_type) RTC_RUN_ON(decode_queue_);
84 VideoStreamDecoderImpl::DecodeResult DecodeFrame(
85 std::unique_ptr<video_coding::EncodedFrame> frame)
86 RTC_RUN_ON(decode_queue_);
philipel6847f9b2018-04-20 15:05:37 +020087
philipel97187112018-03-23 10:43:21 +010088 VCMTiming timing_;
philipel539f9b32020-01-09 16:12:25 +010089 DecodeCallbacks decode_callbacks_;
philipel844876d2018-04-05 11:02:54 +020090
philipel6847f9b2018-04-20 15:05:37 +020091 // Some decoders are pipelined so it is not sufficient to save frame info
92 // for the last frame only.
philipel3dc47802020-09-10 15:30:02 +020093 static constexpr int kFrameInfoMemory = 8;
94 std::array<FrameInfo, kFrameInfoMemory> frame_info_
philipel844876d2018-04-05 11:02:54 +020095 RTC_GUARDED_BY(bookkeeping_queue_);
philipel3dc47802020-09-10 15:30:02 +020096 int next_frame_info_index_ RTC_GUARDED_BY(bookkeeping_queue_);
philipel539f9b32020-01-09 16:12:25 +010097 VideoStreamDecoderInterface::Callbacks* const callbacks_
98 RTC_PT_GUARDED_BY(bookkeeping_queue_);
99 video_coding::VideoLayerFrameId last_continuous_id_
100 RTC_GUARDED_BY(bookkeeping_queue_);
101 bool keyframe_required_ RTC_GUARDED_BY(bookkeeping_queue_);
102
103 absl::optional<int> current_payload_type_ RTC_GUARDED_BY(decode_queue_);
104 VideoDecoderFactory* const decoder_factory_ RTC_PT_GUARDED_BY(decode_queue_);
105 std::map<int, std::pair<SdpVideoFormat, int>> decoder_settings_
106 RTC_GUARDED_BY(decode_queue_);
107
108 // The |bookkeeping_queue_| use the |frame_buffer_| and also posts tasks to
109 // the |decode_queue_|. The |decode_queue_| in turn use the |decoder_| to
110 // decode frames. When the |decoder_| is done it will post back to the
111 // |bookkeeping_queue_| with the decoded frame. During shutdown we start by
112 // isolating the |bookkeeping_queue_| from the |decode_queue_|, so now it's
113 // safe for the |decode_queue_| to be destructed. After that the |decoder_|
114 // can be destructed, and then the |bookkeeping_queue_|. Finally the
115 // |frame_buffer_| can be destructed.
Markus Handell265931e2020-07-10 12:23:48 +0200116 Mutex shut_down_mutex_;
117 bool shut_down_ RTC_GUARDED_BY(shut_down_mutex_);
philipel539f9b32020-01-09 16:12:25 +0100118 video_coding::FrameBuffer frame_buffer_ RTC_GUARDED_BY(bookkeeping_queue_);
119 rtc::TaskQueue bookkeeping_queue_;
120 std::unique_ptr<VideoDecoder> decoder_ RTC_GUARDED_BY(decode_queue_);
121 rtc::TaskQueue decode_queue_;
philipel2fee4d62018-03-21 16:52:13 +0100122};
123
124} // namespace webrtc
125
126#endif // VIDEO_VIDEO_STREAM_DECODER_IMPL_H_