mirror of
https://github.com/mollyim/webrtc.git
synced 2025-05-12 21:30:45 +01:00

Environment provides non-null interface for FieldTrialsView and thus VideoCodingModule no longer need to rely on FieldTrialBasedConfig class to provide field_trials when not passed at construction. Bug: webrtc:10335 Change-Id: Iedfb29e8b29056618a85f2e7a1528da29e3be5c1 Reviewed-on: https://webrtc-review.googlesource.com/c/src/+/347701 Auto-Submit: Danil Chapovalov <danilchap@webrtc.org> Reviewed-by: Philip Eliasson <philipel@webrtc.org> Commit-Queue: Philip Eliasson <philipel@webrtc.org> Cr-Commit-Position: refs/heads/main@{#42163}
247 lines
8.3 KiB
C++
247 lines
8.3 KiB
C++
/*
|
|
* Copyright (c) 2012 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_coding_impl.h"
|
|
|
|
#include <algorithm>
|
|
#include <memory>
|
|
|
|
#include "api/field_trials_view.h"
|
|
#include "api/sequence_checker.h"
|
|
#include "api/video/encoded_image.h"
|
|
#include "modules/video_coding/include/video_codec_interface.h"
|
|
#include "modules/video_coding/timing/timing.h"
|
|
#include "rtc_base/logging.h"
|
|
#include "system_wrappers/include/clock.h"
|
|
|
|
namespace webrtc {
|
|
namespace vcm {
|
|
|
|
int64_t VCMProcessTimer::Period() const {
|
|
return _periodMs;
|
|
}
|
|
|
|
int64_t VCMProcessTimer::TimeUntilProcess() const {
|
|
const int64_t time_since_process = _clock->TimeInMilliseconds() - _latestMs;
|
|
const int64_t time_until_process = _periodMs - time_since_process;
|
|
return std::max<int64_t>(time_until_process, 0);
|
|
}
|
|
|
|
void VCMProcessTimer::Processed() {
|
|
_latestMs = _clock->TimeInMilliseconds();
|
|
}
|
|
|
|
DEPRECATED_VCMDecoderDataBase::DEPRECATED_VCMDecoderDataBase() {
|
|
decoder_sequence_checker_.Detach();
|
|
}
|
|
|
|
VideoDecoder* DEPRECATED_VCMDecoderDataBase::DeregisterExternalDecoder(
|
|
uint8_t payload_type) {
|
|
RTC_DCHECK_RUN_ON(&decoder_sequence_checker_);
|
|
auto it = decoders_.find(payload_type);
|
|
if (it == decoders_.end()) {
|
|
return nullptr;
|
|
}
|
|
|
|
// We can't use payload_type to check if the decoder is currently in use,
|
|
// because payload type may be out of date (e.g. before we decode the first
|
|
// frame after RegisterReceiveCodec).
|
|
if (current_decoder_ && current_decoder_->IsSameDecoder(it->second)) {
|
|
// Release it if it was registered and in use.
|
|
current_decoder_ = absl::nullopt;
|
|
}
|
|
VideoDecoder* ret = it->second;
|
|
decoders_.erase(it);
|
|
return ret;
|
|
}
|
|
|
|
// Add the external decoder object to the list of external decoders.
|
|
// Won't be registered as a receive codec until RegisterReceiveCodec is called.
|
|
void DEPRECATED_VCMDecoderDataBase::RegisterExternalDecoder(
|
|
uint8_t payload_type,
|
|
VideoDecoder* external_decoder) {
|
|
RTC_DCHECK_RUN_ON(&decoder_sequence_checker_);
|
|
// If payload value already exists, erase old and insert new.
|
|
DeregisterExternalDecoder(payload_type);
|
|
decoders_[payload_type] = external_decoder;
|
|
}
|
|
|
|
bool DEPRECATED_VCMDecoderDataBase::IsExternalDecoderRegistered(
|
|
uint8_t payload_type) const {
|
|
RTC_DCHECK_RUN_ON(&decoder_sequence_checker_);
|
|
return payload_type == current_payload_type_ ||
|
|
decoders_.find(payload_type) != decoders_.end();
|
|
}
|
|
|
|
void DEPRECATED_VCMDecoderDataBase::RegisterReceiveCodec(
|
|
uint8_t payload_type,
|
|
const VideoDecoder::Settings& settings) {
|
|
// If payload value already exists, erase old and insert new.
|
|
if (payload_type == current_payload_type_) {
|
|
current_payload_type_ = absl::nullopt;
|
|
}
|
|
decoder_settings_[payload_type] = settings;
|
|
}
|
|
|
|
bool DEPRECATED_VCMDecoderDataBase::DeregisterReceiveCodec(
|
|
uint8_t payload_type) {
|
|
if (decoder_settings_.erase(payload_type) == 0) {
|
|
return false;
|
|
}
|
|
if (payload_type == current_payload_type_) {
|
|
// This codec is currently in use.
|
|
current_payload_type_ = absl::nullopt;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
VCMGenericDecoder* DEPRECATED_VCMDecoderDataBase::GetDecoder(
|
|
const VCMEncodedFrame& frame,
|
|
VCMDecodedFrameCallback* decoded_frame_callback) {
|
|
RTC_DCHECK_RUN_ON(&decoder_sequence_checker_);
|
|
RTC_DCHECK(decoded_frame_callback->UserReceiveCallback());
|
|
uint8_t payload_type = frame.PayloadType();
|
|
if (payload_type == current_payload_type_ || payload_type == 0) {
|
|
return current_decoder_.has_value() ? &*current_decoder_ : nullptr;
|
|
}
|
|
// If decoder exists - delete.
|
|
if (current_decoder_.has_value()) {
|
|
current_decoder_ = absl::nullopt;
|
|
current_payload_type_ = absl::nullopt;
|
|
}
|
|
|
|
CreateAndInitDecoder(frame);
|
|
if (current_decoder_ == absl::nullopt) {
|
|
return nullptr;
|
|
}
|
|
|
|
VCMReceiveCallback* callback = decoded_frame_callback->UserReceiveCallback();
|
|
callback->OnIncomingPayloadType(payload_type);
|
|
if (current_decoder_->RegisterDecodeCompleteCallback(decoded_frame_callback) <
|
|
0) {
|
|
current_decoder_ = absl::nullopt;
|
|
return nullptr;
|
|
}
|
|
|
|
current_payload_type_ = payload_type;
|
|
return &*current_decoder_;
|
|
}
|
|
|
|
void DEPRECATED_VCMDecoderDataBase::CreateAndInitDecoder(
|
|
const VCMEncodedFrame& frame) {
|
|
uint8_t payload_type = frame.PayloadType();
|
|
RTC_LOG(LS_INFO) << "Initializing decoder with payload type '"
|
|
<< int{payload_type} << "'.";
|
|
auto decoder_item = decoder_settings_.find(payload_type);
|
|
if (decoder_item == decoder_settings_.end()) {
|
|
RTC_LOG(LS_ERROR) << "Can't find a decoder associated with payload type: "
|
|
<< int{payload_type};
|
|
return;
|
|
}
|
|
auto external_dec_item = decoders_.find(payload_type);
|
|
if (external_dec_item == decoders_.end()) {
|
|
RTC_LOG(LS_ERROR) << "No decoder of this type exists.";
|
|
return;
|
|
}
|
|
current_decoder_.emplace(external_dec_item->second);
|
|
|
|
// Copy over input resolutions to prevent codec reinitialization due to
|
|
// the first frame being of a different resolution than the database values.
|
|
// This is best effort, since there's no guarantee that width/height have been
|
|
// parsed yet (and may be zero).
|
|
RenderResolution frame_resolution(frame.EncodedImage()._encodedWidth,
|
|
frame.EncodedImage()._encodedHeight);
|
|
if (frame_resolution.Valid()) {
|
|
decoder_item->second.set_max_render_resolution(frame_resolution);
|
|
}
|
|
if (!current_decoder_->Configure(decoder_item->second)) {
|
|
current_decoder_ = absl::nullopt;
|
|
RTC_LOG(LS_ERROR) << "Failed to initialize decoder.";
|
|
}
|
|
}
|
|
|
|
} // namespace vcm
|
|
|
|
namespace {
|
|
|
|
class VideoCodingModuleImpl : public VideoCodingModule {
|
|
public:
|
|
explicit VideoCodingModuleImpl(const Environment& env)
|
|
: env_(env),
|
|
timing_(&env_.clock(), env_.field_trials()),
|
|
receiver_(&env_.clock(), &timing_, env_.field_trials()) {}
|
|
|
|
~VideoCodingModuleImpl() override = default;
|
|
|
|
void Process() override { receiver_.Process(); }
|
|
|
|
void RegisterReceiveCodec(
|
|
uint8_t payload_type,
|
|
const VideoDecoder::Settings& decoder_settings) override {
|
|
receiver_.RegisterReceiveCodec(payload_type, decoder_settings);
|
|
}
|
|
|
|
void RegisterExternalDecoder(VideoDecoder* externalDecoder,
|
|
uint8_t payloadType) override {
|
|
receiver_.RegisterExternalDecoder(externalDecoder, payloadType);
|
|
}
|
|
|
|
int32_t RegisterReceiveCallback(
|
|
VCMReceiveCallback* receiveCallback) override {
|
|
RTC_DCHECK(construction_thread_.IsCurrent());
|
|
return receiver_.RegisterReceiveCallback(receiveCallback);
|
|
}
|
|
|
|
int32_t RegisterFrameTypeCallback(
|
|
VCMFrameTypeCallback* frameTypeCallback) override {
|
|
return receiver_.RegisterFrameTypeCallback(frameTypeCallback);
|
|
}
|
|
|
|
int32_t RegisterPacketRequestCallback(
|
|
VCMPacketRequestCallback* callback) override {
|
|
RTC_DCHECK(construction_thread_.IsCurrent());
|
|
return receiver_.RegisterPacketRequestCallback(callback);
|
|
}
|
|
|
|
int32_t Decode(uint16_t maxWaitTimeMs) override {
|
|
return receiver_.Decode(maxWaitTimeMs);
|
|
}
|
|
|
|
int32_t IncomingPacket(const uint8_t* incomingPayload,
|
|
size_t payloadLength,
|
|
const RTPHeader& rtp_header,
|
|
const RTPVideoHeader& video_header) override {
|
|
return receiver_.IncomingPacket(incomingPayload, payloadLength, rtp_header,
|
|
video_header);
|
|
}
|
|
|
|
void SetNackSettings(size_t max_nack_list_size,
|
|
int max_packet_age_to_nack,
|
|
int max_incomplete_time_ms) override {
|
|
return receiver_.SetNackSettings(max_nack_list_size, max_packet_age_to_nack,
|
|
max_incomplete_time_ms);
|
|
}
|
|
|
|
private:
|
|
const Environment env_;
|
|
SequenceChecker construction_thread_;
|
|
VCMTiming timing_;
|
|
vcm::VideoReceiver receiver_;
|
|
};
|
|
} // namespace
|
|
|
|
// DEPRECATED. Create method for current interface, will be removed when the
|
|
// new jitter buffer is in place.
|
|
std::unique_ptr<VideoCodingModule> VideoCodingModule::CreateDeprecated(
|
|
const Environment& env) {
|
|
return std::make_unique<VideoCodingModuleImpl>(env);
|
|
}
|
|
|
|
} // namespace webrtc
|