153a5a1b3Sopenharmony_ci/*** 253a5a1b3Sopenharmony_ci This file is part of PulseAudio. 353a5a1b3Sopenharmony_ci 453a5a1b3Sopenharmony_ci Copyright 2004-2006 Lennart Poettering 553a5a1b3Sopenharmony_ci Copyright 2006 Pierre Ossman <ossman@cendio.se> for Cendio AB 653a5a1b3Sopenharmony_ci 753a5a1b3Sopenharmony_ci PulseAudio is free software; you can redistribute it and/or modify 853a5a1b3Sopenharmony_ci it under the terms of the GNU Lesser General Public License as published 953a5a1b3Sopenharmony_ci by the Free Software Foundation; either version 2.1 of the License, 1053a5a1b3Sopenharmony_ci or (at your option) any later version. 1153a5a1b3Sopenharmony_ci 1253a5a1b3Sopenharmony_ci PulseAudio is distributed in the hope that it will be useful, but 1353a5a1b3Sopenharmony_ci WITHOUT ANY WARRANTY; without even the implied warranty of 1453a5a1b3Sopenharmony_ci MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 1553a5a1b3Sopenharmony_ci General Public License for more details. 1653a5a1b3Sopenharmony_ci 1753a5a1b3Sopenharmony_ci You should have received a copy of the GNU Lesser General Public License 1853a5a1b3Sopenharmony_ci along with PulseAudio; if not, see <http://www.gnu.org/licenses/>. 1953a5a1b3Sopenharmony_ci***/ 2053a5a1b3Sopenharmony_ci 2153a5a1b3Sopenharmony_ci#ifdef HAVE_CONFIG_H 2253a5a1b3Sopenharmony_ci#include <config.h> 2353a5a1b3Sopenharmony_ci#endif 2453a5a1b3Sopenharmony_ci 2553a5a1b3Sopenharmony_ci#include <pulsecore/macro.h> 2653a5a1b3Sopenharmony_ci#include <pulsecore/g711.h> 2753a5a1b3Sopenharmony_ci#include <pulsecore/endianmacros.h> 2853a5a1b3Sopenharmony_ci 2953a5a1b3Sopenharmony_ci#include "sample-util.h" 3053a5a1b3Sopenharmony_ci 3153a5a1b3Sopenharmony_cistatic void pa_volume_u8_c(uint8_t *samples, const int32_t *volumes, unsigned channels, unsigned length) { 3253a5a1b3Sopenharmony_ci unsigned channel; 3353a5a1b3Sopenharmony_ci 3453a5a1b3Sopenharmony_ci for (channel = 0; length; length--) { 3553a5a1b3Sopenharmony_ci int32_t t = pa_mult_s16_volume(*samples - 0x80, volumes[channel]); 3653a5a1b3Sopenharmony_ci 3753a5a1b3Sopenharmony_ci t = PA_CLAMP_UNLIKELY(t, -0x80, 0x7F); 3853a5a1b3Sopenharmony_ci *samples++ = (uint8_t) (t + 0x80); 3953a5a1b3Sopenharmony_ci 4053a5a1b3Sopenharmony_ci if (PA_UNLIKELY(++channel >= channels)) 4153a5a1b3Sopenharmony_ci channel = 0; 4253a5a1b3Sopenharmony_ci } 4353a5a1b3Sopenharmony_ci} 4453a5a1b3Sopenharmony_ci 4553a5a1b3Sopenharmony_cistatic void pa_volume_alaw_c(uint8_t *samples, const int32_t *volumes, unsigned channels, unsigned length) { 4653a5a1b3Sopenharmony_ci unsigned channel; 4753a5a1b3Sopenharmony_ci 4853a5a1b3Sopenharmony_ci for (channel = 0; length; length--) { 4953a5a1b3Sopenharmony_ci int32_t t = pa_mult_s16_volume(st_alaw2linear16(*samples), volumes[channel]); 5053a5a1b3Sopenharmony_ci 5153a5a1b3Sopenharmony_ci t = PA_CLAMP_UNLIKELY(t, -0x8000, 0x7FFF); 5253a5a1b3Sopenharmony_ci *samples++ = (uint8_t) st_13linear2alaw((int16_t) t >> 3); 5353a5a1b3Sopenharmony_ci 5453a5a1b3Sopenharmony_ci if (PA_UNLIKELY(++channel >= channels)) 5553a5a1b3Sopenharmony_ci channel = 0; 5653a5a1b3Sopenharmony_ci } 5753a5a1b3Sopenharmony_ci} 5853a5a1b3Sopenharmony_ci 5953a5a1b3Sopenharmony_cistatic void pa_volume_ulaw_c(uint8_t *samples, const int32_t *volumes, unsigned channels, unsigned length) { 6053a5a1b3Sopenharmony_ci unsigned channel; 6153a5a1b3Sopenharmony_ci 6253a5a1b3Sopenharmony_ci for (channel = 0; length; length--) { 6353a5a1b3Sopenharmony_ci int32_t t = pa_mult_s16_volume(st_ulaw2linear16(*samples), volumes[channel]); 6453a5a1b3Sopenharmony_ci 6553a5a1b3Sopenharmony_ci t = PA_CLAMP_UNLIKELY(t, -0x8000, 0x7FFF); 6653a5a1b3Sopenharmony_ci *samples++ = (uint8_t) st_14linear2ulaw((int16_t) t >> 2); 6753a5a1b3Sopenharmony_ci 6853a5a1b3Sopenharmony_ci if (PA_UNLIKELY(++channel >= channels)) 6953a5a1b3Sopenharmony_ci channel = 0; 7053a5a1b3Sopenharmony_ci } 7153a5a1b3Sopenharmony_ci} 7253a5a1b3Sopenharmony_ci 7353a5a1b3Sopenharmony_cistatic void pa_volume_s16ne_c(int16_t *samples, const int32_t *volumes, unsigned channels, unsigned length) { 7453a5a1b3Sopenharmony_ci unsigned channel; 7553a5a1b3Sopenharmony_ci 7653a5a1b3Sopenharmony_ci length /= sizeof(int16_t); 7753a5a1b3Sopenharmony_ci 7853a5a1b3Sopenharmony_ci for (channel = 0; length; length--) { 7953a5a1b3Sopenharmony_ci int32_t t = pa_mult_s16_volume(*samples, volumes[channel]); 8053a5a1b3Sopenharmony_ci 8153a5a1b3Sopenharmony_ci t = PA_CLAMP_UNLIKELY(t, -0x8000, 0x7FFF); 8253a5a1b3Sopenharmony_ci *samples++ = (int16_t) t; 8353a5a1b3Sopenharmony_ci 8453a5a1b3Sopenharmony_ci if (PA_UNLIKELY(++channel >= channels)) 8553a5a1b3Sopenharmony_ci channel = 0; 8653a5a1b3Sopenharmony_ci } 8753a5a1b3Sopenharmony_ci} 8853a5a1b3Sopenharmony_ci 8953a5a1b3Sopenharmony_cistatic void pa_volume_s16re_c(int16_t *samples, const int32_t *volumes, unsigned channels, unsigned length) { 9053a5a1b3Sopenharmony_ci unsigned channel; 9153a5a1b3Sopenharmony_ci 9253a5a1b3Sopenharmony_ci length /= sizeof(int16_t); 9353a5a1b3Sopenharmony_ci 9453a5a1b3Sopenharmony_ci for (channel = 0; length; length--) { 9553a5a1b3Sopenharmony_ci int32_t t = pa_mult_s16_volume(PA_INT16_SWAP(*samples), volumes[channel]); 9653a5a1b3Sopenharmony_ci 9753a5a1b3Sopenharmony_ci t = PA_CLAMP_UNLIKELY(t, -0x8000, 0x7FFF); 9853a5a1b3Sopenharmony_ci *samples++ = PA_INT16_SWAP((int16_t) t); 9953a5a1b3Sopenharmony_ci 10053a5a1b3Sopenharmony_ci if (PA_UNLIKELY(++channel >= channels)) 10153a5a1b3Sopenharmony_ci channel = 0; 10253a5a1b3Sopenharmony_ci } 10353a5a1b3Sopenharmony_ci} 10453a5a1b3Sopenharmony_ci 10553a5a1b3Sopenharmony_cistatic void pa_volume_float32ne_c(float *samples, const float *volumes, unsigned channels, unsigned length) { 10653a5a1b3Sopenharmony_ci unsigned channel; 10753a5a1b3Sopenharmony_ci 10853a5a1b3Sopenharmony_ci length /= sizeof(float); 10953a5a1b3Sopenharmony_ci 11053a5a1b3Sopenharmony_ci for (channel = 0; length; length--) { 11153a5a1b3Sopenharmony_ci *samples++ *= volumes[channel]; 11253a5a1b3Sopenharmony_ci 11353a5a1b3Sopenharmony_ci if (PA_UNLIKELY(++channel >= channels)) 11453a5a1b3Sopenharmony_ci channel = 0; 11553a5a1b3Sopenharmony_ci } 11653a5a1b3Sopenharmony_ci} 11753a5a1b3Sopenharmony_ci 11853a5a1b3Sopenharmony_cistatic void pa_volume_float32re_c(float *samples, const float *volumes, unsigned channels, unsigned length) { 11953a5a1b3Sopenharmony_ci unsigned channel; 12053a5a1b3Sopenharmony_ci 12153a5a1b3Sopenharmony_ci length /= sizeof(float); 12253a5a1b3Sopenharmony_ci 12353a5a1b3Sopenharmony_ci for (channel = 0; length; length--) { 12453a5a1b3Sopenharmony_ci float t; 12553a5a1b3Sopenharmony_ci 12653a5a1b3Sopenharmony_ci t = PA_READ_FLOAT32RE(samples); 12753a5a1b3Sopenharmony_ci t *= volumes[channel]; 12853a5a1b3Sopenharmony_ci PA_WRITE_FLOAT32RE(samples++, t); 12953a5a1b3Sopenharmony_ci 13053a5a1b3Sopenharmony_ci if (PA_UNLIKELY(++channel >= channels)) 13153a5a1b3Sopenharmony_ci channel = 0; 13253a5a1b3Sopenharmony_ci } 13353a5a1b3Sopenharmony_ci} 13453a5a1b3Sopenharmony_ci 13553a5a1b3Sopenharmony_cistatic void pa_volume_s32ne_c(int32_t *samples, const int32_t *volumes, unsigned channels, unsigned length) { 13653a5a1b3Sopenharmony_ci unsigned channel; 13753a5a1b3Sopenharmony_ci 13853a5a1b3Sopenharmony_ci length /= sizeof(int32_t); 13953a5a1b3Sopenharmony_ci 14053a5a1b3Sopenharmony_ci for (channel = 0; length; length--) { 14153a5a1b3Sopenharmony_ci int64_t t; 14253a5a1b3Sopenharmony_ci 14353a5a1b3Sopenharmony_ci t = (int64_t)(*samples); 14453a5a1b3Sopenharmony_ci t = (t * volumes[channel]) >> 16; 14553a5a1b3Sopenharmony_ci t = PA_CLAMP_UNLIKELY(t, -0x80000000LL, 0x7FFFFFFFLL); 14653a5a1b3Sopenharmony_ci *samples++ = (int32_t) t; 14753a5a1b3Sopenharmony_ci 14853a5a1b3Sopenharmony_ci if (PA_UNLIKELY(++channel >= channels)) 14953a5a1b3Sopenharmony_ci channel = 0; 15053a5a1b3Sopenharmony_ci } 15153a5a1b3Sopenharmony_ci} 15253a5a1b3Sopenharmony_ci 15353a5a1b3Sopenharmony_cistatic void pa_volume_s32re_c(int32_t *samples, const int32_t *volumes, unsigned channels, unsigned length) { 15453a5a1b3Sopenharmony_ci unsigned channel; 15553a5a1b3Sopenharmony_ci 15653a5a1b3Sopenharmony_ci length /= sizeof(int32_t); 15753a5a1b3Sopenharmony_ci 15853a5a1b3Sopenharmony_ci for (channel = 0; length; length--) { 15953a5a1b3Sopenharmony_ci int64_t t; 16053a5a1b3Sopenharmony_ci 16153a5a1b3Sopenharmony_ci t = (int64_t) PA_INT32_SWAP(*samples); 16253a5a1b3Sopenharmony_ci t = (t * volumes[channel]) >> 16; 16353a5a1b3Sopenharmony_ci t = PA_CLAMP_UNLIKELY(t, -0x80000000LL, 0x7FFFFFFFLL); 16453a5a1b3Sopenharmony_ci *samples++ = PA_INT32_SWAP((int32_t) t); 16553a5a1b3Sopenharmony_ci 16653a5a1b3Sopenharmony_ci if (PA_UNLIKELY(++channel >= channels)) 16753a5a1b3Sopenharmony_ci channel = 0; 16853a5a1b3Sopenharmony_ci } 16953a5a1b3Sopenharmony_ci} 17053a5a1b3Sopenharmony_ci 17153a5a1b3Sopenharmony_cistatic void pa_volume_s24ne_c(uint8_t *samples, const int32_t *volumes, unsigned channels, unsigned length) { 17253a5a1b3Sopenharmony_ci unsigned channel; 17353a5a1b3Sopenharmony_ci uint8_t *e; 17453a5a1b3Sopenharmony_ci 17553a5a1b3Sopenharmony_ci e = samples + length; 17653a5a1b3Sopenharmony_ci 17753a5a1b3Sopenharmony_ci for (channel = 0; samples < e; samples += 3) { 17853a5a1b3Sopenharmony_ci int64_t t; 17953a5a1b3Sopenharmony_ci 18053a5a1b3Sopenharmony_ci t = (int64_t)((int32_t) (PA_READ24NE(samples) << 8)); 18153a5a1b3Sopenharmony_ci t = (t * volumes[channel]) >> 16; 18253a5a1b3Sopenharmony_ci t = PA_CLAMP_UNLIKELY(t, -0x80000000LL, 0x7FFFFFFFLL); 18353a5a1b3Sopenharmony_ci PA_WRITE24NE(samples, ((uint32_t) (int32_t) t) >> 8); 18453a5a1b3Sopenharmony_ci 18553a5a1b3Sopenharmony_ci if (PA_UNLIKELY(++channel >= channels)) 18653a5a1b3Sopenharmony_ci channel = 0; 18753a5a1b3Sopenharmony_ci } 18853a5a1b3Sopenharmony_ci} 18953a5a1b3Sopenharmony_ci 19053a5a1b3Sopenharmony_cistatic void pa_volume_s24re_c(uint8_t *samples, const int32_t *volumes, unsigned channels, unsigned length) { 19153a5a1b3Sopenharmony_ci unsigned channel; 19253a5a1b3Sopenharmony_ci uint8_t *e; 19353a5a1b3Sopenharmony_ci 19453a5a1b3Sopenharmony_ci e = samples + length; 19553a5a1b3Sopenharmony_ci 19653a5a1b3Sopenharmony_ci for (channel = 0; samples < e; samples += 3) { 19753a5a1b3Sopenharmony_ci int64_t t; 19853a5a1b3Sopenharmony_ci 19953a5a1b3Sopenharmony_ci t = (int64_t)((int32_t) (PA_READ24RE(samples) << 8)); 20053a5a1b3Sopenharmony_ci t = (t * volumes[channel]) >> 16; 20153a5a1b3Sopenharmony_ci t = PA_CLAMP_UNLIKELY(t, -0x80000000LL, 0x7FFFFFFFLL); 20253a5a1b3Sopenharmony_ci PA_WRITE24RE(samples, ((uint32_t) (int32_t) t) >> 8); 20353a5a1b3Sopenharmony_ci 20453a5a1b3Sopenharmony_ci if (PA_UNLIKELY(++channel >= channels)) 20553a5a1b3Sopenharmony_ci channel = 0; 20653a5a1b3Sopenharmony_ci } 20753a5a1b3Sopenharmony_ci} 20853a5a1b3Sopenharmony_ci 20953a5a1b3Sopenharmony_cistatic void pa_volume_s24_32ne_c(uint32_t *samples, const int32_t *volumes, unsigned channels, unsigned length) { 21053a5a1b3Sopenharmony_ci unsigned channel; 21153a5a1b3Sopenharmony_ci 21253a5a1b3Sopenharmony_ci length /= sizeof(uint32_t); 21353a5a1b3Sopenharmony_ci 21453a5a1b3Sopenharmony_ci for (channel = 0; length; length--) { 21553a5a1b3Sopenharmony_ci int64_t t; 21653a5a1b3Sopenharmony_ci 21753a5a1b3Sopenharmony_ci t = (int64_t) ((int32_t) (*samples << 8)); 21853a5a1b3Sopenharmony_ci t = (t * volumes[channel]) >> 16; 21953a5a1b3Sopenharmony_ci t = PA_CLAMP_UNLIKELY(t, -0x80000000LL, 0x7FFFFFFFLL); 22053a5a1b3Sopenharmony_ci *samples++ = ((uint32_t) ((int32_t) t)) >> 8; 22153a5a1b3Sopenharmony_ci 22253a5a1b3Sopenharmony_ci if (PA_UNLIKELY(++channel >= channels)) 22353a5a1b3Sopenharmony_ci channel = 0; 22453a5a1b3Sopenharmony_ci } 22553a5a1b3Sopenharmony_ci} 22653a5a1b3Sopenharmony_ci 22753a5a1b3Sopenharmony_cistatic void pa_volume_s24_32re_c(uint32_t *samples, const int32_t *volumes, unsigned channels, unsigned length) { 22853a5a1b3Sopenharmony_ci unsigned channel; 22953a5a1b3Sopenharmony_ci 23053a5a1b3Sopenharmony_ci length /= sizeof(uint32_t); 23153a5a1b3Sopenharmony_ci 23253a5a1b3Sopenharmony_ci for (channel = 0; length; length--) { 23353a5a1b3Sopenharmony_ci int64_t t; 23453a5a1b3Sopenharmony_ci 23553a5a1b3Sopenharmony_ci t = (int64_t) ((int32_t) (PA_UINT32_SWAP(*samples) << 8)); 23653a5a1b3Sopenharmony_ci t = (t * volumes[channel]) >> 16; 23753a5a1b3Sopenharmony_ci t = PA_CLAMP_UNLIKELY(t, -0x80000000LL, 0x7FFFFFFFLL); 23853a5a1b3Sopenharmony_ci *samples++ = PA_UINT32_SWAP(((uint32_t) ((int32_t) t)) >> 8); 23953a5a1b3Sopenharmony_ci 24053a5a1b3Sopenharmony_ci if (PA_UNLIKELY(++channel >= channels)) 24153a5a1b3Sopenharmony_ci channel = 0; 24253a5a1b3Sopenharmony_ci } 24353a5a1b3Sopenharmony_ci} 24453a5a1b3Sopenharmony_ci 24553a5a1b3Sopenharmony_cistatic pa_do_volume_func_t do_volume_table[] = { 24653a5a1b3Sopenharmony_ci [PA_SAMPLE_U8] = (pa_do_volume_func_t) pa_volume_u8_c, 24753a5a1b3Sopenharmony_ci [PA_SAMPLE_ALAW] = (pa_do_volume_func_t) pa_volume_alaw_c, 24853a5a1b3Sopenharmony_ci [PA_SAMPLE_ULAW] = (pa_do_volume_func_t) pa_volume_ulaw_c, 24953a5a1b3Sopenharmony_ci [PA_SAMPLE_S16NE] = (pa_do_volume_func_t) pa_volume_s16ne_c, 25053a5a1b3Sopenharmony_ci [PA_SAMPLE_S16RE] = (pa_do_volume_func_t) pa_volume_s16re_c, 25153a5a1b3Sopenharmony_ci [PA_SAMPLE_FLOAT32NE] = (pa_do_volume_func_t) pa_volume_float32ne_c, 25253a5a1b3Sopenharmony_ci [PA_SAMPLE_FLOAT32RE] = (pa_do_volume_func_t) pa_volume_float32re_c, 25353a5a1b3Sopenharmony_ci [PA_SAMPLE_S32NE] = (pa_do_volume_func_t) pa_volume_s32ne_c, 25453a5a1b3Sopenharmony_ci [PA_SAMPLE_S32RE] = (pa_do_volume_func_t) pa_volume_s32re_c, 25553a5a1b3Sopenharmony_ci [PA_SAMPLE_S24NE] = (pa_do_volume_func_t) pa_volume_s24ne_c, 25653a5a1b3Sopenharmony_ci [PA_SAMPLE_S24RE] = (pa_do_volume_func_t) pa_volume_s24re_c, 25753a5a1b3Sopenharmony_ci [PA_SAMPLE_S24_32NE] = (pa_do_volume_func_t) pa_volume_s24_32ne_c, 25853a5a1b3Sopenharmony_ci [PA_SAMPLE_S24_32RE] = (pa_do_volume_func_t) pa_volume_s24_32re_c 25953a5a1b3Sopenharmony_ci}; 26053a5a1b3Sopenharmony_ci 26153a5a1b3Sopenharmony_cipa_do_volume_func_t pa_get_volume_func(pa_sample_format_t f) { 26253a5a1b3Sopenharmony_ci pa_assert(pa_sample_format_valid(f)); 26353a5a1b3Sopenharmony_ci 26453a5a1b3Sopenharmony_ci return do_volume_table[f]; 26553a5a1b3Sopenharmony_ci} 26653a5a1b3Sopenharmony_ci 26753a5a1b3Sopenharmony_civoid pa_set_volume_func(pa_sample_format_t f, pa_do_volume_func_t func) { 26853a5a1b3Sopenharmony_ci pa_assert(pa_sample_format_valid(f)); 26953a5a1b3Sopenharmony_ci 27053a5a1b3Sopenharmony_ci do_volume_table[f] = func; 27153a5a1b3Sopenharmony_ci} 272