From c18b3945a1ccde3a644e20d1f1aa973ec7fd9ed4 Mon Sep 17 00:00:00 2001 From: "Charles E. Lehner" Date: Sun, 28 Mar 2021 20:36:31 +0300 Subject: [PATCH] Support big-endian wav read/write --- webrtc/common_audio/wav_file.cc | 51 +++++++++++++++++++++------------ 1 file changed, 33 insertions(+), 18 deletions(-) diff --git a/webrtc/common_audio/wav_file.cc b/webrtc/common_audio/wav_file.cc index e49126f..982d0b5 100644 --- a/webrtc/common_audio/wav_file.cc +++ b/webrtc/common_audio/wav_file.cc @@ -14,6 +14,7 @@ #include #include +#include #include #include #include @@ -89,10 +90,6 @@ void WavReader::Reset() { size_t WavReader::ReadSamples(const size_t num_samples, int16_t* const samples) { -#ifndef WEBRTC_ARCH_LITTLE_ENDIAN -#error "Need to convert samples to big-endian when reading from WAV file" -#endif - size_t num_samples_left_to_read = num_samples; size_t next_chunk_start = 0; while (num_samples_left_to_read > 0 && num_unread_samples_ > 0) { @@ -124,15 +121,16 @@ size_t WavReader::ReadSamples(const size_t num_samples, num_unread_samples_ -= num_samples_read; num_samples_left_to_read -= num_samples_read; } +#ifndef WEBRTC_ARCH_LITTLE_ENDIAN + for (size_t i = 0; i < num_samples; i++) { + samples[i] = bswap_16(samples[i]); + } +#endif return num_samples - num_samples_left_to_read; } size_t WavReader::ReadSamples(const size_t num_samples, float* const samples) { -#ifndef WEBRTC_ARCH_LITTLE_ENDIAN -#error "Need to convert samples to big-endian when reading from WAV file" -#endif - size_t num_samples_left_to_read = num_samples; size_t next_chunk_start = 0; while (num_samples_left_to_read > 0 && num_unread_samples_ > 0) { @@ -170,6 +168,12 @@ size_t WavReader::ReadSamples(const size_t num_samples, float* const samples) { num_unread_samples_ -= num_samples_read; num_samples_left_to_read -= num_samples_read; } +#ifndef WEBRTC_ARCH_LITTLE_ENDIAN + // TODO: is this the right place for this? + for (size_t i = 0; i < num_samples; i++) { + samples[i] = bswap_16(samples[i]); + } +#endif return num_samples - num_samples_left_to_read; } @@ -213,23 +217,33 @@ WavWriter::WavWriter(FileWrapper file, } void WavWriter::WriteSamples(const int16_t* samples, size_t num_samples) { -#ifndef WEBRTC_ARCH_LITTLE_ENDIAN -#error "Need to convert samples to little-endian when writing to WAV file" -#endif - for (size_t i = 0; i < num_samples; i += kMaxChunksize) { const size_t num_remaining_samples = num_samples - i; const size_t num_samples_to_write = std::min(kMaxChunksize, num_remaining_samples); if (format_ == WavFormat::kWavFormatPcm) { +#ifndef WEBRTC_ARCH_LITTLE_ENDIAN RTC_CHECK( file_.Write(&samples[i], num_samples_to_write * sizeof(samples[0]))); +#else + std::array converted_samples; + for (size_t j = 0; j < num_samples_to_write; ++j) { + converted_samples[j] = bswap_16(samples[i + j]); + } + RTC_CHECK( + file_.Write(converted_samples.data(), + num_samples_to_write * sizeof(converted_samples[0]))); +#endif } else { RTC_CHECK_EQ(format_, WavFormat::kWavFormatIeeeFloat); std::array converted_samples; for (size_t j = 0; j < num_samples_to_write; ++j) { - converted_samples[j] = S16ToFloat(samples[i + j]); + int16_t sample = samples[i + j]; +#ifndef WEBRTC_ARCH_LITTLE_ENDIAN + sample = bswap_16(sample); +#endif + converted_samples[j] = S16ToFloat(sample); } RTC_CHECK( file_.Write(converted_samples.data(), @@ -243,10 +257,6 @@ void WavWriter::WriteSamples(const int16_t* samples, size_t num_samples) { } void WavWriter::WriteSamples(const float* samples, size_t num_samples) { -#ifndef WEBRTC_ARCH_LITTLE_ENDIAN -#error "Need to convert samples to little-endian when writing to WAV file" -#endif - for (size_t i = 0; i < num_samples; i += kMaxChunksize) { const size_t num_remaining_samples = num_samples - i; const size_t num_samples_to_write = @@ -255,7 +265,11 @@ void WavWriter::WriteSamples(const float* samples, size_t num_samples) { if (format_ == WavFormat::kWavFormatPcm) { std::array converted_samples; for (size_t j = 0; j < num_samples_to_write; ++j) { - converted_samples[j] = FloatS16ToS16(samples[i + j]); + int16_t sample = FloatS16ToS16(samples[i + j]); +#ifndef WEBRTC_ARCH_LITTLE_ENDIAN + sample = bswap_16(sample); +#endif + converted_samples[j] = sample; } RTC_CHECK( file_.Write(converted_samples.data(), @@ -264,6 +278,7 @@ void WavWriter::WriteSamples(const float* samples, size_t num_samples) { RTC_CHECK_EQ(format_, WavFormat::kWavFormatIeeeFloat); std::array converted_samples; for (size_t j = 0; j < num_samples_to_write; ++j) { + // TODO: is swap needed for big-endian here? converted_samples[j] = FloatS16ToFloat(samples[i + j]); } RTC_CHECK( -- GitLab