blob: 36f53a70a442a340ccac3aa3e5612e0e58c88895 [file] [log] [blame]
/*
* Copyright (c) 2013 The WebRTC project authors. All Rights Reserved.
*
* Use of this source code is governed by a BSD-style license
* that can be found in the LICENSE file in the root of the source
* tree. An additional intellectual property rights grant can be found
* in the file PATENTS. All contributing project authors may
* be found in the AUTHORS file in the root of the source tree.
*/
#include "modules/video_coding/video_receiver2.h"
#include <stddef.h>
#include <cstdint>
#include <utility>
#include <vector>
#include "absl/algorithm/container.h"
#include "api/video_codecs/video_codec.h"
#include "api/video_codecs/video_decoder.h"
#include "modules/video_coding/decoder_database.h"
#include "modules/video_coding/encoded_frame.h"
#include "modules/video_coding/generic_decoder.h"
#include "modules/video_coding/include/video_coding_defines.h"
#include "modules/video_coding/timing/timing.h"
#include "rtc_base/checks.h"
#include "rtc_base/trace_event.h"
#include "system_wrappers/include/clock.h"
namespace webrtc {
VideoReceiver2::VideoReceiver2(Clock* clock,
VCMTiming* timing,
const FieldTrialsView& field_trials)
: clock_(clock),
decoded_frame_callback_(timing, clock_, field_trials),
codec_database_() {
decoder_sequence_checker_.Detach();
}
VideoReceiver2::~VideoReceiver2() {
RTC_DCHECK_RUN_ON(&construction_sequence_checker_);
}
// Register a receive callback. Will be called whenever there is a new frame
// ready for rendering.
int32_t VideoReceiver2::RegisterReceiveCallback(
VCMReceiveCallback* receive_callback) {
RTC_DCHECK_RUN_ON(&construction_sequence_checker_);
// This value is set before the decoder thread starts and unset after
// the decoder thread has been stopped.
decoded_frame_callback_.SetUserReceiveCallback(receive_callback);
return VCM_OK;
}
void VideoReceiver2::RegisterExternalDecoder(
std::unique_ptr<VideoDecoder> decoder,
uint8_t payload_type) {
RTC_DCHECK_RUN_ON(&decoder_sequence_checker_);
RTC_DCHECK(decoded_frame_callback_.UserReceiveCallback());
if (decoder) {
RTC_DCHECK(!codec_database_.IsExternalDecoderRegistered(payload_type));
codec_database_.RegisterExternalDecoder(payload_type, decoder.get());
video_decoders_.push_back(std::move(decoder));
} else {
VideoDecoder* registered =
codec_database_.DeregisterExternalDecoder(payload_type);
if (registered) {
video_decoders_.erase(absl::c_find_if(
video_decoders_,
[registered](const auto& d) { return d.get() == registered; }));
}
}
}
bool VideoReceiver2::IsExternalDecoderRegistered(uint8_t payload_type) const {
RTC_DCHECK_RUN_ON(&decoder_sequence_checker_);
return codec_database_.IsExternalDecoderRegistered(payload_type);
}
// Must be called from inside the receive side critical section.
int32_t VideoReceiver2::Decode(const VCMEncodedFrame* frame) {
RTC_DCHECK_RUN_ON(&decoder_sequence_checker_);
TRACE_EVENT0("webrtc", "VideoReceiver2::Decode");
// Change decoder if payload type has changed.
VCMGenericDecoder* decoder =
codec_database_.GetDecoder(*frame, &decoded_frame_callback_);
if (decoder == nullptr) {
return VCM_NO_CODEC_REGISTERED;
}
return decoder->Decode(*frame, clock_->CurrentTime());
}
// Register possible receive codecs, can be called multiple times.
// Called before decoder thread is started.
void VideoReceiver2::RegisterReceiveCodec(
uint8_t payload_type,
const VideoDecoder::Settings& settings) {
RTC_DCHECK_RUN_ON(&construction_sequence_checker_);
codec_database_.RegisterReceiveCodec(payload_type, settings);
}
} // namespace webrtc