1
// Copyright (c) 2007-2008, Niels Martin Hansen
2
// All rights reserved.
4
// Redistribution and use in source and binary forms, with or without
5
// modification, are permitted provided that the following conditions are met:
7
// * Redistributions of source code must retain the above copyright notice,
8
// this list of conditions and the following disclaimer.
9
// * Redistributions in binary form must reproduce the above copyright notice,
10
// this list of conditions and the following disclaimer in the documentation
11
// and/or other materials provided with the distribution.
12
// * Neither the name of the Aegisub Group nor the names of its contributors
13
// may be used to endorse or promote products derived from this software
14
// without specific prior written permission.
16
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
17
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19
// ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
20
// LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
21
// CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
22
// SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
23
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
24
// CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
25
// ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
26
// POSSIBILITY OF SUCH DAMAGE.
28
// -----------------------------------------------------------------------------
32
// Website: http://aegisub.cellosoft.com
33
// Contact: mailto:jiifurusu@gmail.com
40
#include "audio_provider_downmix.h"
45
DownmixingAudioProvider::DownmixingAudioProvider(AudioProvider *source) {
46
filename = source->GetFilename();
47
channels = 1; // target
48
src_channels = source->GetChannels();
49
num_samples = source->GetNumSamples();
50
bytes_per_sample = source->GetBytesPerSample();
51
sample_rate = source->GetSampleRate();
56
if (!(bytes_per_sample == 1 || bytes_per_sample == 2))
57
throw _T("Downmixing Audio Provider: Can only downmix 8 and 16 bit audio");
58
if (!source->AreSamplesNativeEndian())
59
throw _T("Downmixing Audio Provider: Source must have machine endian samples");
64
DownmixingAudioProvider::~DownmixingAudioProvider() {
69
// Actual work happens here
70
void DownmixingAudioProvider::GetAudio(void *buf, int64_t start, int64_t count) {
71
if (count == 0) return;
73
// We can do this ourselves
74
if (start >= num_samples) {
75
if (bytes_per_sample == 1)
76
// 8 bit formats are usually unsigned with bias 127
77
memset(buf, 127, count);
79
// While everything else is signed
80
memset(buf, 0, count*bytes_per_sample);
85
// So alloc some temporary memory for this
86
// Depending on use, this might be made faster by using
87
// a pre-allocced block of memory...?
88
char *tmp = new char[count*bytes_per_sample*src_channels];
90
provider->GetAudio(tmp, start, count);
93
// Just average the samples over the channels (really bad if they're out of phase!)
94
// XXX: Assuming here that sample data are in machine endian, an upstream provider should ensure that
95
if (bytes_per_sample == 1) {
96
uint8_t *src = (uint8_t *)tmp;
97
uint8_t *dst = (uint8_t *)buf;
101
for (int c = 0; c < src_channels; c++)
103
*(dst++) = (uint8_t)(sum / src_channels);
107
else if (bytes_per_sample == 2) {
108
int16_t *src = (int16_t *)tmp;
109
int16_t *dst = (int16_t *)buf;
113
for (int c = 0; c < src_channels; c++)
115
*(dst++) = (int16_t)(sum / src_channels);
120
// Done downmixing, free the work buffer