PcmMix.cxx 5.06 KB
Newer Older
1
/*
2
 * Copyright (C) 2003-2013 The Music Player Daemon Project
3 4 5 6 7 8 9 10 11 12 13
 * http://www.musicpd.org
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
14 15 16 17
 *
 * You should have received a copy of the GNU General Public License along
 * with this program; if not, write to the Free Software Foundation, Inc.,
 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
18 19
 */

20
#include "config.h"
Max Kellermann's avatar
Max Kellermann committed
21 22 23
#include "PcmMix.hxx"
#include "PcmVolume.hxx"
#include "PcmUtils.hxx"
24
#include "AudioFormat.hxx"
25 26 27

#include <math.h>

28 29 30
template<typename T, typename U, unsigned bits>
static T
PcmAddVolume(T _a, T _b, int volume1, int volume2)
31
{
32
	U a(_a), b(_b);
33

34 35 36
	U c = ((a * volume1 + b * volume2) +
	       pcm_volume_dither() + PCM_VOLUME_1 / 2)
		/ PCM_VOLUME_1;
37

38
	return PcmClamp<T, U, bits>(c);
39 40
}

41
template<typename T, typename U, unsigned bits>
42
static void
43
PcmAddVolume(T *a, const T *b, unsigned n, int volume1, int volume2)
44
{
45 46
	for (size_t i = 0; i != n; ++i)
		a[i] = PcmAddVolume<T, U, bits>(a[i], b[i], volume1, volume2);
47 48
}

49
template<typename T, typename U, unsigned bits>
50
static void
51
PcmAddVolumeVoid(void *a, const void *b, size_t size, int volume1, int volume2)
52
{
53 54
	constexpr size_t sample_size = sizeof(T);
	assert(size % sample_size == 0);
55

56 57
	PcmAddVolume<T, U, bits>((T *)a, (const T *)b, size / sample_size,
				 volume1, volume2);
58 59
}

60 61 62 63 64 65 66 67 68 69 70 71 72 73
static void
pcm_add_vol_float(float *buffer1, const float *buffer2,
		  unsigned num_samples, float volume1, float volume2)
{
	while (num_samples > 0) {
		float sample1 = *buffer1;
		float sample2 = *buffer2++;

		sample1 = (sample1 * volume1 + sample2 * volume2);
		*buffer1++ = sample1;
		--num_samples;
	}
}

74
static bool
75 76
pcm_add_vol(void *buffer1, const void *buffer2, size_t size,
	    int vol1, int vol2,
77
	    SampleFormat format)
78
{
79
	switch (format) {
80 81
	case SampleFormat::UNDEFINED:
	case SampleFormat::DSD:
82 83 84
		/* not implemented */
		return false;

85
	case SampleFormat::S8:
86 87
		PcmAddVolumeVoid<int8_t, int32_t, 8>(buffer1, buffer2, size,
						     vol1, vol2);
88
		return true;
89

90
	case SampleFormat::S16:
91 92
		PcmAddVolumeVoid<int16_t, int32_t, 16>(buffer1, buffer2, size,
						       vol1, vol2);
93
		return true;
94

95
	case SampleFormat::S24_P32:
96 97
		PcmAddVolumeVoid<int32_t, int64_t, 24>(buffer1, buffer2, size,
						       vol1, vol2);
98
		return true;
99

100
	case SampleFormat::S32:
101 102
		PcmAddVolumeVoid<int32_t, int64_t, 32>(buffer1, buffer2, size,
						       vol1, vol2);
103
		return true;
104

105
	case SampleFormat::FLOAT:
Max Kellermann's avatar
Max Kellermann committed
106 107
		pcm_add_vol_float((float *)buffer1, (const float *)buffer2,
				  size / 4,
108 109 110
				  pcm_volume_to_float(vol1),
				  pcm_volume_to_float(vol2));
		return true;
111
	}
112 113

	assert(false);
114
	gcc_unreachable();
115 116
}

117 118 119
template<typename T, typename U, unsigned bits>
static T
PcmAdd(T _a, T _b)
120
{
121 122
	U a(_a), b(_b);
	return PcmClamp<T, U, bits>(a + b);
123 124
}

125
template<typename T, typename U, unsigned bits>
126
static void
127
PcmAdd(T *a, const T *b, unsigned n)
128
{
129 130
	for (size_t i = 0; i != n; ++i)
		a[i] = PcmAdd<T, U, bits>(a[i], b[i]);
131 132
}

133
template<typename T, typename U, unsigned bits>
134
static void
135
PcmAddVoid(void *a, const void *b, size_t size)
136
{
137 138
	constexpr size_t sample_size = sizeof(T);
	assert(size % sample_size == 0);
139

140
	PcmAdd<T, U, bits>((T *)a, (const T *)b, size / sample_size);
141 142
}

143 144 145 146 147 148 149 150 151 152 153
static void
pcm_add_float(float *buffer1, const float *buffer2, unsigned num_samples)
{
	while (num_samples > 0) {
		float sample1 = *buffer1;
		float sample2 = *buffer2++;
		*buffer1++ = sample1 + sample2;
		--num_samples;
	}
}

154
static bool
155
pcm_add(void *buffer1, const void *buffer2, size_t size,
156
	SampleFormat format)
157
{
158
	switch (format) {
159 160
	case SampleFormat::UNDEFINED:
	case SampleFormat::DSD:
161 162 163
		/* not implemented */
		return false;

164
	case SampleFormat::S8:
165
		PcmAddVoid<int8_t, int32_t, 8>(buffer1, buffer2, size);
166
		return true;
167

168
	case SampleFormat::S16:
169
		PcmAddVoid<int16_t, int32_t, 16>(buffer1, buffer2, size);
170
		return true;
171

172
	case SampleFormat::S24_P32:
173
		PcmAddVoid<int32_t, int64_t, 24>(buffer1, buffer2, size);
174
		return true;
175

176
	case SampleFormat::S32:
177
		PcmAddVoid<int32_t, int64_t, 32>(buffer1, buffer2, size);
178
		return true;
179

180
	case SampleFormat::FLOAT:
Max Kellermann's avatar
Max Kellermann committed
181 182
		pcm_add_float((float *)buffer1, (const float *)buffer2,
			      size / 4);
183
		return true;
184
	}
185 186

	assert(false);
187
	gcc_unreachable();
188 189
}

190
bool
191
pcm_mix(void *buffer1, const void *buffer2, size_t size,
192
	SampleFormat format, float portion1)
193 194
{
	int vol1;
195 196
	float s;

197
	/* portion1 is between 0.0 and 1.0 for crossfading, MixRamp uses -1
198
	 * to signal mixing rather than fading */
199
	if (portion1 < 0)
200
		return pcm_add(buffer1, buffer2, size, format);
201 202

	s = sin(M_PI_2 * portion1);
203 204 205 206 207
	s *= s;

	vol1 = s * PCM_VOLUME_1 + 0.5;
	vol1 = vol1 > PCM_VOLUME_1 ? PCM_VOLUME_1 : (vol1 < 0 ? 0 : vol1);

208
	return pcm_add_vol(buffer1, buffer2, size, vol1, PCM_VOLUME_1 - vol1, format);
209
}