115 lines
		
	
	
		
			4.0 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			115 lines
		
	
	
		
			4.0 KiB
		
	
	
	
		
			C++
		
	
	
	
| /*
 | |
|  *  Copyright (c) 2019 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/audio_coding/acm2/acm_remixing.h"
 | |
| 
 | |
| #include "rtc_base/checks.h"
 | |
| 
 | |
| namespace webrtc {
 | |
| 
 | |
| void DownMixFrame(const AudioFrame& input, rtc::ArrayView<int16_t> output) {
 | |
|   RTC_DCHECK_EQ(input.num_channels_, 2);
 | |
|   RTC_DCHECK_EQ(output.size(), input.samples_per_channel_);
 | |
| 
 | |
|   if (input.muted()) {
 | |
|     std::fill(output.begin(), output.begin() + input.samples_per_channel_, 0);
 | |
|   } else {
 | |
|     const int16_t* const input_data = input.data();
 | |
|     for (size_t n = 0; n < input.samples_per_channel_; ++n) {
 | |
|       output[n] = rtc::dchecked_cast<int16_t>(
 | |
|           (int32_t{input_data[2 * n]} + int32_t{input_data[2 * n + 1]}) >> 1);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void ReMixFrame(const AudioFrame& input,
 | |
|                 size_t num_output_channels,
 | |
|                 std::vector<int16_t>* output) {
 | |
|   const size_t output_size = num_output_channels * input.samples_per_channel_;
 | |
|   RTC_DCHECK(!(input.num_channels_ == 0 && num_output_channels > 0 &&
 | |
|                input.samples_per_channel_ > 0));
 | |
| 
 | |
|   if (output->size() != output_size) {
 | |
|     output->resize(output_size);
 | |
|   }
 | |
| 
 | |
|   // For muted frames, fill the frame with zeros.
 | |
|   if (input.muted()) {
 | |
|     std::fill(output->begin(), output->end(), 0);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   // Ensure that the special case of zero input channels is handled correctly
 | |
|   // (zero samples per channel is already handled correctly in the code below).
 | |
|   if (input.num_channels_ == 0) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   const int16_t* const input_data = input.data();
 | |
|   size_t out_index = 0;
 | |
| 
 | |
|   // When upmixing is needed and the input is mono copy the left channel
 | |
|   // into the left and right channels, and set any remaining channels to zero.
 | |
|   if (input.num_channels_ == 1 && input.num_channels_ < num_output_channels) {
 | |
|     for (size_t k = 0; k < input.samples_per_channel_; ++k) {
 | |
|       (*output)[out_index++] = input_data[k];
 | |
|       (*output)[out_index++] = input_data[k];
 | |
|       for (size_t j = 2; j < num_output_channels; ++j) {
 | |
|         (*output)[out_index++] = 0;
 | |
|       }
 | |
|       RTC_DCHECK_EQ(out_index, (k + 1) * num_output_channels);
 | |
|     }
 | |
|     RTC_DCHECK_EQ(out_index, input.samples_per_channel_ * num_output_channels);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   size_t in_index = 0;
 | |
| 
 | |
|   // When upmixing is needed and the output is surround, copy the available
 | |
|   // channels directly, and set the remaining channels to zero.
 | |
|   if (input.num_channels_ < num_output_channels) {
 | |
|     for (size_t k = 0; k < input.samples_per_channel_; ++k) {
 | |
|       for (size_t j = 0; j < input.num_channels_; ++j) {
 | |
|         (*output)[out_index++] = input_data[in_index++];
 | |
|       }
 | |
|       for (size_t j = input.num_channels_; j < num_output_channels; ++j) {
 | |
|         (*output)[out_index++] = 0;
 | |
|       }
 | |
|       RTC_DCHECK_EQ(in_index, (k + 1) * input.num_channels_);
 | |
|       RTC_DCHECK_EQ(out_index, (k + 1) * num_output_channels);
 | |
|     }
 | |
|     RTC_DCHECK_EQ(in_index, input.samples_per_channel_ * input.num_channels_);
 | |
|     RTC_DCHECK_EQ(out_index, input.samples_per_channel_ * num_output_channels);
 | |
| 
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   // When downmixing is needed, and the input is stereo, average the channels.
 | |
|   if (input.num_channels_ == 2) {
 | |
|     for (size_t n = 0; n < input.samples_per_channel_; ++n) {
 | |
|       (*output)[n] = rtc::dchecked_cast<int16_t>(
 | |
|           (int32_t{input_data[2 * n]} + int32_t{input_data[2 * n + 1]}) >> 1);
 | |
|     }
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   // When downmixing is needed, and the input is multichannel, drop the surplus
 | |
|   // channels.
 | |
|   const size_t num_channels_to_drop = input.num_channels_ - num_output_channels;
 | |
|   for (size_t k = 0; k < input.samples_per_channel_; ++k) {
 | |
|     for (size_t j = 0; j < num_output_channels; ++j) {
 | |
|       (*output)[out_index++] = input_data[in_index++];
 | |
|     }
 | |
|     in_index += num_channels_to_drop;
 | |
|   }
 | |
| }
 | |
| 
 | |
| }  // namespace webrtc
 |