/* * 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 "api/audio_codecs/audio_decoder.h" #include #include #include #include "api/array_view.h" #include "rtc_base/checks.h" #include "rtc_base/sanitizer.h" #include "rtc_base/trace_event.h" namespace webrtc { namespace { class OldStyleEncodedFrame final : public AudioDecoder::EncodedAudioFrame { public: OldStyleEncodedFrame(AudioDecoder* decoder, rtc::Buffer&& payload) : decoder_(decoder), payload_(std::move(payload)) {} size_t Duration() const override { const int ret = decoder_->PacketDuration(payload_.data(), payload_.size()); return ret < 0 ? 0 : static_cast(ret); } absl::optional Decode( rtc::ArrayView decoded) const override { auto speech_type = AudioDecoder::kSpeech; const int ret = decoder_->Decode( payload_.data(), payload_.size(), decoder_->SampleRateHz(), decoded.size() * sizeof(int16_t), decoded.data(), &speech_type); return ret < 0 ? absl::nullopt : absl::optional( {static_cast(ret), speech_type}); } private: AudioDecoder* const decoder_; const rtc::Buffer payload_; }; } // namespace bool AudioDecoder::EncodedAudioFrame::IsDtxPacket() const { return false; } AudioDecoder::ParseResult::ParseResult() = default; AudioDecoder::ParseResult::ParseResult(ParseResult&& b) = default; AudioDecoder::ParseResult::ParseResult(uint32_t timestamp, int priority, std::unique_ptr frame) : timestamp(timestamp), priority(priority), frame(std::move(frame)) { RTC_DCHECK_GE(priority, 0); } AudioDecoder::ParseResult::~ParseResult() = default; AudioDecoder::ParseResult& AudioDecoder::ParseResult::operator=( ParseResult&& b) = default; std::vector AudioDecoder::ParsePayload( rtc::Buffer&& payload, uint32_t timestamp) { std::vector results; std::unique_ptr frame( new OldStyleEncodedFrame(this, std::move(payload))); results.emplace_back(timestamp, 0, 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; } // TODO(bugs.webrtc.org/9676): Remove default implementation. void AudioDecoder::GeneratePlc(size_t /*requested_samples_per_channel*/, rtc::BufferT* /*concealment_audio*/) {} 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