/* * 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 "webrtc/modules/audio_coding/codecs/audio_decoder.h" #include #include #include "webrtc/base/array_view.h" #include "webrtc/base/checks.h" #include "webrtc/base/sanitizer.h" #include "webrtc/base/trace_event.h" namespace webrtc { namespace { class LegacyFrame final : public AudioDecoder::EncodedAudioFrame { public: LegacyFrame(AudioDecoder* decoder, rtc::Buffer&& payload, bool is_primary_payload) : decoder_(decoder), payload_(std::move(payload)), is_primary_payload_(is_primary_payload) {} size_t Duration() const override { int ret; if (is_primary_payload_) { ret = decoder_->PacketDuration(payload_.data(), payload_.size()); } else { ret = decoder_->PacketDurationRedundant(payload_.data(), payload_.size()); } return (ret < 0) ? 0 : static_cast(ret); } rtc::Optional Decode( rtc::ArrayView decoded) const override { AudioDecoder::SpeechType speech_type = AudioDecoder::kSpeech; int ret; if (is_primary_payload_) { ret = decoder_->Decode( payload_.data(), payload_.size(), decoder_->SampleRateHz(), decoded.size() * sizeof(int16_t), decoded.data(), &speech_type); } else { ret = decoder_->DecodeRedundant( payload_.data(), payload_.size(), decoder_->SampleRateHz(), decoded.size() * sizeof(int16_t), decoded.data(), &speech_type); } if (ret < 0) return rtc::Optional(); return rtc::Optional({static_cast(ret), speech_type}); } private: AudioDecoder* const decoder_; const rtc::Buffer payload_; const bool is_primary_payload_; }; } // namespace AudioDecoder::ParseResult::ParseResult() = default; AudioDecoder::ParseResult::ParseResult(ParseResult&& b) = default; AudioDecoder::ParseResult::ParseResult(uint32_t timestamp, bool primary, std::unique_ptr frame) : timestamp(timestamp), primary(primary), frame(std::move(frame)) {} AudioDecoder::ParseResult::~ParseResult() = default; AudioDecoder::ParseResult& AudioDecoder::ParseResult::operator=( ParseResult&& b) = default; std::vector AudioDecoder::ParsePayload( rtc::Buffer&& payload, uint32_t timestamp, bool is_primary) { std::vector results; std::unique_ptr frame( new LegacyFrame(this, std::move(payload), is_primary)); results.emplace_back(timestamp, is_primary, std::move(frame)); return results; } int AudioDecoder::Decode(const uint8_t* encoded, size_t encoded_len, int sample_rate_hz, size_t max_decoded_bytes, int16_t* decoded, SpeechType* speech_type) { TRACE_EVENT0("webrtc", "AudioDecoder::Decode"); rtc::MsanCheckInitialized(rtc::MakeArrayView(encoded, encoded_len)); int duration = PacketDuration(encoded, encoded_len); if (duration >= 0 && duration * Channels() * sizeof(int16_t) > max_decoded_bytes) { return -1; } return DecodeInternal(encoded, encoded_len, sample_rate_hz, decoded, speech_type); } int AudioDecoder::DecodeRedundant(const uint8_t* encoded, size_t encoded_len, int sample_rate_hz, size_t max_decoded_bytes, int16_t* decoded, SpeechType* speech_type) { TRACE_EVENT0("webrtc", "AudioDecoder::DecodeRedundant"); rtc::MsanCheckInitialized(rtc::MakeArrayView(encoded, encoded_len)); int duration = PacketDurationRedundant(encoded, encoded_len); if (duration >= 0 && duration * Channels() * sizeof(int16_t) > max_decoded_bytes) { return -1; } return DecodeRedundantInternal(encoded, encoded_len, sample_rate_hz, decoded, speech_type); } int AudioDecoder::DecodeRedundantInternal(const uint8_t* encoded, size_t encoded_len, int sample_rate_hz, int16_t* decoded, SpeechType* speech_type) { return DecodeInternal(encoded, encoded_len, sample_rate_hz, decoded, speech_type); } bool AudioDecoder::HasDecodePlc() const { return false; } size_t AudioDecoder::DecodePlc(size_t num_frames, int16_t* decoded) { return 0; } int AudioDecoder::IncomingPacket(const uint8_t* payload, size_t payload_len, uint16_t rtp_sequence_number, uint32_t rtp_timestamp, uint32_t arrival_timestamp) { return 0; } int AudioDecoder::ErrorCode() { return 0; } int AudioDecoder::PacketDuration(const uint8_t* encoded, size_t encoded_len) const { return kNotImplemented; } int AudioDecoder::PacketDurationRedundant(const uint8_t* encoded, size_t encoded_len) const { return kNotImplemented; } bool AudioDecoder::PacketHasFec(const uint8_t* encoded, size_t encoded_len) const { return false; } AudioDecoder::SpeechType AudioDecoder::ConvertSpeechType(int16_t type) { switch (type) { case 0: // TODO(hlundin): Both iSAC and Opus return 0 for speech. case 1: return kSpeech; case 2: return kComfortNoise; default: assert(false); return kSpeech; } } } // namespace webrtc