| /* |
| * Copyright (c) 2014 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/red/audio_encoder_copy_red.h" |
| |
| #include <string.h> |
| |
| #include "webrtc/base/checks.h" |
| |
| namespace webrtc { |
| |
| AudioEncoderCopyRed::AudioEncoderCopyRed(const Config& config) |
| : speech_encoder_(config.speech_encoder), |
| red_payload_type_(config.payload_type) { |
| CHECK(speech_encoder_) << "Speech encoder not provided."; |
| } |
| |
| AudioEncoderCopyRed::~AudioEncoderCopyRed() { |
| } |
| |
| int AudioEncoderCopyRed::SampleRateHz() const { |
| return speech_encoder_->SampleRateHz(); |
| } |
| |
| int AudioEncoderCopyRed::RtpTimestampRateHz() const { |
| return speech_encoder_->RtpTimestampRateHz(); |
| } |
| |
| int AudioEncoderCopyRed::NumChannels() const { |
| return speech_encoder_->NumChannels(); |
| } |
| |
| size_t AudioEncoderCopyRed::MaxEncodedBytes() const { |
| return 2 * speech_encoder_->MaxEncodedBytes(); |
| } |
| |
| size_t AudioEncoderCopyRed::Num10MsFramesInNextPacket() const { |
| return speech_encoder_->Num10MsFramesInNextPacket(); |
| } |
| |
| size_t AudioEncoderCopyRed::Max10MsFramesInAPacket() const { |
| return speech_encoder_->Max10MsFramesInAPacket(); |
| } |
| |
| int AudioEncoderCopyRed::GetTargetBitrate() const { |
| return speech_encoder_->GetTargetBitrate(); |
| } |
| |
| void AudioEncoderCopyRed::SetTargetBitrate(int bits_per_second) { |
| speech_encoder_->SetTargetBitrate(bits_per_second); |
| } |
| |
| void AudioEncoderCopyRed::SetProjectedPacketLossRate(double fraction) { |
| DCHECK_GE(fraction, 0.0); |
| DCHECK_LE(fraction, 1.0); |
| speech_encoder_->SetProjectedPacketLossRate(fraction); |
| } |
| |
| AudioEncoder::EncodedInfo AudioEncoderCopyRed::EncodeInternal( |
| uint32_t rtp_timestamp, |
| const int16_t* audio, |
| size_t max_encoded_bytes, |
| uint8_t* encoded) { |
| EncodedInfo info = speech_encoder_->Encode( |
| rtp_timestamp, audio, static_cast<size_t>(SampleRateHz() / 100), |
| max_encoded_bytes, encoded); |
| CHECK_GE(max_encoded_bytes, |
| info.encoded_bytes + secondary_info_.encoded_bytes); |
| CHECK(info.redundant.empty()) << "Cannot use nested redundant encoders."; |
| |
| if (info.encoded_bytes > 0) { |
| // |info| will be implicitly cast to an EncodedInfoLeaf struct, effectively |
| // discarding the (empty) vector of redundant information. This is |
| // intentional. |
| info.redundant.push_back(info); |
| DCHECK_EQ(info.redundant.size(), 1u); |
| if (secondary_info_.encoded_bytes > 0) { |
| memcpy(&encoded[info.encoded_bytes], secondary_encoded_.data(), |
| secondary_info_.encoded_bytes); |
| info.redundant.push_back(secondary_info_); |
| DCHECK_EQ(info.redundant.size(), 2u); |
| } |
| // Save primary to secondary. |
| secondary_encoded_.SetSize(info.encoded_bytes); |
| memcpy(secondary_encoded_.data(), encoded, info.encoded_bytes); |
| secondary_info_ = info; |
| DCHECK_EQ(info.speech, info.redundant[0].speech); |
| } |
| // Update main EncodedInfo. |
| info.payload_type = red_payload_type_; |
| info.encoded_bytes = 0; |
| for (std::vector<EncodedInfoLeaf>::const_iterator it = info.redundant.begin(); |
| it != info.redundant.end(); ++it) { |
| info.encoded_bytes += it->encoded_bytes; |
| } |
| return info; |
| } |
| |
| } // namespace webrtc |