blob: 467e99c1d2d2b11a520bf521f94519ec3b64e84d [file] [log] [blame]
Niels Möllerf9063782018-02-20 16:09:48 +01001/*
2 * Copyright (c) 2018 The WebRTC project authors. All Rights Reserved.
3 *
4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 */
10
11#include "modules/video_coding/decoder_database.h"
Jonas Olssona4d87372019-07-05 19:08:33 +020012
Niels Möllerf9063782018-02-20 16:09:48 +010013#include "rtc_base/checks.h"
14#include "rtc_base/logging.h"
15
16namespace webrtc {
17
Niels Möllerf9063782018-02-20 16:09:48 +010018bool VCMDecoderDataBase::DeregisterExternalDecoder(uint8_t payload_type) {
Danil Chapovalov7b78a312021-08-06 12:30:02 +020019 auto it = decoders_.find(payload_type);
20 if (it == decoders_.end()) {
Åsa Persson6cb74fd2018-06-01 15:13:42 +020021 // Not found.
Niels Möllerf9063782018-02-20 16:09:48 +010022 return false;
23 }
24 // We can't use payload_type to check if the decoder is currently in use,
25 // because payload type may be out of date (e.g. before we decode the first
Åsa Persson6cb74fd2018-06-01 15:13:42 +020026 // frame after RegisterReceiveCodec).
Danil Chapovalov7b78a312021-08-06 12:30:02 +020027 if (current_decoder_ && current_decoder_->IsSameDecoder(it->second)) {
Niels Möllerf9063782018-02-20 16:09:48 +010028 // Release it if it was registered and in use.
Danil Chapovalov7b78a312021-08-06 12:30:02 +020029 current_decoder_ = absl::nullopt;
Niels Möllerf9063782018-02-20 16:09:48 +010030 }
Danil Chapovalov7b78a312021-08-06 12:30:02 +020031 decoders_.erase(it);
Niels Möllerf9063782018-02-20 16:09:48 +010032 return true;
33}
34
Åsa Persson6cb74fd2018-06-01 15:13:42 +020035// Add the external decoder object to the list of external decoders.
Niels Möllerf9063782018-02-20 16:09:48 +010036// Won't be registered as a receive codec until RegisterReceiveCodec is called.
Danil Chapovalov7b78a312021-08-06 12:30:02 +020037void VCMDecoderDataBase::RegisterExternalDecoder(
38 uint8_t payload_type,
39 VideoDecoder* external_decoder) {
Åsa Persson6cb74fd2018-06-01 15:13:42 +020040 // If payload value already exists, erase old and insert new.
Niels Möllerf9063782018-02-20 16:09:48 +010041 DeregisterExternalDecoder(payload_type);
Danil Chapovalov7b78a312021-08-06 12:30:02 +020042 decoders_[payload_type] = external_decoder;
Niels Möllerf9063782018-02-20 16:09:48 +010043}
44
Johannes Kron16359f62021-02-18 23:37:22 +010045bool VCMDecoderDataBase::IsExternalDecoderRegistered(
46 uint8_t payload_type) const {
47 return payload_type == current_payload_type_ ||
Danil Chapovalov7b78a312021-08-06 12:30:02 +020048 decoders_.find(payload_type) != decoders_.end();
Johannes Kron16359f62021-02-18 23:37:22 +010049}
50
Danil Chapovalov355b8d22021-08-13 16:50:37 +020051bool VCMDecoderDataBase::RegisterReceiveCodec(
52 uint8_t payload_type,
53 const VideoDecoder::Settings& settings) {
54 if (settings.number_of_cores() < 0) {
Niels Möllerf9063782018-02-20 16:09:48 +010055 return false;
56 }
Åsa Persson6cb74fd2018-06-01 15:13:42 +020057 // If payload value already exists, erase old and insert new.
Danil Chapovalov7b78a312021-08-06 12:30:02 +020058 if (payload_type == current_payload_type_) {
59 current_payload_type_ = absl::nullopt;
60 }
Danil Chapovalov355b8d22021-08-13 16:50:37 +020061 decoder_settings_[payload_type] = settings;
Niels Möllerf9063782018-02-20 16:09:48 +010062 return true;
63}
64
65bool VCMDecoderDataBase::DeregisterReceiveCodec(uint8_t payload_type) {
Danil Chapovalov7b78a312021-08-06 12:30:02 +020066 if (decoder_settings_.erase(payload_type) == 0) {
Niels Möllerf9063782018-02-20 16:09:48 +010067 return false;
68 }
Niels Möller5401bad2020-08-11 12:17:42 +020069 if (payload_type == current_payload_type_) {
Niels Möllerf9063782018-02-20 16:09:48 +010070 // This codec is currently in use.
Danil Chapovalov7b78a312021-08-06 12:30:02 +020071 current_payload_type_ = absl::nullopt;
Niels Möllerf9063782018-02-20 16:09:48 +010072 }
73 return true;
74}
75
76VCMGenericDecoder* VCMDecoderDataBase::GetDecoder(
77 const VCMEncodedFrame& frame,
78 VCMDecodedFrameCallback* decoded_frame_callback) {
79 RTC_DCHECK(decoded_frame_callback->UserReceiveCallback());
80 uint8_t payload_type = frame.PayloadType();
Niels Möller5401bad2020-08-11 12:17:42 +020081 if (payload_type == current_payload_type_ || payload_type == 0) {
Danil Chapovalov7b78a312021-08-06 12:30:02 +020082 return current_decoder_.has_value() ? &*current_decoder_ : nullptr;
Niels Möllerf9063782018-02-20 16:09:48 +010083 }
Åsa Persson6cb74fd2018-06-01 15:13:42 +020084 // If decoder exists - delete.
Danil Chapovalov7b78a312021-08-06 12:30:02 +020085 if (current_decoder_.has_value()) {
86 current_decoder_ = absl::nullopt;
87 current_payload_type_ = absl::nullopt;
Niels Möllerf9063782018-02-20 16:09:48 +010088 }
Danil Chapovalov7b78a312021-08-06 12:30:02 +020089
90 CreateAndInitDecoder(frame);
91 if (current_decoder_ == absl::nullopt) {
Niels Möllerf9063782018-02-20 16:09:48 +010092 return nullptr;
93 }
Danil Chapovalov7b78a312021-08-06 12:30:02 +020094
Niels Möllerf9063782018-02-20 16:09:48 +010095 VCMReceiveCallback* callback = decoded_frame_callback->UserReceiveCallback();
Danil Chapovalov7b78a312021-08-06 12:30:02 +020096 callback->OnIncomingPayloadType(payload_type);
97 if (current_decoder_->RegisterDecodeCompleteCallback(decoded_frame_callback) <
Niels Möllerf9063782018-02-20 16:09:48 +010098 0) {
Danil Chapovalov7b78a312021-08-06 12:30:02 +020099 current_decoder_ = absl::nullopt;
Niels Möllerf9063782018-02-20 16:09:48 +0100100 return nullptr;
101 }
Danil Chapovalov7b78a312021-08-06 12:30:02 +0200102
103 current_payload_type_ = payload_type;
104 return &*current_decoder_;
Niels Möllerf9063782018-02-20 16:09:48 +0100105}
106
Danil Chapovalov7b78a312021-08-06 12:30:02 +0200107void VCMDecoderDataBase::CreateAndInitDecoder(const VCMEncodedFrame& frame) {
Niels Möllerf9063782018-02-20 16:09:48 +0100108 uint8_t payload_type = frame.PayloadType();
109 RTC_LOG(LS_INFO) << "Initializing decoder with payload type '"
Danil Chapovalov7b78a312021-08-06 12:30:02 +0200110 << int{payload_type} << "'.";
111 auto decoder_item = decoder_settings_.find(payload_type);
112 if (decoder_item == decoder_settings_.end()) {
Niels Möllerf9063782018-02-20 16:09:48 +0100113 RTC_LOG(LS_ERROR) << "Can't find a decoder associated with payload type: "
Danil Chapovalov7b78a312021-08-06 12:30:02 +0200114 << int{payload_type};
115 return;
Niels Möllerf9063782018-02-20 16:09:48 +0100116 }
Danil Chapovalov7b78a312021-08-06 12:30:02 +0200117 auto external_dec_item = decoders_.find(payload_type);
118 if (external_dec_item == decoders_.end()) {
Niels Möllerf9063782018-02-20 16:09:48 +0100119 RTC_LOG(LS_ERROR) << "No decoder of this type exists.";
Danil Chapovalov7b78a312021-08-06 12:30:02 +0200120 return;
Niels Möllerf9063782018-02-20 16:09:48 +0100121 }
Danil Chapovalov7b78a312021-08-06 12:30:02 +0200122 current_decoder_.emplace(external_dec_item->second);
Niels Möllerf9063782018-02-20 16:09:48 +0100123
124 // Copy over input resolutions to prevent codec reinitialization due to
125 // the first frame being of a different resolution than the database values.
126 // This is best effort, since there's no guarantee that width/height have been
127 // parsed yet (and may be zero).
Danil Chapovalov355b8d22021-08-13 16:50:37 +0200128 RenderResolution frame_resolution(frame.EncodedImage()._encodedWidth,
129 frame.EncodedImage()._encodedHeight);
130 if (frame_resolution.Valid()) {
131 decoder_item->second.set_max_render_resolution(frame_resolution);
Niels Möllerf9063782018-02-20 16:09:48 +0100132 }
Danil Chapovalov355b8d22021-08-13 16:50:37 +0200133 if (!current_decoder_->Configure(decoder_item->second)) {
Danil Chapovalov7b78a312021-08-06 12:30:02 +0200134 current_decoder_ = absl::nullopt;
Danil Chapovalov355b8d22021-08-13 16:50:37 +0200135 RTC_LOG(LS_ERROR) << "Failed to initialize decoder.";
Niels Möllerf9063782018-02-20 16:09:48 +0100136 }
Niels Möllerf9063782018-02-20 16:09:48 +0100137}
138
139} // namespace webrtc