1*d9f75844SAndroid Build Coastguard Worker /*
2*d9f75844SAndroid Build Coastguard Worker * Copyright (c) 2020 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 #include "video/video_receive_stream2.h"
12*d9f75844SAndroid Build Coastguard Worker
13*d9f75844SAndroid Build Coastguard Worker #include <stdlib.h>
14*d9f75844SAndroid Build Coastguard Worker #include <string.h>
15*d9f75844SAndroid Build Coastguard Worker
16*d9f75844SAndroid Build Coastguard Worker #include <algorithm>
17*d9f75844SAndroid Build Coastguard Worker #include <memory>
18*d9f75844SAndroid Build Coastguard Worker #include <set>
19*d9f75844SAndroid Build Coastguard Worker #include <string>
20*d9f75844SAndroid Build Coastguard Worker #include <utility>
21*d9f75844SAndroid Build Coastguard Worker
22*d9f75844SAndroid Build Coastguard Worker #include "absl/algorithm/container.h"
23*d9f75844SAndroid Build Coastguard Worker #include "absl/types/optional.h"
24*d9f75844SAndroid Build Coastguard Worker #include "api/array_view.h"
25*d9f75844SAndroid Build Coastguard Worker #include "api/crypto/frame_decryptor_interface.h"
26*d9f75844SAndroid Build Coastguard Worker #include "api/scoped_refptr.h"
27*d9f75844SAndroid Build Coastguard Worker #include "api/sequence_checker.h"
28*d9f75844SAndroid Build Coastguard Worker #include "api/task_queue/pending_task_safety_flag.h"
29*d9f75844SAndroid Build Coastguard Worker #include "api/task_queue/task_queue_base.h"
30*d9f75844SAndroid Build Coastguard Worker #include "api/units/frequency.h"
31*d9f75844SAndroid Build Coastguard Worker #include "api/units/time_delta.h"
32*d9f75844SAndroid Build Coastguard Worker #include "api/units/timestamp.h"
33*d9f75844SAndroid Build Coastguard Worker #include "api/video/encoded_image.h"
34*d9f75844SAndroid Build Coastguard Worker #include "api/video_codecs/sdp_video_format.h"
35*d9f75844SAndroid Build Coastguard Worker #include "api/video_codecs/video_codec.h"
36*d9f75844SAndroid Build Coastguard Worker #include "api/video_codecs/video_decoder_factory.h"
37*d9f75844SAndroid Build Coastguard Worker #include "call/rtp_stream_receiver_controller_interface.h"
38*d9f75844SAndroid Build Coastguard Worker #include "call/rtx_receive_stream.h"
39*d9f75844SAndroid Build Coastguard Worker #include "modules/video_coding/include/video_codec_interface.h"
40*d9f75844SAndroid Build Coastguard Worker #include "modules/video_coding/include/video_coding_defines.h"
41*d9f75844SAndroid Build Coastguard Worker #include "modules/video_coding/include/video_error_codes.h"
42*d9f75844SAndroid Build Coastguard Worker #include "modules/video_coding/timing/timing.h"
43*d9f75844SAndroid Build Coastguard Worker #include "modules/video_coding/utility/vp8_header_parser.h"
44*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/checks.h"
45*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/event.h"
46*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/logging.h"
47*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/strings/string_builder.h"
48*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/synchronization/mutex.h"
49*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/thread_annotations.h"
50*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/time_utils.h"
51*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/trace_event.h"
52*d9f75844SAndroid Build Coastguard Worker #include "system_wrappers/include/clock.h"
53*d9f75844SAndroid Build Coastguard Worker #include "video/call_stats2.h"
54*d9f75844SAndroid Build Coastguard Worker #include "video/frame_dumping_decoder.h"
55*d9f75844SAndroid Build Coastguard Worker #include "video/receive_statistics_proxy2.h"
56*d9f75844SAndroid Build Coastguard Worker #include "video/render/incoming_video_stream.h"
57*d9f75844SAndroid Build Coastguard Worker #include "video/task_queue_frame_decode_scheduler.h"
58*d9f75844SAndroid Build Coastguard Worker
59*d9f75844SAndroid Build Coastguard Worker namespace webrtc {
60*d9f75844SAndroid Build Coastguard Worker
61*d9f75844SAndroid Build Coastguard Worker namespace internal {
62*d9f75844SAndroid Build Coastguard Worker
63*d9f75844SAndroid Build Coastguard Worker namespace {
64*d9f75844SAndroid Build Coastguard Worker
65*d9f75844SAndroid Build Coastguard Worker // The default delay before re-requesting a key frame to be sent.
66*d9f75844SAndroid Build Coastguard Worker constexpr TimeDelta kMinBaseMinimumDelay = TimeDelta::Zero();
67*d9f75844SAndroid Build Coastguard Worker constexpr TimeDelta kMaxBaseMinimumDelay = TimeDelta::Seconds(10);
68*d9f75844SAndroid Build Coastguard Worker
69*d9f75844SAndroid Build Coastguard Worker // Concrete instance of RecordableEncodedFrame wrapping needed content
70*d9f75844SAndroid Build Coastguard Worker // from EncodedFrame.
71*d9f75844SAndroid Build Coastguard Worker class WebRtcRecordableEncodedFrame : public RecordableEncodedFrame {
72*d9f75844SAndroid Build Coastguard Worker public:
WebRtcRecordableEncodedFrame(const EncodedFrame & frame,RecordableEncodedFrame::EncodedResolution resolution)73*d9f75844SAndroid Build Coastguard Worker explicit WebRtcRecordableEncodedFrame(
74*d9f75844SAndroid Build Coastguard Worker const EncodedFrame& frame,
75*d9f75844SAndroid Build Coastguard Worker RecordableEncodedFrame::EncodedResolution resolution)
76*d9f75844SAndroid Build Coastguard Worker : buffer_(frame.GetEncodedData()),
77*d9f75844SAndroid Build Coastguard Worker render_time_ms_(frame.RenderTime()),
78*d9f75844SAndroid Build Coastguard Worker codec_(frame.CodecSpecific()->codecType),
79*d9f75844SAndroid Build Coastguard Worker is_key_frame_(frame.FrameType() == VideoFrameType::kVideoFrameKey),
80*d9f75844SAndroid Build Coastguard Worker resolution_(resolution) {
81*d9f75844SAndroid Build Coastguard Worker if (frame.ColorSpace()) {
82*d9f75844SAndroid Build Coastguard Worker color_space_ = *frame.ColorSpace();
83*d9f75844SAndroid Build Coastguard Worker }
84*d9f75844SAndroid Build Coastguard Worker }
85*d9f75844SAndroid Build Coastguard Worker
86*d9f75844SAndroid Build Coastguard Worker // VideoEncodedSinkInterface::FrameBuffer
encoded_buffer() const87*d9f75844SAndroid Build Coastguard Worker rtc::scoped_refptr<const EncodedImageBufferInterface> encoded_buffer()
88*d9f75844SAndroid Build Coastguard Worker const override {
89*d9f75844SAndroid Build Coastguard Worker return buffer_;
90*d9f75844SAndroid Build Coastguard Worker }
91*d9f75844SAndroid Build Coastguard Worker
color_space() const92*d9f75844SAndroid Build Coastguard Worker absl::optional<webrtc::ColorSpace> color_space() const override {
93*d9f75844SAndroid Build Coastguard Worker return color_space_;
94*d9f75844SAndroid Build Coastguard Worker }
95*d9f75844SAndroid Build Coastguard Worker
codec() const96*d9f75844SAndroid Build Coastguard Worker VideoCodecType codec() const override { return codec_; }
97*d9f75844SAndroid Build Coastguard Worker
is_key_frame() const98*d9f75844SAndroid Build Coastguard Worker bool is_key_frame() const override { return is_key_frame_; }
99*d9f75844SAndroid Build Coastguard Worker
resolution() const100*d9f75844SAndroid Build Coastguard Worker EncodedResolution resolution() const override { return resolution_; }
101*d9f75844SAndroid Build Coastguard Worker
render_time() const102*d9f75844SAndroid Build Coastguard Worker Timestamp render_time() const override {
103*d9f75844SAndroid Build Coastguard Worker return Timestamp::Millis(render_time_ms_);
104*d9f75844SAndroid Build Coastguard Worker }
105*d9f75844SAndroid Build Coastguard Worker
106*d9f75844SAndroid Build Coastguard Worker private:
107*d9f75844SAndroid Build Coastguard Worker rtc::scoped_refptr<EncodedImageBufferInterface> buffer_;
108*d9f75844SAndroid Build Coastguard Worker int64_t render_time_ms_;
109*d9f75844SAndroid Build Coastguard Worker VideoCodecType codec_;
110*d9f75844SAndroid Build Coastguard Worker bool is_key_frame_;
111*d9f75844SAndroid Build Coastguard Worker EncodedResolution resolution_;
112*d9f75844SAndroid Build Coastguard Worker absl::optional<webrtc::ColorSpace> color_space_;
113*d9f75844SAndroid Build Coastguard Worker };
114*d9f75844SAndroid Build Coastguard Worker
InitialDecoderResolution(const FieldTrialsView & field_trials)115*d9f75844SAndroid Build Coastguard Worker RenderResolution InitialDecoderResolution(const FieldTrialsView& field_trials) {
116*d9f75844SAndroid Build Coastguard Worker FieldTrialOptional<int> width("w");
117*d9f75844SAndroid Build Coastguard Worker FieldTrialOptional<int> height("h");
118*d9f75844SAndroid Build Coastguard Worker ParseFieldTrial({&width, &height},
119*d9f75844SAndroid Build Coastguard Worker field_trials.Lookup("WebRTC-Video-InitialDecoderResolution"));
120*d9f75844SAndroid Build Coastguard Worker if (width && height) {
121*d9f75844SAndroid Build Coastguard Worker return RenderResolution(width.Value(), height.Value());
122*d9f75844SAndroid Build Coastguard Worker }
123*d9f75844SAndroid Build Coastguard Worker
124*d9f75844SAndroid Build Coastguard Worker return RenderResolution(320, 180);
125*d9f75844SAndroid Build Coastguard Worker }
126*d9f75844SAndroid Build Coastguard Worker
127*d9f75844SAndroid Build Coastguard Worker // Video decoder class to be used for unknown codecs. Doesn't support decoding
128*d9f75844SAndroid Build Coastguard Worker // but logs messages to LS_ERROR.
129*d9f75844SAndroid Build Coastguard Worker class NullVideoDecoder : public webrtc::VideoDecoder {
130*d9f75844SAndroid Build Coastguard Worker public:
Configure(const Settings & settings)131*d9f75844SAndroid Build Coastguard Worker bool Configure(const Settings& settings) override {
132*d9f75844SAndroid Build Coastguard Worker RTC_LOG(LS_ERROR) << "Can't initialize NullVideoDecoder.";
133*d9f75844SAndroid Build Coastguard Worker return true;
134*d9f75844SAndroid Build Coastguard Worker }
135*d9f75844SAndroid Build Coastguard Worker
Decode(const webrtc::EncodedImage & input_image,bool missing_frames,int64_t render_time_ms)136*d9f75844SAndroid Build Coastguard Worker int32_t Decode(const webrtc::EncodedImage& input_image,
137*d9f75844SAndroid Build Coastguard Worker bool missing_frames,
138*d9f75844SAndroid Build Coastguard Worker int64_t render_time_ms) override {
139*d9f75844SAndroid Build Coastguard Worker RTC_LOG(LS_ERROR) << "The NullVideoDecoder doesn't support decoding.";
140*d9f75844SAndroid Build Coastguard Worker return WEBRTC_VIDEO_CODEC_OK;
141*d9f75844SAndroid Build Coastguard Worker }
142*d9f75844SAndroid Build Coastguard Worker
RegisterDecodeCompleteCallback(webrtc::DecodedImageCallback * callback)143*d9f75844SAndroid Build Coastguard Worker int32_t RegisterDecodeCompleteCallback(
144*d9f75844SAndroid Build Coastguard Worker webrtc::DecodedImageCallback* callback) override {
145*d9f75844SAndroid Build Coastguard Worker RTC_LOG(LS_ERROR)
146*d9f75844SAndroid Build Coastguard Worker << "Can't register decode complete callback on NullVideoDecoder.";
147*d9f75844SAndroid Build Coastguard Worker return WEBRTC_VIDEO_CODEC_OK;
148*d9f75844SAndroid Build Coastguard Worker }
149*d9f75844SAndroid Build Coastguard Worker
Release()150*d9f75844SAndroid Build Coastguard Worker int32_t Release() override { return WEBRTC_VIDEO_CODEC_OK; }
151*d9f75844SAndroid Build Coastguard Worker
ImplementationName() const152*d9f75844SAndroid Build Coastguard Worker const char* ImplementationName() const override { return "NullVideoDecoder"; }
153*d9f75844SAndroid Build Coastguard Worker };
154*d9f75844SAndroid Build Coastguard Worker
IsKeyFrameAndUnspecifiedResolution(const EncodedFrame & frame)155*d9f75844SAndroid Build Coastguard Worker bool IsKeyFrameAndUnspecifiedResolution(const EncodedFrame& frame) {
156*d9f75844SAndroid Build Coastguard Worker return frame.FrameType() == VideoFrameType::kVideoFrameKey &&
157*d9f75844SAndroid Build Coastguard Worker frame.EncodedImage()._encodedWidth == 0 &&
158*d9f75844SAndroid Build Coastguard Worker frame.EncodedImage()._encodedHeight == 0;
159*d9f75844SAndroid Build Coastguard Worker }
160*d9f75844SAndroid Build Coastguard Worker
OptionalDelayToLogString(const absl::optional<TimeDelta> opt)161*d9f75844SAndroid Build Coastguard Worker std::string OptionalDelayToLogString(const absl::optional<TimeDelta> opt) {
162*d9f75844SAndroid Build Coastguard Worker return opt.has_value() ? ToLogString(*opt) : "<unset>";
163*d9f75844SAndroid Build Coastguard Worker }
164*d9f75844SAndroid Build Coastguard Worker
165*d9f75844SAndroid Build Coastguard Worker } // namespace
166*d9f75844SAndroid Build Coastguard Worker
DetermineMaxWaitForFrame(TimeDelta rtp_history,bool is_keyframe)167*d9f75844SAndroid Build Coastguard Worker TimeDelta DetermineMaxWaitForFrame(TimeDelta rtp_history, bool is_keyframe) {
168*d9f75844SAndroid Build Coastguard Worker // A (arbitrary) conversion factor between the remotely signalled NACK buffer
169*d9f75844SAndroid Build Coastguard Worker // time (if not present defaults to 1000ms) and the maximum time we wait for a
170*d9f75844SAndroid Build Coastguard Worker // remote frame. Chosen to not change existing defaults when using not
171*d9f75844SAndroid Build Coastguard Worker // rtx-time.
172*d9f75844SAndroid Build Coastguard Worker const int conversion_factor = 3;
173*d9f75844SAndroid Build Coastguard Worker if (rtp_history > TimeDelta::Zero() &&
174*d9f75844SAndroid Build Coastguard Worker conversion_factor * rtp_history < kMaxWaitForFrame) {
175*d9f75844SAndroid Build Coastguard Worker return is_keyframe ? rtp_history : conversion_factor * rtp_history;
176*d9f75844SAndroid Build Coastguard Worker }
177*d9f75844SAndroid Build Coastguard Worker return is_keyframe ? kMaxWaitForKeyFrame : kMaxWaitForFrame;
178*d9f75844SAndroid Build Coastguard Worker }
179*d9f75844SAndroid Build Coastguard Worker
VideoReceiveStream2(TaskQueueFactory * task_queue_factory,Call * call,int num_cpu_cores,PacketRouter * packet_router,VideoReceiveStreamInterface::Config config,CallStats * call_stats,Clock * clock,std::unique_ptr<VCMTiming> timing,NackPeriodicProcessor * nack_periodic_processor,DecodeSynchronizer * decode_sync,RtcEventLog * event_log)180*d9f75844SAndroid Build Coastguard Worker VideoReceiveStream2::VideoReceiveStream2(
181*d9f75844SAndroid Build Coastguard Worker TaskQueueFactory* task_queue_factory,
182*d9f75844SAndroid Build Coastguard Worker Call* call,
183*d9f75844SAndroid Build Coastguard Worker int num_cpu_cores,
184*d9f75844SAndroid Build Coastguard Worker PacketRouter* packet_router,
185*d9f75844SAndroid Build Coastguard Worker VideoReceiveStreamInterface::Config config,
186*d9f75844SAndroid Build Coastguard Worker CallStats* call_stats,
187*d9f75844SAndroid Build Coastguard Worker Clock* clock,
188*d9f75844SAndroid Build Coastguard Worker std::unique_ptr<VCMTiming> timing,
189*d9f75844SAndroid Build Coastguard Worker NackPeriodicProcessor* nack_periodic_processor,
190*d9f75844SAndroid Build Coastguard Worker DecodeSynchronizer* decode_sync,
191*d9f75844SAndroid Build Coastguard Worker RtcEventLog* event_log)
192*d9f75844SAndroid Build Coastguard Worker : task_queue_factory_(task_queue_factory),
193*d9f75844SAndroid Build Coastguard Worker transport_adapter_(config.rtcp_send_transport),
194*d9f75844SAndroid Build Coastguard Worker config_(std::move(config)),
195*d9f75844SAndroid Build Coastguard Worker num_cpu_cores_(num_cpu_cores),
196*d9f75844SAndroid Build Coastguard Worker call_(call),
197*d9f75844SAndroid Build Coastguard Worker clock_(clock),
198*d9f75844SAndroid Build Coastguard Worker call_stats_(call_stats),
199*d9f75844SAndroid Build Coastguard Worker source_tracker_(clock_),
200*d9f75844SAndroid Build Coastguard Worker stats_proxy_(remote_ssrc(), clock_, call->worker_thread()),
201*d9f75844SAndroid Build Coastguard Worker rtp_receive_statistics_(ReceiveStatistics::Create(clock_)),
202*d9f75844SAndroid Build Coastguard Worker timing_(std::move(timing)),
203*d9f75844SAndroid Build Coastguard Worker video_receiver_(clock_, timing_.get(), call->trials()),
204*d9f75844SAndroid Build Coastguard Worker rtp_video_stream_receiver_(call->worker_thread(),
205*d9f75844SAndroid Build Coastguard Worker clock_,
206*d9f75844SAndroid Build Coastguard Worker &transport_adapter_,
207*d9f75844SAndroid Build Coastguard Worker call_stats->AsRtcpRttStats(),
208*d9f75844SAndroid Build Coastguard Worker packet_router,
209*d9f75844SAndroid Build Coastguard Worker &config_,
210*d9f75844SAndroid Build Coastguard Worker rtp_receive_statistics_.get(),
211*d9f75844SAndroid Build Coastguard Worker &stats_proxy_,
212*d9f75844SAndroid Build Coastguard Worker &stats_proxy_,
213*d9f75844SAndroid Build Coastguard Worker nack_periodic_processor,
214*d9f75844SAndroid Build Coastguard Worker this, // OnCompleteFrameCallback
215*d9f75844SAndroid Build Coastguard Worker std::move(config_.frame_decryptor),
216*d9f75844SAndroid Build Coastguard Worker std::move(config_.frame_transformer),
217*d9f75844SAndroid Build Coastguard Worker call->trials(),
218*d9f75844SAndroid Build Coastguard Worker event_log),
219*d9f75844SAndroid Build Coastguard Worker rtp_stream_sync_(call->worker_thread(), this),
220*d9f75844SAndroid Build Coastguard Worker max_wait_for_keyframe_(DetermineMaxWaitForFrame(
221*d9f75844SAndroid Build Coastguard Worker TimeDelta::Millis(config_.rtp.nack.rtp_history_ms),
222*d9f75844SAndroid Build Coastguard Worker true)),
223*d9f75844SAndroid Build Coastguard Worker max_wait_for_frame_(DetermineMaxWaitForFrame(
224*d9f75844SAndroid Build Coastguard Worker TimeDelta::Millis(config_.rtp.nack.rtp_history_ms),
225*d9f75844SAndroid Build Coastguard Worker false)),
226*d9f75844SAndroid Build Coastguard Worker decode_queue_(task_queue_factory_->CreateTaskQueue(
227*d9f75844SAndroid Build Coastguard Worker "DecodingQueue",
228*d9f75844SAndroid Build Coastguard Worker TaskQueueFactory::Priority::HIGH)) {
229*d9f75844SAndroid Build Coastguard Worker RTC_LOG(LS_INFO) << "VideoReceiveStream2: " << config_.ToString();
230*d9f75844SAndroid Build Coastguard Worker
231*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK(call_->worker_thread());
232*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK(config_.renderer);
233*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK(call_stats_);
234*d9f75844SAndroid Build Coastguard Worker packet_sequence_checker_.Detach();
235*d9f75844SAndroid Build Coastguard Worker
236*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK(!config_.decoders.empty());
237*d9f75844SAndroid Build Coastguard Worker RTC_CHECK(config_.decoder_factory);
238*d9f75844SAndroid Build Coastguard Worker std::set<int> decoder_payload_types;
239*d9f75844SAndroid Build Coastguard Worker for (const Decoder& decoder : config_.decoders) {
240*d9f75844SAndroid Build Coastguard Worker RTC_CHECK(decoder_payload_types.find(decoder.payload_type) ==
241*d9f75844SAndroid Build Coastguard Worker decoder_payload_types.end())
242*d9f75844SAndroid Build Coastguard Worker << "Duplicate payload type (" << decoder.payload_type
243*d9f75844SAndroid Build Coastguard Worker << ") for different decoders.";
244*d9f75844SAndroid Build Coastguard Worker decoder_payload_types.insert(decoder.payload_type);
245*d9f75844SAndroid Build Coastguard Worker }
246*d9f75844SAndroid Build Coastguard Worker
247*d9f75844SAndroid Build Coastguard Worker timing_->set_render_delay(TimeDelta::Millis(config_.render_delay_ms));
248*d9f75844SAndroid Build Coastguard Worker
249*d9f75844SAndroid Build Coastguard Worker std::unique_ptr<FrameDecodeScheduler> scheduler =
250*d9f75844SAndroid Build Coastguard Worker decode_sync ? decode_sync->CreateSynchronizedFrameScheduler()
251*d9f75844SAndroid Build Coastguard Worker : std::make_unique<TaskQueueFrameDecodeScheduler>(
252*d9f75844SAndroid Build Coastguard Worker clock, call_->worker_thread());
253*d9f75844SAndroid Build Coastguard Worker buffer_ = std::make_unique<VideoStreamBufferController>(
254*d9f75844SAndroid Build Coastguard Worker clock_, call_->worker_thread(), timing_.get(), &stats_proxy_, this,
255*d9f75844SAndroid Build Coastguard Worker max_wait_for_keyframe_, max_wait_for_frame_, std::move(scheduler),
256*d9f75844SAndroid Build Coastguard Worker call_->trials());
257*d9f75844SAndroid Build Coastguard Worker
258*d9f75844SAndroid Build Coastguard Worker if (rtx_ssrc()) {
259*d9f75844SAndroid Build Coastguard Worker rtx_receive_stream_ = std::make_unique<RtxReceiveStream>(
260*d9f75844SAndroid Build Coastguard Worker &rtp_video_stream_receiver_,
261*d9f75844SAndroid Build Coastguard Worker std::move(config_.rtp.rtx_associated_payload_types), remote_ssrc(),
262*d9f75844SAndroid Build Coastguard Worker rtp_receive_statistics_.get());
263*d9f75844SAndroid Build Coastguard Worker } else {
264*d9f75844SAndroid Build Coastguard Worker rtp_receive_statistics_->EnableRetransmitDetection(remote_ssrc(), true);
265*d9f75844SAndroid Build Coastguard Worker }
266*d9f75844SAndroid Build Coastguard Worker }
267*d9f75844SAndroid Build Coastguard Worker
~VideoReceiveStream2()268*d9f75844SAndroid Build Coastguard Worker VideoReceiveStream2::~VideoReceiveStream2() {
269*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&worker_sequence_checker_);
270*d9f75844SAndroid Build Coastguard Worker RTC_LOG(LS_INFO) << "~VideoReceiveStream2: " << config_.ToString();
271*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK(!media_receiver_);
272*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK(!rtx_receiver_);
273*d9f75844SAndroid Build Coastguard Worker Stop();
274*d9f75844SAndroid Build Coastguard Worker }
275*d9f75844SAndroid Build Coastguard Worker
RegisterWithTransport(RtpStreamReceiverControllerInterface * receiver_controller)276*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::RegisterWithTransport(
277*d9f75844SAndroid Build Coastguard Worker RtpStreamReceiverControllerInterface* receiver_controller) {
278*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
279*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK(!media_receiver_);
280*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK(!rtx_receiver_);
281*d9f75844SAndroid Build Coastguard Worker
282*d9f75844SAndroid Build Coastguard Worker // Register with RtpStreamReceiverController.
283*d9f75844SAndroid Build Coastguard Worker media_receiver_ = receiver_controller->CreateReceiver(
284*d9f75844SAndroid Build Coastguard Worker remote_ssrc(), &rtp_video_stream_receiver_);
285*d9f75844SAndroid Build Coastguard Worker if (rtx_ssrc()) {
286*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK(rtx_receive_stream_);
287*d9f75844SAndroid Build Coastguard Worker rtx_receiver_ = receiver_controller->CreateReceiver(
288*d9f75844SAndroid Build Coastguard Worker rtx_ssrc(), rtx_receive_stream_.get());
289*d9f75844SAndroid Build Coastguard Worker }
290*d9f75844SAndroid Build Coastguard Worker }
291*d9f75844SAndroid Build Coastguard Worker
UnregisterFromTransport()292*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::UnregisterFromTransport() {
293*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
294*d9f75844SAndroid Build Coastguard Worker media_receiver_.reset();
295*d9f75844SAndroid Build Coastguard Worker rtx_receiver_.reset();
296*d9f75844SAndroid Build Coastguard Worker }
297*d9f75844SAndroid Build Coastguard Worker
sync_group() const298*d9f75844SAndroid Build Coastguard Worker const std::string& VideoReceiveStream2::sync_group() const {
299*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
300*d9f75844SAndroid Build Coastguard Worker return config_.sync_group;
301*d9f75844SAndroid Build Coastguard Worker }
302*d9f75844SAndroid Build Coastguard Worker
SignalNetworkState(NetworkState state)303*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::SignalNetworkState(NetworkState state) {
304*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&worker_sequence_checker_);
305*d9f75844SAndroid Build Coastguard Worker rtp_video_stream_receiver_.SignalNetworkState(state);
306*d9f75844SAndroid Build Coastguard Worker }
307*d9f75844SAndroid Build Coastguard Worker
DeliverRtcp(const uint8_t * packet,size_t length)308*d9f75844SAndroid Build Coastguard Worker bool VideoReceiveStream2::DeliverRtcp(const uint8_t* packet, size_t length) {
309*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
310*d9f75844SAndroid Build Coastguard Worker return rtp_video_stream_receiver_.DeliverRtcp(packet, length);
311*d9f75844SAndroid Build Coastguard Worker }
312*d9f75844SAndroid Build Coastguard Worker
SetSync(Syncable * audio_syncable)313*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::SetSync(Syncable* audio_syncable) {
314*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
315*d9f75844SAndroid Build Coastguard Worker rtp_stream_sync_.ConfigureSync(audio_syncable);
316*d9f75844SAndroid Build Coastguard Worker }
317*d9f75844SAndroid Build Coastguard Worker
SetLocalSsrc(uint32_t local_ssrc)318*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::SetLocalSsrc(uint32_t local_ssrc) {
319*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
320*d9f75844SAndroid Build Coastguard Worker if (config_.rtp.local_ssrc == local_ssrc)
321*d9f75844SAndroid Build Coastguard Worker return;
322*d9f75844SAndroid Build Coastguard Worker
323*d9f75844SAndroid Build Coastguard Worker // TODO(tommi): Make sure we don't rely on local_ssrc via the config struct.
324*d9f75844SAndroid Build Coastguard Worker const_cast<uint32_t&>(config_.rtp.local_ssrc) = local_ssrc;
325*d9f75844SAndroid Build Coastguard Worker rtp_video_stream_receiver_.OnLocalSsrcChange(local_ssrc);
326*d9f75844SAndroid Build Coastguard Worker }
327*d9f75844SAndroid Build Coastguard Worker
Start()328*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::Start() {
329*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&worker_sequence_checker_);
330*d9f75844SAndroid Build Coastguard Worker
331*d9f75844SAndroid Build Coastguard Worker if (decoder_running_) {
332*d9f75844SAndroid Build Coastguard Worker return;
333*d9f75844SAndroid Build Coastguard Worker }
334*d9f75844SAndroid Build Coastguard Worker
335*d9f75844SAndroid Build Coastguard Worker const bool protected_by_fec =
336*d9f75844SAndroid Build Coastguard Worker config_.rtp.protected_by_flexfec ||
337*d9f75844SAndroid Build Coastguard Worker rtp_video_stream_receiver_.ulpfec_payload_type() != -1;
338*d9f75844SAndroid Build Coastguard Worker
339*d9f75844SAndroid Build Coastguard Worker if (config_.rtp.nack.rtp_history_ms > 0 && protected_by_fec) {
340*d9f75844SAndroid Build Coastguard Worker buffer_->SetProtectionMode(kProtectionNackFEC);
341*d9f75844SAndroid Build Coastguard Worker }
342*d9f75844SAndroid Build Coastguard Worker
343*d9f75844SAndroid Build Coastguard Worker transport_adapter_.Enable();
344*d9f75844SAndroid Build Coastguard Worker rtc::VideoSinkInterface<VideoFrame>* renderer = nullptr;
345*d9f75844SAndroid Build Coastguard Worker if (config_.enable_prerenderer_smoothing) {
346*d9f75844SAndroid Build Coastguard Worker incoming_video_stream_.reset(new IncomingVideoStream(
347*d9f75844SAndroid Build Coastguard Worker task_queue_factory_, config_.render_delay_ms, this));
348*d9f75844SAndroid Build Coastguard Worker renderer = incoming_video_stream_.get();
349*d9f75844SAndroid Build Coastguard Worker } else {
350*d9f75844SAndroid Build Coastguard Worker renderer = this;
351*d9f75844SAndroid Build Coastguard Worker }
352*d9f75844SAndroid Build Coastguard Worker
353*d9f75844SAndroid Build Coastguard Worker for (const Decoder& decoder : config_.decoders) {
354*d9f75844SAndroid Build Coastguard Worker VideoDecoder::Settings settings;
355*d9f75844SAndroid Build Coastguard Worker settings.set_codec_type(
356*d9f75844SAndroid Build Coastguard Worker PayloadStringToCodecType(decoder.video_format.name));
357*d9f75844SAndroid Build Coastguard Worker settings.set_max_render_resolution(
358*d9f75844SAndroid Build Coastguard Worker InitialDecoderResolution(call_->trials()));
359*d9f75844SAndroid Build Coastguard Worker settings.set_number_of_cores(num_cpu_cores_);
360*d9f75844SAndroid Build Coastguard Worker
361*d9f75844SAndroid Build Coastguard Worker const bool raw_payload =
362*d9f75844SAndroid Build Coastguard Worker config_.rtp.raw_payload_types.count(decoder.payload_type) > 0;
363*d9f75844SAndroid Build Coastguard Worker {
364*d9f75844SAndroid Build Coastguard Worker // TODO(bugs.webrtc.org/11993): Make this call on the network thread.
365*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
366*d9f75844SAndroid Build Coastguard Worker rtp_video_stream_receiver_.AddReceiveCodec(
367*d9f75844SAndroid Build Coastguard Worker decoder.payload_type, settings.codec_type(),
368*d9f75844SAndroid Build Coastguard Worker decoder.video_format.parameters, raw_payload);
369*d9f75844SAndroid Build Coastguard Worker }
370*d9f75844SAndroid Build Coastguard Worker video_receiver_.RegisterReceiveCodec(decoder.payload_type, settings);
371*d9f75844SAndroid Build Coastguard Worker }
372*d9f75844SAndroid Build Coastguard Worker
373*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK(renderer != nullptr);
374*d9f75844SAndroid Build Coastguard Worker video_stream_decoder_.reset(
375*d9f75844SAndroid Build Coastguard Worker new VideoStreamDecoder(&video_receiver_, &stats_proxy_, renderer));
376*d9f75844SAndroid Build Coastguard Worker
377*d9f75844SAndroid Build Coastguard Worker // Make sure we register as a stats observer *after* we've prepared the
378*d9f75844SAndroid Build Coastguard Worker // `video_stream_decoder_`.
379*d9f75844SAndroid Build Coastguard Worker call_stats_->RegisterStatsObserver(this);
380*d9f75844SAndroid Build Coastguard Worker
381*d9f75844SAndroid Build Coastguard Worker // Start decoding on task queue.
382*d9f75844SAndroid Build Coastguard Worker stats_proxy_.DecoderThreadStarting();
383*d9f75844SAndroid Build Coastguard Worker decode_queue_.PostTask([this] {
384*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&decode_queue_);
385*d9f75844SAndroid Build Coastguard Worker decoder_stopped_ = false;
386*d9f75844SAndroid Build Coastguard Worker });
387*d9f75844SAndroid Build Coastguard Worker buffer_->StartNextDecode(true);
388*d9f75844SAndroid Build Coastguard Worker decoder_running_ = true;
389*d9f75844SAndroid Build Coastguard Worker
390*d9f75844SAndroid Build Coastguard Worker {
391*d9f75844SAndroid Build Coastguard Worker // TODO(bugs.webrtc.org/11993): Make this call on the network thread.
392*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
393*d9f75844SAndroid Build Coastguard Worker rtp_video_stream_receiver_.StartReceive();
394*d9f75844SAndroid Build Coastguard Worker }
395*d9f75844SAndroid Build Coastguard Worker }
396*d9f75844SAndroid Build Coastguard Worker
Stop()397*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::Stop() {
398*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&worker_sequence_checker_);
399*d9f75844SAndroid Build Coastguard Worker
400*d9f75844SAndroid Build Coastguard Worker // TODO(bugs.webrtc.org/11993): Make this call on the network thread.
401*d9f75844SAndroid Build Coastguard Worker // Also call `GetUniqueFramesSeen()` at the same time (since it's a counter
402*d9f75844SAndroid Build Coastguard Worker // that's updated on the network thread).
403*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
404*d9f75844SAndroid Build Coastguard Worker rtp_video_stream_receiver_.StopReceive();
405*d9f75844SAndroid Build Coastguard Worker
406*d9f75844SAndroid Build Coastguard Worker stats_proxy_.OnUniqueFramesCounted(
407*d9f75844SAndroid Build Coastguard Worker rtp_video_stream_receiver_.GetUniqueFramesSeen());
408*d9f75844SAndroid Build Coastguard Worker
409*d9f75844SAndroid Build Coastguard Worker buffer_->Stop();
410*d9f75844SAndroid Build Coastguard Worker call_stats_->DeregisterStatsObserver(this);
411*d9f75844SAndroid Build Coastguard Worker
412*d9f75844SAndroid Build Coastguard Worker if (decoder_running_) {
413*d9f75844SAndroid Build Coastguard Worker rtc::Event done;
414*d9f75844SAndroid Build Coastguard Worker decode_queue_.PostTask([this, &done] {
415*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&decode_queue_);
416*d9f75844SAndroid Build Coastguard Worker // Set `decoder_stopped_` before deregistering all decoders. This means
417*d9f75844SAndroid Build Coastguard Worker // that any pending encoded frame will return early without trying to
418*d9f75844SAndroid Build Coastguard Worker // access the decoder database.
419*d9f75844SAndroid Build Coastguard Worker decoder_stopped_ = true;
420*d9f75844SAndroid Build Coastguard Worker for (const Decoder& decoder : config_.decoders) {
421*d9f75844SAndroid Build Coastguard Worker video_receiver_.RegisterExternalDecoder(nullptr, decoder.payload_type);
422*d9f75844SAndroid Build Coastguard Worker }
423*d9f75844SAndroid Build Coastguard Worker done.Set();
424*d9f75844SAndroid Build Coastguard Worker });
425*d9f75844SAndroid Build Coastguard Worker done.Wait(rtc::Event::kForever);
426*d9f75844SAndroid Build Coastguard Worker
427*d9f75844SAndroid Build Coastguard Worker decoder_running_ = false;
428*d9f75844SAndroid Build Coastguard Worker stats_proxy_.DecoderThreadStopped();
429*d9f75844SAndroid Build Coastguard Worker
430*d9f75844SAndroid Build Coastguard Worker UpdateHistograms();
431*d9f75844SAndroid Build Coastguard Worker }
432*d9f75844SAndroid Build Coastguard Worker
433*d9f75844SAndroid Build Coastguard Worker // TODO(bugs.webrtc.org/11993): Make these calls on the network thread.
434*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
435*d9f75844SAndroid Build Coastguard Worker rtp_video_stream_receiver_.RemoveReceiveCodecs();
436*d9f75844SAndroid Build Coastguard Worker video_receiver_.DeregisterReceiveCodecs();
437*d9f75844SAndroid Build Coastguard Worker
438*d9f75844SAndroid Build Coastguard Worker video_stream_decoder_.reset();
439*d9f75844SAndroid Build Coastguard Worker incoming_video_stream_.reset();
440*d9f75844SAndroid Build Coastguard Worker transport_adapter_.Disable();
441*d9f75844SAndroid Build Coastguard Worker }
442*d9f75844SAndroid Build Coastguard Worker
SetRtpExtensions(std::vector<RtpExtension> extensions)443*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::SetRtpExtensions(
444*d9f75844SAndroid Build Coastguard Worker std::vector<RtpExtension> extensions) {
445*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
446*d9f75844SAndroid Build Coastguard Worker rtp_video_stream_receiver_.SetRtpExtensions(extensions);
447*d9f75844SAndroid Build Coastguard Worker // TODO(tommi): We don't use the `c.rtp.extensions` member in the
448*d9f75844SAndroid Build Coastguard Worker // VideoReceiveStream2 class, so this const_cast<> is a temporary hack to keep
449*d9f75844SAndroid Build Coastguard Worker // things consistent between VideoReceiveStream2 and RtpVideoStreamReceiver2
450*d9f75844SAndroid Build Coastguard Worker // for debugging purposes. The `packet_sequence_checker_` gives us assurances
451*d9f75844SAndroid Build Coastguard Worker // that from a threading perspective, this is still safe. The accessors that
452*d9f75844SAndroid Build Coastguard Worker // give read access to this state, run behind the same check.
453*d9f75844SAndroid Build Coastguard Worker // The alternative to the const_cast<> would be to make `config_` non-const
454*d9f75844SAndroid Build Coastguard Worker // and guarded by `packet_sequence_checker_`. However the scope of that state
455*d9f75844SAndroid Build Coastguard Worker // is huge (the whole Config struct), and would require all methods that touch
456*d9f75844SAndroid Build Coastguard Worker // the struct to abide the needs of the `extensions` member.
457*d9f75844SAndroid Build Coastguard Worker const_cast<std::vector<RtpExtension>&>(config_.rtp.extensions) =
458*d9f75844SAndroid Build Coastguard Worker std::move(extensions);
459*d9f75844SAndroid Build Coastguard Worker }
460*d9f75844SAndroid Build Coastguard Worker
GetRtpExtensionMap() const461*d9f75844SAndroid Build Coastguard Worker RtpHeaderExtensionMap VideoReceiveStream2::GetRtpExtensionMap() const {
462*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
463*d9f75844SAndroid Build Coastguard Worker return rtp_video_stream_receiver_.GetRtpExtensions();
464*d9f75844SAndroid Build Coastguard Worker }
465*d9f75844SAndroid Build Coastguard Worker
transport_cc() const466*d9f75844SAndroid Build Coastguard Worker bool VideoReceiveStream2::transport_cc() const {
467*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
468*d9f75844SAndroid Build Coastguard Worker return config_.rtp.transport_cc;
469*d9f75844SAndroid Build Coastguard Worker }
470*d9f75844SAndroid Build Coastguard Worker
SetTransportCc(bool transport_cc)471*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::SetTransportCc(bool transport_cc) {
472*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
473*d9f75844SAndroid Build Coastguard Worker // TODO(tommi): Stop using the config struct for the internal state.
474*d9f75844SAndroid Build Coastguard Worker const_cast<bool&>(config_.rtp.transport_cc) = transport_cc;
475*d9f75844SAndroid Build Coastguard Worker }
476*d9f75844SAndroid Build Coastguard Worker
SetRtcpMode(RtcpMode mode)477*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::SetRtcpMode(RtcpMode mode) {
478*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
479*d9f75844SAndroid Build Coastguard Worker // TODO(tommi): Stop using the config struct for the internal state.
480*d9f75844SAndroid Build Coastguard Worker const_cast<RtcpMode&>(config_.rtp.rtcp_mode) = mode;
481*d9f75844SAndroid Build Coastguard Worker rtp_video_stream_receiver_.SetRtcpMode(mode);
482*d9f75844SAndroid Build Coastguard Worker }
483*d9f75844SAndroid Build Coastguard Worker
SetFlexFecProtection(RtpPacketSinkInterface * flexfec_sink)484*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::SetFlexFecProtection(
485*d9f75844SAndroid Build Coastguard Worker RtpPacketSinkInterface* flexfec_sink) {
486*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
487*d9f75844SAndroid Build Coastguard Worker rtp_video_stream_receiver_.SetPacketSink(flexfec_sink);
488*d9f75844SAndroid Build Coastguard Worker // TODO(tommi): Stop using the config struct for the internal state.
489*d9f75844SAndroid Build Coastguard Worker const_cast<RtpPacketSinkInterface*&>(config_.rtp.packet_sink_) = flexfec_sink;
490*d9f75844SAndroid Build Coastguard Worker const_cast<bool&>(config_.rtp.protected_by_flexfec) =
491*d9f75844SAndroid Build Coastguard Worker (flexfec_sink != nullptr);
492*d9f75844SAndroid Build Coastguard Worker }
493*d9f75844SAndroid Build Coastguard Worker
SetLossNotificationEnabled(bool enabled)494*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::SetLossNotificationEnabled(bool enabled) {
495*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
496*d9f75844SAndroid Build Coastguard Worker // TODO(tommi): Stop using the config struct for the internal state.
497*d9f75844SAndroid Build Coastguard Worker const_cast<bool&>(config_.rtp.lntf.enabled) = enabled;
498*d9f75844SAndroid Build Coastguard Worker rtp_video_stream_receiver_.SetLossNotificationEnabled(enabled);
499*d9f75844SAndroid Build Coastguard Worker }
500*d9f75844SAndroid Build Coastguard Worker
SetNackHistory(TimeDelta history)501*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::SetNackHistory(TimeDelta history) {
502*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
503*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_GE(history.ms(), 0);
504*d9f75844SAndroid Build Coastguard Worker
505*d9f75844SAndroid Build Coastguard Worker if (config_.rtp.nack.rtp_history_ms == history.ms())
506*d9f75844SAndroid Build Coastguard Worker return;
507*d9f75844SAndroid Build Coastguard Worker
508*d9f75844SAndroid Build Coastguard Worker // TODO(tommi): Stop using the config struct for the internal state.
509*d9f75844SAndroid Build Coastguard Worker const_cast<int&>(config_.rtp.nack.rtp_history_ms) = history.ms();
510*d9f75844SAndroid Build Coastguard Worker
511*d9f75844SAndroid Build Coastguard Worker const bool protected_by_fec =
512*d9f75844SAndroid Build Coastguard Worker config_.rtp.protected_by_flexfec ||
513*d9f75844SAndroid Build Coastguard Worker rtp_video_stream_receiver_.ulpfec_payload_type() != -1;
514*d9f75844SAndroid Build Coastguard Worker
515*d9f75844SAndroid Build Coastguard Worker buffer_->SetProtectionMode(history.ms() > 0 && protected_by_fec
516*d9f75844SAndroid Build Coastguard Worker ? kProtectionNackFEC
517*d9f75844SAndroid Build Coastguard Worker : kProtectionNack);
518*d9f75844SAndroid Build Coastguard Worker
519*d9f75844SAndroid Build Coastguard Worker rtp_video_stream_receiver_.SetNackHistory(history);
520*d9f75844SAndroid Build Coastguard Worker TimeDelta max_wait_for_keyframe = DetermineMaxWaitForFrame(history, true);
521*d9f75844SAndroid Build Coastguard Worker TimeDelta max_wait_for_frame = DetermineMaxWaitForFrame(history, false);
522*d9f75844SAndroid Build Coastguard Worker
523*d9f75844SAndroid Build Coastguard Worker max_wait_for_keyframe_ = max_wait_for_keyframe;
524*d9f75844SAndroid Build Coastguard Worker max_wait_for_frame_ = max_wait_for_frame;
525*d9f75844SAndroid Build Coastguard Worker
526*d9f75844SAndroid Build Coastguard Worker buffer_->SetMaxWaits(max_wait_for_keyframe, max_wait_for_frame);
527*d9f75844SAndroid Build Coastguard Worker }
528*d9f75844SAndroid Build Coastguard Worker
SetProtectionPayloadTypes(int red_payload_type,int ulpfec_payload_type)529*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::SetProtectionPayloadTypes(int red_payload_type,
530*d9f75844SAndroid Build Coastguard Worker int ulpfec_payload_type) {
531*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
532*d9f75844SAndroid Build Coastguard Worker rtp_video_stream_receiver_.SetProtectionPayloadTypes(red_payload_type,
533*d9f75844SAndroid Build Coastguard Worker ulpfec_payload_type);
534*d9f75844SAndroid Build Coastguard Worker }
535*d9f75844SAndroid Build Coastguard Worker
SetRtcpXr(Config::Rtp::RtcpXr rtcp_xr)536*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::SetRtcpXr(Config::Rtp::RtcpXr rtcp_xr) {
537*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
538*d9f75844SAndroid Build Coastguard Worker rtp_video_stream_receiver_.SetReferenceTimeReport(
539*d9f75844SAndroid Build Coastguard Worker rtcp_xr.receiver_reference_time_report);
540*d9f75844SAndroid Build Coastguard Worker }
541*d9f75844SAndroid Build Coastguard Worker
SetAssociatedPayloadTypes(std::map<int,int> associated_payload_types)542*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::SetAssociatedPayloadTypes(
543*d9f75844SAndroid Build Coastguard Worker std::map<int, int> associated_payload_types) {
544*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
545*d9f75844SAndroid Build Coastguard Worker
546*d9f75844SAndroid Build Coastguard Worker // For setting the associated payload types after construction, we currently
547*d9f75844SAndroid Build Coastguard Worker // assume that the rtx_ssrc cannot change. In such a case we can know that
548*d9f75844SAndroid Build Coastguard Worker // if the ssrc is non-0, a `rtx_receive_stream_` instance has previously been
549*d9f75844SAndroid Build Coastguard Worker // created and configured (and is referenced by `rtx_receiver_`) and we can
550*d9f75844SAndroid Build Coastguard Worker // simply reconfigure it.
551*d9f75844SAndroid Build Coastguard Worker // If rtx_ssrc is 0 however, we ignore this call.
552*d9f75844SAndroid Build Coastguard Worker if (!rtx_ssrc())
553*d9f75844SAndroid Build Coastguard Worker return;
554*d9f75844SAndroid Build Coastguard Worker
555*d9f75844SAndroid Build Coastguard Worker rtx_receive_stream_->SetAssociatedPayloadTypes(
556*d9f75844SAndroid Build Coastguard Worker std::move(associated_payload_types));
557*d9f75844SAndroid Build Coastguard Worker }
558*d9f75844SAndroid Build Coastguard Worker
CreateAndRegisterExternalDecoder(const Decoder & decoder)559*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::CreateAndRegisterExternalDecoder(
560*d9f75844SAndroid Build Coastguard Worker const Decoder& decoder) {
561*d9f75844SAndroid Build Coastguard Worker TRACE_EVENT0("webrtc",
562*d9f75844SAndroid Build Coastguard Worker "VideoReceiveStream2::CreateAndRegisterExternalDecoder");
563*d9f75844SAndroid Build Coastguard Worker std::unique_ptr<VideoDecoder> video_decoder =
564*d9f75844SAndroid Build Coastguard Worker config_.decoder_factory->CreateVideoDecoder(decoder.video_format);
565*d9f75844SAndroid Build Coastguard Worker // If we still have no valid decoder, we have to create a "Null" decoder
566*d9f75844SAndroid Build Coastguard Worker // that ignores all calls. The reason we can get into this state is that the
567*d9f75844SAndroid Build Coastguard Worker // old decoder factory interface doesn't have a way to query supported
568*d9f75844SAndroid Build Coastguard Worker // codecs.
569*d9f75844SAndroid Build Coastguard Worker if (!video_decoder) {
570*d9f75844SAndroid Build Coastguard Worker video_decoder = std::make_unique<NullVideoDecoder>();
571*d9f75844SAndroid Build Coastguard Worker }
572*d9f75844SAndroid Build Coastguard Worker
573*d9f75844SAndroid Build Coastguard Worker std::string decoded_output_file =
574*d9f75844SAndroid Build Coastguard Worker call_->trials().Lookup("WebRTC-DecoderDataDumpDirectory");
575*d9f75844SAndroid Build Coastguard Worker // Because '/' can't be used inside a field trial parameter, we use ';'
576*d9f75844SAndroid Build Coastguard Worker // instead.
577*d9f75844SAndroid Build Coastguard Worker // This is only relevant to WebRTC-DecoderDataDumpDirectory
578*d9f75844SAndroid Build Coastguard Worker // field trial. ';' is chosen arbitrary. Even though it's a legal character
579*d9f75844SAndroid Build Coastguard Worker // in some file systems, we can sacrifice ability to use it in the path to
580*d9f75844SAndroid Build Coastguard Worker // dumped video, since it's developers-only feature for debugging.
581*d9f75844SAndroid Build Coastguard Worker absl::c_replace(decoded_output_file, ';', '/');
582*d9f75844SAndroid Build Coastguard Worker if (!decoded_output_file.empty()) {
583*d9f75844SAndroid Build Coastguard Worker char filename_buffer[256];
584*d9f75844SAndroid Build Coastguard Worker rtc::SimpleStringBuilder ssb(filename_buffer);
585*d9f75844SAndroid Build Coastguard Worker ssb << decoded_output_file << "/webrtc_receive_stream_" << remote_ssrc()
586*d9f75844SAndroid Build Coastguard Worker << "-" << rtc::TimeMicros() << ".ivf";
587*d9f75844SAndroid Build Coastguard Worker video_decoder = CreateFrameDumpingDecoderWrapper(
588*d9f75844SAndroid Build Coastguard Worker std::move(video_decoder), FileWrapper::OpenWriteOnly(ssb.str()));
589*d9f75844SAndroid Build Coastguard Worker }
590*d9f75844SAndroid Build Coastguard Worker
591*d9f75844SAndroid Build Coastguard Worker video_receiver_.RegisterExternalDecoder(std::move(video_decoder),
592*d9f75844SAndroid Build Coastguard Worker decoder.payload_type);
593*d9f75844SAndroid Build Coastguard Worker }
594*d9f75844SAndroid Build Coastguard Worker
GetStats() const595*d9f75844SAndroid Build Coastguard Worker VideoReceiveStreamInterface::Stats VideoReceiveStream2::GetStats() const {
596*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&worker_sequence_checker_);
597*d9f75844SAndroid Build Coastguard Worker VideoReceiveStream2::Stats stats = stats_proxy_.GetStats();
598*d9f75844SAndroid Build Coastguard Worker stats.total_bitrate_bps = 0;
599*d9f75844SAndroid Build Coastguard Worker StreamStatistician* statistician =
600*d9f75844SAndroid Build Coastguard Worker rtp_receive_statistics_->GetStatistician(stats.ssrc);
601*d9f75844SAndroid Build Coastguard Worker if (statistician) {
602*d9f75844SAndroid Build Coastguard Worker stats.rtp_stats = statistician->GetStats();
603*d9f75844SAndroid Build Coastguard Worker stats.total_bitrate_bps = statistician->BitrateReceived();
604*d9f75844SAndroid Build Coastguard Worker }
605*d9f75844SAndroid Build Coastguard Worker if (rtx_ssrc()) {
606*d9f75844SAndroid Build Coastguard Worker StreamStatistician* rtx_statistician =
607*d9f75844SAndroid Build Coastguard Worker rtp_receive_statistics_->GetStatistician(rtx_ssrc());
608*d9f75844SAndroid Build Coastguard Worker if (rtx_statistician)
609*d9f75844SAndroid Build Coastguard Worker stats.total_bitrate_bps += rtx_statistician->BitrateReceived();
610*d9f75844SAndroid Build Coastguard Worker }
611*d9f75844SAndroid Build Coastguard Worker return stats;
612*d9f75844SAndroid Build Coastguard Worker }
613*d9f75844SAndroid Build Coastguard Worker
UpdateHistograms()614*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::UpdateHistograms() {
615*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&worker_sequence_checker_);
616*d9f75844SAndroid Build Coastguard Worker absl::optional<int> fraction_lost;
617*d9f75844SAndroid Build Coastguard Worker StreamDataCounters rtp_stats;
618*d9f75844SAndroid Build Coastguard Worker StreamStatistician* statistician =
619*d9f75844SAndroid Build Coastguard Worker rtp_receive_statistics_->GetStatistician(remote_ssrc());
620*d9f75844SAndroid Build Coastguard Worker if (statistician) {
621*d9f75844SAndroid Build Coastguard Worker fraction_lost = statistician->GetFractionLostInPercent();
622*d9f75844SAndroid Build Coastguard Worker rtp_stats = statistician->GetReceiveStreamDataCounters();
623*d9f75844SAndroid Build Coastguard Worker }
624*d9f75844SAndroid Build Coastguard Worker if (rtx_ssrc()) {
625*d9f75844SAndroid Build Coastguard Worker StreamStatistician* rtx_statistician =
626*d9f75844SAndroid Build Coastguard Worker rtp_receive_statistics_->GetStatistician(rtx_ssrc());
627*d9f75844SAndroid Build Coastguard Worker if (rtx_statistician) {
628*d9f75844SAndroid Build Coastguard Worker StreamDataCounters rtx_stats =
629*d9f75844SAndroid Build Coastguard Worker rtx_statistician->GetReceiveStreamDataCounters();
630*d9f75844SAndroid Build Coastguard Worker stats_proxy_.UpdateHistograms(fraction_lost, rtp_stats, &rtx_stats);
631*d9f75844SAndroid Build Coastguard Worker return;
632*d9f75844SAndroid Build Coastguard Worker }
633*d9f75844SAndroid Build Coastguard Worker }
634*d9f75844SAndroid Build Coastguard Worker stats_proxy_.UpdateHistograms(fraction_lost, rtp_stats, nullptr);
635*d9f75844SAndroid Build Coastguard Worker }
636*d9f75844SAndroid Build Coastguard Worker
SetBaseMinimumPlayoutDelayMs(int delay_ms)637*d9f75844SAndroid Build Coastguard Worker bool VideoReceiveStream2::SetBaseMinimumPlayoutDelayMs(int delay_ms) {
638*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&worker_sequence_checker_);
639*d9f75844SAndroid Build Coastguard Worker TimeDelta delay = TimeDelta::Millis(delay_ms);
640*d9f75844SAndroid Build Coastguard Worker if (delay < kMinBaseMinimumDelay || delay > kMaxBaseMinimumDelay) {
641*d9f75844SAndroid Build Coastguard Worker return false;
642*d9f75844SAndroid Build Coastguard Worker }
643*d9f75844SAndroid Build Coastguard Worker
644*d9f75844SAndroid Build Coastguard Worker base_minimum_playout_delay_ = delay;
645*d9f75844SAndroid Build Coastguard Worker UpdatePlayoutDelays();
646*d9f75844SAndroid Build Coastguard Worker return true;
647*d9f75844SAndroid Build Coastguard Worker }
648*d9f75844SAndroid Build Coastguard Worker
GetBaseMinimumPlayoutDelayMs() const649*d9f75844SAndroid Build Coastguard Worker int VideoReceiveStream2::GetBaseMinimumPlayoutDelayMs() const {
650*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&worker_sequence_checker_);
651*d9f75844SAndroid Build Coastguard Worker constexpr TimeDelta kDefaultBaseMinPlayoutDelay = TimeDelta::Millis(-1);
652*d9f75844SAndroid Build Coastguard Worker // Unset must be -1.
653*d9f75844SAndroid Build Coastguard Worker static_assert(-1 == kDefaultBaseMinPlayoutDelay.ms(), "");
654*d9f75844SAndroid Build Coastguard Worker return base_minimum_playout_delay_.value_or(kDefaultBaseMinPlayoutDelay).ms();
655*d9f75844SAndroid Build Coastguard Worker }
656*d9f75844SAndroid Build Coastguard Worker
OnFrame(const VideoFrame & video_frame)657*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::OnFrame(const VideoFrame& video_frame) {
658*d9f75844SAndroid Build Coastguard Worker VideoFrameMetaData frame_meta(video_frame, clock_->CurrentTime());
659*d9f75844SAndroid Build Coastguard Worker
660*d9f75844SAndroid Build Coastguard Worker // TODO(bugs.webrtc.org/10739): we should set local capture clock offset for
661*d9f75844SAndroid Build Coastguard Worker // `video_frame.packet_infos`. But VideoFrame is const qualified here.
662*d9f75844SAndroid Build Coastguard Worker
663*d9f75844SAndroid Build Coastguard Worker call_->worker_thread()->PostTask(
664*d9f75844SAndroid Build Coastguard Worker SafeTask(task_safety_.flag(), [frame_meta, this]() {
665*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&worker_sequence_checker_);
666*d9f75844SAndroid Build Coastguard Worker int64_t video_playout_ntp_ms;
667*d9f75844SAndroid Build Coastguard Worker int64_t sync_offset_ms;
668*d9f75844SAndroid Build Coastguard Worker double estimated_freq_khz;
669*d9f75844SAndroid Build Coastguard Worker if (rtp_stream_sync_.GetStreamSyncOffsetInMs(
670*d9f75844SAndroid Build Coastguard Worker frame_meta.rtp_timestamp, frame_meta.render_time_ms(),
671*d9f75844SAndroid Build Coastguard Worker &video_playout_ntp_ms, &sync_offset_ms, &estimated_freq_khz)) {
672*d9f75844SAndroid Build Coastguard Worker stats_proxy_.OnSyncOffsetUpdated(video_playout_ntp_ms, sync_offset_ms,
673*d9f75844SAndroid Build Coastguard Worker estimated_freq_khz);
674*d9f75844SAndroid Build Coastguard Worker }
675*d9f75844SAndroid Build Coastguard Worker stats_proxy_.OnRenderedFrame(frame_meta);
676*d9f75844SAndroid Build Coastguard Worker }));
677*d9f75844SAndroid Build Coastguard Worker
678*d9f75844SAndroid Build Coastguard Worker source_tracker_.OnFrameDelivered(video_frame.packet_infos());
679*d9f75844SAndroid Build Coastguard Worker config_.renderer->OnFrame(video_frame);
680*d9f75844SAndroid Build Coastguard Worker webrtc::MutexLock lock(&pending_resolution_mutex_);
681*d9f75844SAndroid Build Coastguard Worker if (pending_resolution_.has_value()) {
682*d9f75844SAndroid Build Coastguard Worker if (!pending_resolution_->empty() &&
683*d9f75844SAndroid Build Coastguard Worker (video_frame.width() != static_cast<int>(pending_resolution_->width) ||
684*d9f75844SAndroid Build Coastguard Worker video_frame.height() !=
685*d9f75844SAndroid Build Coastguard Worker static_cast<int>(pending_resolution_->height))) {
686*d9f75844SAndroid Build Coastguard Worker RTC_LOG(LS_WARNING)
687*d9f75844SAndroid Build Coastguard Worker << "Recordable encoded frame stream resolution was reported as "
688*d9f75844SAndroid Build Coastguard Worker << pending_resolution_->width << "x" << pending_resolution_->height
689*d9f75844SAndroid Build Coastguard Worker << " but the stream is now " << video_frame.width()
690*d9f75844SAndroid Build Coastguard Worker << video_frame.height();
691*d9f75844SAndroid Build Coastguard Worker }
692*d9f75844SAndroid Build Coastguard Worker pending_resolution_ = RecordableEncodedFrame::EncodedResolution{
693*d9f75844SAndroid Build Coastguard Worker static_cast<unsigned>(video_frame.width()),
694*d9f75844SAndroid Build Coastguard Worker static_cast<unsigned>(video_frame.height())};
695*d9f75844SAndroid Build Coastguard Worker }
696*d9f75844SAndroid Build Coastguard Worker }
697*d9f75844SAndroid Build Coastguard Worker
SetFrameDecryptor(rtc::scoped_refptr<webrtc::FrameDecryptorInterface> frame_decryptor)698*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::SetFrameDecryptor(
699*d9f75844SAndroid Build Coastguard Worker rtc::scoped_refptr<webrtc::FrameDecryptorInterface> frame_decryptor) {
700*d9f75844SAndroid Build Coastguard Worker rtp_video_stream_receiver_.SetFrameDecryptor(std::move(frame_decryptor));
701*d9f75844SAndroid Build Coastguard Worker }
702*d9f75844SAndroid Build Coastguard Worker
SetDepacketizerToDecoderFrameTransformer(rtc::scoped_refptr<FrameTransformerInterface> frame_transformer)703*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::SetDepacketizerToDecoderFrameTransformer(
704*d9f75844SAndroid Build Coastguard Worker rtc::scoped_refptr<FrameTransformerInterface> frame_transformer) {
705*d9f75844SAndroid Build Coastguard Worker rtp_video_stream_receiver_.SetDepacketizerToDecoderFrameTransformer(
706*d9f75844SAndroid Build Coastguard Worker std::move(frame_transformer));
707*d9f75844SAndroid Build Coastguard Worker }
708*d9f75844SAndroid Build Coastguard Worker
RequestKeyFrame(Timestamp now)709*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::RequestKeyFrame(Timestamp now) {
710*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&worker_sequence_checker_);
711*d9f75844SAndroid Build Coastguard Worker // Called from RtpVideoStreamReceiver (rtp_video_stream_receiver_ is
712*d9f75844SAndroid Build Coastguard Worker // ultimately responsible).
713*d9f75844SAndroid Build Coastguard Worker rtp_video_stream_receiver_.RequestKeyFrame();
714*d9f75844SAndroid Build Coastguard Worker last_keyframe_request_ = now;
715*d9f75844SAndroid Build Coastguard Worker }
716*d9f75844SAndroid Build Coastguard Worker
OnCompleteFrame(std::unique_ptr<EncodedFrame> frame)717*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::OnCompleteFrame(std::unique_ptr<EncodedFrame> frame) {
718*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&worker_sequence_checker_);
719*d9f75844SAndroid Build Coastguard Worker
720*d9f75844SAndroid Build Coastguard Worker const VideoPlayoutDelay& playout_delay = frame->EncodedImage().playout_delay_;
721*d9f75844SAndroid Build Coastguard Worker if (playout_delay.min_ms >= 0) {
722*d9f75844SAndroid Build Coastguard Worker frame_minimum_playout_delay_ = TimeDelta::Millis(playout_delay.min_ms);
723*d9f75844SAndroid Build Coastguard Worker UpdatePlayoutDelays();
724*d9f75844SAndroid Build Coastguard Worker }
725*d9f75844SAndroid Build Coastguard Worker if (playout_delay.max_ms >= 0) {
726*d9f75844SAndroid Build Coastguard Worker frame_maximum_playout_delay_ = TimeDelta::Millis(playout_delay.max_ms);
727*d9f75844SAndroid Build Coastguard Worker UpdatePlayoutDelays();
728*d9f75844SAndroid Build Coastguard Worker }
729*d9f75844SAndroid Build Coastguard Worker
730*d9f75844SAndroid Build Coastguard Worker auto last_continuous_pid = buffer_->InsertFrame(std::move(frame));
731*d9f75844SAndroid Build Coastguard Worker if (last_continuous_pid.has_value()) {
732*d9f75844SAndroid Build Coastguard Worker {
733*d9f75844SAndroid Build Coastguard Worker // TODO(bugs.webrtc.org/11993): Call on the network thread.
734*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
735*d9f75844SAndroid Build Coastguard Worker rtp_video_stream_receiver_.FrameContinuous(*last_continuous_pid);
736*d9f75844SAndroid Build Coastguard Worker }
737*d9f75844SAndroid Build Coastguard Worker }
738*d9f75844SAndroid Build Coastguard Worker }
739*d9f75844SAndroid Build Coastguard Worker
OnRttUpdate(int64_t avg_rtt_ms,int64_t max_rtt_ms)740*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::OnRttUpdate(int64_t avg_rtt_ms, int64_t max_rtt_ms) {
741*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&worker_sequence_checker_);
742*d9f75844SAndroid Build Coastguard Worker // TODO(bugs.webrtc.org/13757): Replace with TimeDelta.
743*d9f75844SAndroid Build Coastguard Worker buffer_->UpdateRtt(max_rtt_ms);
744*d9f75844SAndroid Build Coastguard Worker rtp_video_stream_receiver_.UpdateRtt(max_rtt_ms);
745*d9f75844SAndroid Build Coastguard Worker stats_proxy_.OnRttUpdate(avg_rtt_ms);
746*d9f75844SAndroid Build Coastguard Worker }
747*d9f75844SAndroid Build Coastguard Worker
id() const748*d9f75844SAndroid Build Coastguard Worker uint32_t VideoReceiveStream2::id() const {
749*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&worker_sequence_checker_);
750*d9f75844SAndroid Build Coastguard Worker return remote_ssrc();
751*d9f75844SAndroid Build Coastguard Worker }
752*d9f75844SAndroid Build Coastguard Worker
GetInfo() const753*d9f75844SAndroid Build Coastguard Worker absl::optional<Syncable::Info> VideoReceiveStream2::GetInfo() const {
754*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
755*d9f75844SAndroid Build Coastguard Worker absl::optional<Syncable::Info> info =
756*d9f75844SAndroid Build Coastguard Worker rtp_video_stream_receiver_.GetSyncInfo();
757*d9f75844SAndroid Build Coastguard Worker
758*d9f75844SAndroid Build Coastguard Worker if (!info)
759*d9f75844SAndroid Build Coastguard Worker return absl::nullopt;
760*d9f75844SAndroid Build Coastguard Worker
761*d9f75844SAndroid Build Coastguard Worker info->current_delay_ms = timing_->TargetVideoDelay().ms();
762*d9f75844SAndroid Build Coastguard Worker return info;
763*d9f75844SAndroid Build Coastguard Worker }
764*d9f75844SAndroid Build Coastguard Worker
GetPlayoutRtpTimestamp(uint32_t * rtp_timestamp,int64_t * time_ms) const765*d9f75844SAndroid Build Coastguard Worker bool VideoReceiveStream2::GetPlayoutRtpTimestamp(uint32_t* rtp_timestamp,
766*d9f75844SAndroid Build Coastguard Worker int64_t* time_ms) const {
767*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_NOTREACHED();
768*d9f75844SAndroid Build Coastguard Worker return false;
769*d9f75844SAndroid Build Coastguard Worker }
770*d9f75844SAndroid Build Coastguard Worker
SetEstimatedPlayoutNtpTimestampMs(int64_t ntp_timestamp_ms,int64_t time_ms)771*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::SetEstimatedPlayoutNtpTimestampMs(
772*d9f75844SAndroid Build Coastguard Worker int64_t ntp_timestamp_ms,
773*d9f75844SAndroid Build Coastguard Worker int64_t time_ms) {
774*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_NOTREACHED();
775*d9f75844SAndroid Build Coastguard Worker }
776*d9f75844SAndroid Build Coastguard Worker
SetMinimumPlayoutDelay(int delay_ms)777*d9f75844SAndroid Build Coastguard Worker bool VideoReceiveStream2::SetMinimumPlayoutDelay(int delay_ms) {
778*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&worker_sequence_checker_);
779*d9f75844SAndroid Build Coastguard Worker syncable_minimum_playout_delay_ = TimeDelta::Millis(delay_ms);
780*d9f75844SAndroid Build Coastguard Worker UpdatePlayoutDelays();
781*d9f75844SAndroid Build Coastguard Worker return true;
782*d9f75844SAndroid Build Coastguard Worker }
783*d9f75844SAndroid Build Coastguard Worker
OnEncodedFrame(std::unique_ptr<EncodedFrame> frame)784*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::OnEncodedFrame(std::unique_ptr<EncodedFrame> frame) {
785*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
786*d9f75844SAndroid Build Coastguard Worker Timestamp now = clock_->CurrentTime();
787*d9f75844SAndroid Build Coastguard Worker const bool keyframe_request_is_due =
788*d9f75844SAndroid Build Coastguard Worker !last_keyframe_request_ ||
789*d9f75844SAndroid Build Coastguard Worker now >= (*last_keyframe_request_ + max_wait_for_keyframe_);
790*d9f75844SAndroid Build Coastguard Worker const bool received_frame_is_keyframe =
791*d9f75844SAndroid Build Coastguard Worker frame->FrameType() == VideoFrameType::kVideoFrameKey;
792*d9f75844SAndroid Build Coastguard Worker
793*d9f75844SAndroid Build Coastguard Worker // Current OnPreDecode only cares about QP for VP8.
794*d9f75844SAndroid Build Coastguard Worker int qp = -1;
795*d9f75844SAndroid Build Coastguard Worker if (frame->CodecSpecific()->codecType == kVideoCodecVP8) {
796*d9f75844SAndroid Build Coastguard Worker if (!vp8::GetQp(frame->data(), frame->size(), &qp)) {
797*d9f75844SAndroid Build Coastguard Worker RTC_LOG(LS_WARNING) << "Failed to extract QP from VP8 video frame";
798*d9f75844SAndroid Build Coastguard Worker }
799*d9f75844SAndroid Build Coastguard Worker }
800*d9f75844SAndroid Build Coastguard Worker stats_proxy_.OnPreDecode(frame->CodecSpecific()->codecType, qp);
801*d9f75844SAndroid Build Coastguard Worker
802*d9f75844SAndroid Build Coastguard Worker decode_queue_.PostTask([this, now, keyframe_request_is_due,
803*d9f75844SAndroid Build Coastguard Worker received_frame_is_keyframe, frame = std::move(frame),
804*d9f75844SAndroid Build Coastguard Worker keyframe_required = keyframe_required_]() mutable {
805*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&decode_queue_);
806*d9f75844SAndroid Build Coastguard Worker if (decoder_stopped_)
807*d9f75844SAndroid Build Coastguard Worker return;
808*d9f75844SAndroid Build Coastguard Worker DecodeFrameResult result = HandleEncodedFrameOnDecodeQueue(
809*d9f75844SAndroid Build Coastguard Worker std::move(frame), keyframe_request_is_due, keyframe_required);
810*d9f75844SAndroid Build Coastguard Worker
811*d9f75844SAndroid Build Coastguard Worker // TODO(bugs.webrtc.org/11993): Make this PostTask to the network thread.
812*d9f75844SAndroid Build Coastguard Worker call_->worker_thread()->PostTask(
813*d9f75844SAndroid Build Coastguard Worker SafeTask(task_safety_.flag(),
814*d9f75844SAndroid Build Coastguard Worker [this, now, result = std::move(result),
815*d9f75844SAndroid Build Coastguard Worker received_frame_is_keyframe, keyframe_request_is_due]() {
816*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
817*d9f75844SAndroid Build Coastguard Worker keyframe_required_ = result.keyframe_required;
818*d9f75844SAndroid Build Coastguard Worker
819*d9f75844SAndroid Build Coastguard Worker if (result.decoded_frame_picture_id) {
820*d9f75844SAndroid Build Coastguard Worker rtp_video_stream_receiver_.FrameDecoded(
821*d9f75844SAndroid Build Coastguard Worker *result.decoded_frame_picture_id);
822*d9f75844SAndroid Build Coastguard Worker }
823*d9f75844SAndroid Build Coastguard Worker
824*d9f75844SAndroid Build Coastguard Worker HandleKeyFrameGeneration(received_frame_is_keyframe, now,
825*d9f75844SAndroid Build Coastguard Worker result.force_request_key_frame,
826*d9f75844SAndroid Build Coastguard Worker keyframe_request_is_due);
827*d9f75844SAndroid Build Coastguard Worker buffer_->StartNextDecode(keyframe_required_);
828*d9f75844SAndroid Build Coastguard Worker }));
829*d9f75844SAndroid Build Coastguard Worker });
830*d9f75844SAndroid Build Coastguard Worker }
831*d9f75844SAndroid Build Coastguard Worker
OnDecodableFrameTimeout(TimeDelta wait)832*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::OnDecodableFrameTimeout(TimeDelta wait) {
833*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
834*d9f75844SAndroid Build Coastguard Worker Timestamp now = clock_->CurrentTime();
835*d9f75844SAndroid Build Coastguard Worker
836*d9f75844SAndroid Build Coastguard Worker absl::optional<int64_t> last_packet_ms =
837*d9f75844SAndroid Build Coastguard Worker rtp_video_stream_receiver_.LastReceivedPacketMs();
838*d9f75844SAndroid Build Coastguard Worker
839*d9f75844SAndroid Build Coastguard Worker // To avoid spamming keyframe requests for a stream that is not active we
840*d9f75844SAndroid Build Coastguard Worker // check if we have received a packet within the last 5 seconds.
841*d9f75844SAndroid Build Coastguard Worker constexpr TimeDelta kInactiveDuration = TimeDelta::Seconds(5);
842*d9f75844SAndroid Build Coastguard Worker const bool stream_is_active =
843*d9f75844SAndroid Build Coastguard Worker last_packet_ms &&
844*d9f75844SAndroid Build Coastguard Worker now - Timestamp::Millis(*last_packet_ms) < kInactiveDuration;
845*d9f75844SAndroid Build Coastguard Worker if (!stream_is_active)
846*d9f75844SAndroid Build Coastguard Worker stats_proxy_.OnStreamInactive();
847*d9f75844SAndroid Build Coastguard Worker
848*d9f75844SAndroid Build Coastguard Worker if (stream_is_active && !IsReceivingKeyFrame(now) &&
849*d9f75844SAndroid Build Coastguard Worker (!config_.crypto_options.sframe.require_frame_encryption ||
850*d9f75844SAndroid Build Coastguard Worker rtp_video_stream_receiver_.IsDecryptable())) {
851*d9f75844SAndroid Build Coastguard Worker RTC_LOG(LS_WARNING) << "No decodable frame in " << wait
852*d9f75844SAndroid Build Coastguard Worker << ", requesting keyframe.";
853*d9f75844SAndroid Build Coastguard Worker RequestKeyFrame(now);
854*d9f75844SAndroid Build Coastguard Worker }
855*d9f75844SAndroid Build Coastguard Worker
856*d9f75844SAndroid Build Coastguard Worker buffer_->StartNextDecode(keyframe_required_);
857*d9f75844SAndroid Build Coastguard Worker }
858*d9f75844SAndroid Build Coastguard Worker
859*d9f75844SAndroid Build Coastguard Worker VideoReceiveStream2::DecodeFrameResult
HandleEncodedFrameOnDecodeQueue(std::unique_ptr<EncodedFrame> frame,bool keyframe_request_is_due,bool keyframe_required)860*d9f75844SAndroid Build Coastguard Worker VideoReceiveStream2::HandleEncodedFrameOnDecodeQueue(
861*d9f75844SAndroid Build Coastguard Worker std::unique_ptr<EncodedFrame> frame,
862*d9f75844SAndroid Build Coastguard Worker bool keyframe_request_is_due,
863*d9f75844SAndroid Build Coastguard Worker bool keyframe_required) {
864*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&decode_queue_);
865*d9f75844SAndroid Build Coastguard Worker
866*d9f75844SAndroid Build Coastguard Worker bool force_request_key_frame = false;
867*d9f75844SAndroid Build Coastguard Worker absl::optional<int64_t> decoded_frame_picture_id;
868*d9f75844SAndroid Build Coastguard Worker
869*d9f75844SAndroid Build Coastguard Worker if (!video_receiver_.IsExternalDecoderRegistered(frame->PayloadType())) {
870*d9f75844SAndroid Build Coastguard Worker // Look for the decoder with this payload type.
871*d9f75844SAndroid Build Coastguard Worker for (const Decoder& decoder : config_.decoders) {
872*d9f75844SAndroid Build Coastguard Worker if (decoder.payload_type == frame->PayloadType()) {
873*d9f75844SAndroid Build Coastguard Worker CreateAndRegisterExternalDecoder(decoder);
874*d9f75844SAndroid Build Coastguard Worker break;
875*d9f75844SAndroid Build Coastguard Worker }
876*d9f75844SAndroid Build Coastguard Worker }
877*d9f75844SAndroid Build Coastguard Worker }
878*d9f75844SAndroid Build Coastguard Worker
879*d9f75844SAndroid Build Coastguard Worker int64_t frame_id = frame->Id();
880*d9f75844SAndroid Build Coastguard Worker int decode_result = DecodeAndMaybeDispatchEncodedFrame(std::move(frame));
881*d9f75844SAndroid Build Coastguard Worker if (decode_result == WEBRTC_VIDEO_CODEC_OK ||
882*d9f75844SAndroid Build Coastguard Worker decode_result == WEBRTC_VIDEO_CODEC_OK_REQUEST_KEYFRAME) {
883*d9f75844SAndroid Build Coastguard Worker keyframe_required = false;
884*d9f75844SAndroid Build Coastguard Worker frame_decoded_ = true;
885*d9f75844SAndroid Build Coastguard Worker
886*d9f75844SAndroid Build Coastguard Worker decoded_frame_picture_id = frame_id;
887*d9f75844SAndroid Build Coastguard Worker
888*d9f75844SAndroid Build Coastguard Worker if (decode_result == WEBRTC_VIDEO_CODEC_OK_REQUEST_KEYFRAME)
889*d9f75844SAndroid Build Coastguard Worker force_request_key_frame = true;
890*d9f75844SAndroid Build Coastguard Worker } else if (!frame_decoded_ || !keyframe_required || keyframe_request_is_due) {
891*d9f75844SAndroid Build Coastguard Worker keyframe_required = true;
892*d9f75844SAndroid Build Coastguard Worker // TODO(philipel): Remove this keyframe request when downstream project
893*d9f75844SAndroid Build Coastguard Worker // has been fixed.
894*d9f75844SAndroid Build Coastguard Worker force_request_key_frame = true;
895*d9f75844SAndroid Build Coastguard Worker }
896*d9f75844SAndroid Build Coastguard Worker
897*d9f75844SAndroid Build Coastguard Worker return DecodeFrameResult{
898*d9f75844SAndroid Build Coastguard Worker .force_request_key_frame = force_request_key_frame,
899*d9f75844SAndroid Build Coastguard Worker .decoded_frame_picture_id = std::move(decoded_frame_picture_id),
900*d9f75844SAndroid Build Coastguard Worker .keyframe_required = keyframe_required,
901*d9f75844SAndroid Build Coastguard Worker };
902*d9f75844SAndroid Build Coastguard Worker }
903*d9f75844SAndroid Build Coastguard Worker
DecodeAndMaybeDispatchEncodedFrame(std::unique_ptr<EncodedFrame> frame)904*d9f75844SAndroid Build Coastguard Worker int VideoReceiveStream2::DecodeAndMaybeDispatchEncodedFrame(
905*d9f75844SAndroid Build Coastguard Worker std::unique_ptr<EncodedFrame> frame) {
906*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&decode_queue_);
907*d9f75844SAndroid Build Coastguard Worker
908*d9f75844SAndroid Build Coastguard Worker // If `buffered_encoded_frames_` grows out of control (=60 queued frames),
909*d9f75844SAndroid Build Coastguard Worker // maybe due to a stuck decoder, we just halt the process here and log the
910*d9f75844SAndroid Build Coastguard Worker // error.
911*d9f75844SAndroid Build Coastguard Worker const bool encoded_frame_output_enabled =
912*d9f75844SAndroid Build Coastguard Worker encoded_frame_buffer_function_ != nullptr &&
913*d9f75844SAndroid Build Coastguard Worker buffered_encoded_frames_.size() < kBufferedEncodedFramesMaxSize;
914*d9f75844SAndroid Build Coastguard Worker EncodedFrame* frame_ptr = frame.get();
915*d9f75844SAndroid Build Coastguard Worker if (encoded_frame_output_enabled) {
916*d9f75844SAndroid Build Coastguard Worker // If we receive a key frame with unset resolution, hold on dispatching the
917*d9f75844SAndroid Build Coastguard Worker // frame and following ones until we know a resolution of the stream.
918*d9f75844SAndroid Build Coastguard Worker // NOTE: The code below has a race where it can report the wrong
919*d9f75844SAndroid Build Coastguard Worker // resolution for keyframes after an initial keyframe of other resolution.
920*d9f75844SAndroid Build Coastguard Worker // However, the only known consumer of this information is the W3C
921*d9f75844SAndroid Build Coastguard Worker // MediaRecorder and it will only use the resolution in the first encoded
922*d9f75844SAndroid Build Coastguard Worker // keyframe from WebRTC, so misreporting is fine.
923*d9f75844SAndroid Build Coastguard Worker buffered_encoded_frames_.push_back(std::move(frame));
924*d9f75844SAndroid Build Coastguard Worker if (buffered_encoded_frames_.size() == kBufferedEncodedFramesMaxSize)
925*d9f75844SAndroid Build Coastguard Worker RTC_LOG(LS_ERROR) << "About to halt recordable encoded frame output due "
926*d9f75844SAndroid Build Coastguard Worker "to too many buffered frames.";
927*d9f75844SAndroid Build Coastguard Worker
928*d9f75844SAndroid Build Coastguard Worker webrtc::MutexLock lock(&pending_resolution_mutex_);
929*d9f75844SAndroid Build Coastguard Worker if (IsKeyFrameAndUnspecifiedResolution(*frame_ptr) &&
930*d9f75844SAndroid Build Coastguard Worker !pending_resolution_.has_value())
931*d9f75844SAndroid Build Coastguard Worker pending_resolution_.emplace();
932*d9f75844SAndroid Build Coastguard Worker }
933*d9f75844SAndroid Build Coastguard Worker
934*d9f75844SAndroid Build Coastguard Worker int decode_result = video_receiver_.Decode(frame_ptr);
935*d9f75844SAndroid Build Coastguard Worker if (encoded_frame_output_enabled) {
936*d9f75844SAndroid Build Coastguard Worker absl::optional<RecordableEncodedFrame::EncodedResolution>
937*d9f75844SAndroid Build Coastguard Worker pending_resolution;
938*d9f75844SAndroid Build Coastguard Worker {
939*d9f75844SAndroid Build Coastguard Worker // Fish out `pending_resolution_` to avoid taking the mutex on every lap
940*d9f75844SAndroid Build Coastguard Worker // or dispatching under the mutex in the flush loop.
941*d9f75844SAndroid Build Coastguard Worker webrtc::MutexLock lock(&pending_resolution_mutex_);
942*d9f75844SAndroid Build Coastguard Worker if (pending_resolution_.has_value())
943*d9f75844SAndroid Build Coastguard Worker pending_resolution = *pending_resolution_;
944*d9f75844SAndroid Build Coastguard Worker }
945*d9f75844SAndroid Build Coastguard Worker if (!pending_resolution.has_value() || !pending_resolution->empty()) {
946*d9f75844SAndroid Build Coastguard Worker // Flush the buffered frames.
947*d9f75844SAndroid Build Coastguard Worker for (const auto& frame : buffered_encoded_frames_) {
948*d9f75844SAndroid Build Coastguard Worker RecordableEncodedFrame::EncodedResolution resolution{
949*d9f75844SAndroid Build Coastguard Worker frame->EncodedImage()._encodedWidth,
950*d9f75844SAndroid Build Coastguard Worker frame->EncodedImage()._encodedHeight};
951*d9f75844SAndroid Build Coastguard Worker if (IsKeyFrameAndUnspecifiedResolution(*frame)) {
952*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK(!pending_resolution->empty());
953*d9f75844SAndroid Build Coastguard Worker resolution = *pending_resolution;
954*d9f75844SAndroid Build Coastguard Worker }
955*d9f75844SAndroid Build Coastguard Worker encoded_frame_buffer_function_(
956*d9f75844SAndroid Build Coastguard Worker WebRtcRecordableEncodedFrame(*frame, resolution));
957*d9f75844SAndroid Build Coastguard Worker }
958*d9f75844SAndroid Build Coastguard Worker buffered_encoded_frames_.clear();
959*d9f75844SAndroid Build Coastguard Worker }
960*d9f75844SAndroid Build Coastguard Worker }
961*d9f75844SAndroid Build Coastguard Worker return decode_result;
962*d9f75844SAndroid Build Coastguard Worker }
963*d9f75844SAndroid Build Coastguard Worker
HandleKeyFrameGeneration(bool received_frame_is_keyframe,Timestamp now,bool always_request_key_frame,bool keyframe_request_is_due)964*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::HandleKeyFrameGeneration(
965*d9f75844SAndroid Build Coastguard Worker bool received_frame_is_keyframe,
966*d9f75844SAndroid Build Coastguard Worker Timestamp now,
967*d9f75844SAndroid Build Coastguard Worker bool always_request_key_frame,
968*d9f75844SAndroid Build Coastguard Worker bool keyframe_request_is_due) {
969*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
970*d9f75844SAndroid Build Coastguard Worker bool request_key_frame = always_request_key_frame;
971*d9f75844SAndroid Build Coastguard Worker
972*d9f75844SAndroid Build Coastguard Worker // Repeat sending keyframe requests if we've requested a keyframe.
973*d9f75844SAndroid Build Coastguard Worker if (keyframe_generation_requested_) {
974*d9f75844SAndroid Build Coastguard Worker if (received_frame_is_keyframe) {
975*d9f75844SAndroid Build Coastguard Worker keyframe_generation_requested_ = false;
976*d9f75844SAndroid Build Coastguard Worker } else if (keyframe_request_is_due) {
977*d9f75844SAndroid Build Coastguard Worker if (!IsReceivingKeyFrame(now)) {
978*d9f75844SAndroid Build Coastguard Worker request_key_frame = true;
979*d9f75844SAndroid Build Coastguard Worker }
980*d9f75844SAndroid Build Coastguard Worker } else {
981*d9f75844SAndroid Build Coastguard Worker // It hasn't been long enough since the last keyframe request, do nothing.
982*d9f75844SAndroid Build Coastguard Worker }
983*d9f75844SAndroid Build Coastguard Worker }
984*d9f75844SAndroid Build Coastguard Worker
985*d9f75844SAndroid Build Coastguard Worker if (request_key_frame) {
986*d9f75844SAndroid Build Coastguard Worker // HandleKeyFrameGeneration is initiated from the decode thread -
987*d9f75844SAndroid Build Coastguard Worker // RequestKeyFrame() triggers a call back to the decode thread.
988*d9f75844SAndroid Build Coastguard Worker // Perhaps there's a way to avoid that.
989*d9f75844SAndroid Build Coastguard Worker RequestKeyFrame(now);
990*d9f75844SAndroid Build Coastguard Worker }
991*d9f75844SAndroid Build Coastguard Worker }
992*d9f75844SAndroid Build Coastguard Worker
IsReceivingKeyFrame(Timestamp now) const993*d9f75844SAndroid Build Coastguard Worker bool VideoReceiveStream2::IsReceivingKeyFrame(Timestamp now) const {
994*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
995*d9f75844SAndroid Build Coastguard Worker absl::optional<int64_t> last_keyframe_packet_ms =
996*d9f75844SAndroid Build Coastguard Worker rtp_video_stream_receiver_.LastReceivedKeyframePacketMs();
997*d9f75844SAndroid Build Coastguard Worker
998*d9f75844SAndroid Build Coastguard Worker // If we recently have been receiving packets belonging to a keyframe then
999*d9f75844SAndroid Build Coastguard Worker // we assume a keyframe is currently being received.
1000*d9f75844SAndroid Build Coastguard Worker bool receiving_keyframe = last_keyframe_packet_ms &&
1001*d9f75844SAndroid Build Coastguard Worker now - Timestamp::Millis(*last_keyframe_packet_ms) <
1002*d9f75844SAndroid Build Coastguard Worker max_wait_for_keyframe_;
1003*d9f75844SAndroid Build Coastguard Worker return receiving_keyframe;
1004*d9f75844SAndroid Build Coastguard Worker }
1005*d9f75844SAndroid Build Coastguard Worker
UpdatePlayoutDelays() const1006*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::UpdatePlayoutDelays() const {
1007*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&worker_sequence_checker_);
1008*d9f75844SAndroid Build Coastguard Worker const std::initializer_list<absl::optional<TimeDelta>> min_delays = {
1009*d9f75844SAndroid Build Coastguard Worker frame_minimum_playout_delay_, base_minimum_playout_delay_,
1010*d9f75844SAndroid Build Coastguard Worker syncable_minimum_playout_delay_};
1011*d9f75844SAndroid Build Coastguard Worker
1012*d9f75844SAndroid Build Coastguard Worker // Since nullopt < anything, this will return the largest of the minumum
1013*d9f75844SAndroid Build Coastguard Worker // delays, or nullopt if all are nullopt.
1014*d9f75844SAndroid Build Coastguard Worker absl::optional<TimeDelta> minimum_delay = std::max(min_delays);
1015*d9f75844SAndroid Build Coastguard Worker if (minimum_delay) {
1016*d9f75844SAndroid Build Coastguard Worker auto num_playout_delays_set =
1017*d9f75844SAndroid Build Coastguard Worker absl::c_count_if(min_delays, [](auto opt) { return opt.has_value(); });
1018*d9f75844SAndroid Build Coastguard Worker if (num_playout_delays_set > 1 &&
1019*d9f75844SAndroid Build Coastguard Worker timing_->min_playout_delay() != minimum_delay) {
1020*d9f75844SAndroid Build Coastguard Worker RTC_LOG(LS_WARNING)
1021*d9f75844SAndroid Build Coastguard Worker << "Multiple playout delays set. Actual delay value set to "
1022*d9f75844SAndroid Build Coastguard Worker << *minimum_delay << " frame min delay="
1023*d9f75844SAndroid Build Coastguard Worker << OptionalDelayToLogString(frame_maximum_playout_delay_)
1024*d9f75844SAndroid Build Coastguard Worker << " base min delay="
1025*d9f75844SAndroid Build Coastguard Worker << OptionalDelayToLogString(base_minimum_playout_delay_)
1026*d9f75844SAndroid Build Coastguard Worker << " sync min delay="
1027*d9f75844SAndroid Build Coastguard Worker << OptionalDelayToLogString(syncable_minimum_playout_delay_);
1028*d9f75844SAndroid Build Coastguard Worker }
1029*d9f75844SAndroid Build Coastguard Worker timing_->set_min_playout_delay(*minimum_delay);
1030*d9f75844SAndroid Build Coastguard Worker if (frame_minimum_playout_delay_ == TimeDelta::Zero() &&
1031*d9f75844SAndroid Build Coastguard Worker frame_maximum_playout_delay_ > TimeDelta::Zero()) {
1032*d9f75844SAndroid Build Coastguard Worker // TODO(kron): Estimate frame rate from video stream.
1033*d9f75844SAndroid Build Coastguard Worker constexpr Frequency kFrameRate = Frequency::Hertz(60);
1034*d9f75844SAndroid Build Coastguard Worker // Convert playout delay in ms to number of frames.
1035*d9f75844SAndroid Build Coastguard Worker int max_composition_delay_in_frames =
1036*d9f75844SAndroid Build Coastguard Worker std::lrint(*frame_maximum_playout_delay_ * kFrameRate);
1037*d9f75844SAndroid Build Coastguard Worker // Subtract frames in buffer.
1038*d9f75844SAndroid Build Coastguard Worker max_composition_delay_in_frames =
1039*d9f75844SAndroid Build Coastguard Worker std::max(max_composition_delay_in_frames - buffer_->Size(), 0);
1040*d9f75844SAndroid Build Coastguard Worker timing_->SetMaxCompositionDelayInFrames(max_composition_delay_in_frames);
1041*d9f75844SAndroid Build Coastguard Worker }
1042*d9f75844SAndroid Build Coastguard Worker }
1043*d9f75844SAndroid Build Coastguard Worker
1044*d9f75844SAndroid Build Coastguard Worker if (frame_maximum_playout_delay_) {
1045*d9f75844SAndroid Build Coastguard Worker timing_->set_max_playout_delay(*frame_maximum_playout_delay_);
1046*d9f75844SAndroid Build Coastguard Worker }
1047*d9f75844SAndroid Build Coastguard Worker }
1048*d9f75844SAndroid Build Coastguard Worker
GetSources() const1049*d9f75844SAndroid Build Coastguard Worker std::vector<webrtc::RtpSource> VideoReceiveStream2::GetSources() const {
1050*d9f75844SAndroid Build Coastguard Worker return source_tracker_.GetSources();
1051*d9f75844SAndroid Build Coastguard Worker }
1052*d9f75844SAndroid Build Coastguard Worker
1053*d9f75844SAndroid Build Coastguard Worker VideoReceiveStream2::RecordingState
SetAndGetRecordingState(RecordingState state,bool generate_key_frame)1054*d9f75844SAndroid Build Coastguard Worker VideoReceiveStream2::SetAndGetRecordingState(RecordingState state,
1055*d9f75844SAndroid Build Coastguard Worker bool generate_key_frame) {
1056*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&worker_sequence_checker_);
1057*d9f75844SAndroid Build Coastguard Worker rtc::Event event;
1058*d9f75844SAndroid Build Coastguard Worker
1059*d9f75844SAndroid Build Coastguard Worker // Save old state, set the new state.
1060*d9f75844SAndroid Build Coastguard Worker RecordingState old_state;
1061*d9f75844SAndroid Build Coastguard Worker
1062*d9f75844SAndroid Build Coastguard Worker absl::optional<Timestamp> last_keyframe_request;
1063*d9f75844SAndroid Build Coastguard Worker {
1064*d9f75844SAndroid Build Coastguard Worker // TODO(bugs.webrtc.org/11993): Post this to the network thread.
1065*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
1066*d9f75844SAndroid Build Coastguard Worker last_keyframe_request = last_keyframe_request_;
1067*d9f75844SAndroid Build Coastguard Worker last_keyframe_request_ =
1068*d9f75844SAndroid Build Coastguard Worker generate_key_frame
1069*d9f75844SAndroid Build Coastguard Worker ? clock_->CurrentTime()
1070*d9f75844SAndroid Build Coastguard Worker : Timestamp::Millis(state.last_keyframe_request_ms.value_or(0));
1071*d9f75844SAndroid Build Coastguard Worker }
1072*d9f75844SAndroid Build Coastguard Worker
1073*d9f75844SAndroid Build Coastguard Worker decode_queue_.PostTask(
1074*d9f75844SAndroid Build Coastguard Worker [this, &event, &old_state, callback = std::move(state.callback),
1075*d9f75844SAndroid Build Coastguard Worker last_keyframe_request = std::move(last_keyframe_request)] {
1076*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&decode_queue_);
1077*d9f75844SAndroid Build Coastguard Worker old_state.callback = std::move(encoded_frame_buffer_function_);
1078*d9f75844SAndroid Build Coastguard Worker encoded_frame_buffer_function_ = std::move(callback);
1079*d9f75844SAndroid Build Coastguard Worker
1080*d9f75844SAndroid Build Coastguard Worker old_state.last_keyframe_request_ms =
1081*d9f75844SAndroid Build Coastguard Worker last_keyframe_request.value_or(Timestamp::Zero()).ms();
1082*d9f75844SAndroid Build Coastguard Worker
1083*d9f75844SAndroid Build Coastguard Worker event.Set();
1084*d9f75844SAndroid Build Coastguard Worker });
1085*d9f75844SAndroid Build Coastguard Worker
1086*d9f75844SAndroid Build Coastguard Worker if (generate_key_frame) {
1087*d9f75844SAndroid Build Coastguard Worker rtp_video_stream_receiver_.RequestKeyFrame();
1088*d9f75844SAndroid Build Coastguard Worker {
1089*d9f75844SAndroid Build Coastguard Worker // TODO(bugs.webrtc.org/11993): Post this to the network thread.
1090*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
1091*d9f75844SAndroid Build Coastguard Worker keyframe_generation_requested_ = true;
1092*d9f75844SAndroid Build Coastguard Worker }
1093*d9f75844SAndroid Build Coastguard Worker }
1094*d9f75844SAndroid Build Coastguard Worker
1095*d9f75844SAndroid Build Coastguard Worker event.Wait(rtc::Event::kForever);
1096*d9f75844SAndroid Build Coastguard Worker return old_state;
1097*d9f75844SAndroid Build Coastguard Worker }
1098*d9f75844SAndroid Build Coastguard Worker
GenerateKeyFrame()1099*d9f75844SAndroid Build Coastguard Worker void VideoReceiveStream2::GenerateKeyFrame() {
1100*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK_RUN_ON(&packet_sequence_checker_);
1101*d9f75844SAndroid Build Coastguard Worker RequestKeyFrame(clock_->CurrentTime());
1102*d9f75844SAndroid Build Coastguard Worker keyframe_generation_requested_ = true;
1103*d9f75844SAndroid Build Coastguard Worker }
1104*d9f75844SAndroid Build Coastguard Worker
1105*d9f75844SAndroid Build Coastguard Worker } // namespace internal
1106*d9f75844SAndroid Build Coastguard Worker } // namespace webrtc
1107