2004-06-23 23:17:30 +00:00
|
|
|
#ifndef foosampleutilhfoo
|
|
|
|
|
#define foosampleutilhfoo
|
|
|
|
|
|
2004-07-16 19:56:36 +00:00
|
|
|
/***
|
2006-06-19 21:53:48 +00:00
|
|
|
This file is part of PulseAudio.
|
2007-01-04 13:43:45 +00:00
|
|
|
|
2007-02-13 15:35:19 +00:00
|
|
|
Copyright 2004-2006 Lennart Poettering
|
|
|
|
|
Copyright 2006 Pierre Ossman <ossman@cendio.se> for Cendio AB
|
|
|
|
|
|
2006-06-19 21:53:48 +00:00
|
|
|
PulseAudio is free software; you can redistribute it and/or modify
|
2004-11-14 14:58:54 +00:00
|
|
|
it under the terms of the GNU Lesser General Public License as published
|
2009-03-03 20:23:02 +00:00
|
|
|
by the Free Software Foundation; either version 2.1 of the License,
|
2004-07-16 19:56:36 +00:00
|
|
|
or (at your option) any later version.
|
2007-01-04 13:43:45 +00:00
|
|
|
|
2006-06-19 21:53:48 +00:00
|
|
|
PulseAudio is distributed in the hope that it will be useful, but
|
2004-07-16 19:56:36 +00:00
|
|
|
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.
|
2007-01-04 13:43:45 +00:00
|
|
|
|
2004-11-14 14:58:54 +00:00
|
|
|
You should have received a copy of the GNU Lesser General Public License
|
2014-11-26 14:14:51 +01:00
|
|
|
along with PulseAudio; if not, see <http://www.gnu.org/licenses/>.
|
2004-07-16 19:56:36 +00:00
|
|
|
***/
|
|
|
|
|
|
core: Refactor code to multiply s16 by volume
move code to function pa_mult_s16_volume() in sample-util.h
use 64 bit integers on 64 bit platforms (it's faster)
on i5, 2.5GHz (64-bit)
Running suite(s): Mult-s16
32 bit mult: 1272300 usec (avg: 12723, min = 12533, max = 18749, stddev = 620.48).
64 bit mult: 852241 usec (avg: 8522.41, min = 8420, max = 9148, stddev = 109.388).
100%: Checks: 1, Failures: 0, Errors: 0
on Pentium D, 3.4GHz (32-bit)
Running suite(s): Mult-s16
32 bit mult: 2228504 usec (avg: 22285, min = 18775, max = 29648, stddev = 3865.59).
64 bit mult: 5546861 usec (avg: 55468.6, min = 55028, max = 64924, stddev = 978.981).
100%: Checks: 1, Failures: 0, Errors: 0
on TI DM3730, Cortex-A8, 800MHz (32-bit)
Running suite(s): Mult-s16
32 bit mult: 23708900 usec (avg: 237089, min = 191864, max = 557312, stddev = 77503.6).
64 bit mult: 22190039 usec (avg: 221900, min = 177978, max = 480469, stddev = 68520.5).
100%: Checks: 1, Failures: 0, Errors: 0
there is a test program called mult-s16-test which checks that the functions compute the
same results, and compares runtime
Signed-off-by: Peter Meerwald <pmeerw@pmeerw.net>
2013-02-13 17:27:05 +01:00
|
|
|
#include <inttypes.h>
|
|
|
|
|
#include <limits.h>
|
|
|
|
|
|
2011-06-13 15:04:33 +02:00
|
|
|
#include <pulse/gccmacro.h>
|
2006-06-19 21:53:48 +00:00
|
|
|
#include <pulse/sample.h>
|
|
|
|
|
#include <pulse/volume.h>
|
2009-06-17 03:08:34 +02:00
|
|
|
#include <pulse/channelmap.h>
|
2011-06-13 15:04:33 +02:00
|
|
|
|
2006-06-19 21:53:48 +00:00
|
|
|
#include <pulsecore/memblock.h>
|
|
|
|
|
#include <pulsecore/memchunk.h>
|
2004-06-29 20:37:24 +00:00
|
|
|
|
2008-05-15 23:34:41 +00:00
|
|
|
typedef struct pa_silence_cache {
|
|
|
|
|
pa_memblock* blocks[PA_SAMPLE_MAX];
|
|
|
|
|
} pa_silence_cache;
|
|
|
|
|
|
|
|
|
|
void pa_silence_cache_init(pa_silence_cache *cache);
|
|
|
|
|
void pa_silence_cache_done(pa_silence_cache *cache);
|
|
|
|
|
|
|
|
|
|
void *pa_silence_memory(void *p, size_t length, const pa_sample_spec *spec);
|
|
|
|
|
pa_memchunk* pa_silence_memchunk(pa_memchunk *c, const pa_sample_spec *spec);
|
|
|
|
|
pa_memblock* pa_silence_memblock(pa_memblock *b, const pa_sample_spec *spec);
|
|
|
|
|
|
|
|
|
|
pa_memchunk* pa_silence_memchunk_get(pa_silence_cache *cache, pa_mempool *pool, pa_memchunk* ret, const pa_sample_spec *spec, size_t length);
|
2004-06-23 23:17:30 +00:00
|
|
|
|
2007-10-28 19:13:50 +00:00
|
|
|
size_t pa_frame_align(size_t l, const pa_sample_spec *ss) PA_GCC_PURE;
|
|
|
|
|
|
2013-06-27 19:28:09 +02:00
|
|
|
bool pa_frame_aligned(size_t l, const pa_sample_spec *ss) PA_GCC_PURE;
|
2007-10-28 19:13:50 +00:00
|
|
|
|
|
|
|
|
void pa_interleave(const void *src[], unsigned channels, void *dst, size_t ss, unsigned n);
|
|
|
|
|
void pa_deinterleave(const void *src, void *dst[], unsigned channels, size_t ss, unsigned n);
|
|
|
|
|
|
2008-05-15 23:34:41 +00:00
|
|
|
void pa_sample_clamp(pa_sample_format_t format, void *dst, size_t dstr, const void *src, size_t sstr, unsigned n);
|
|
|
|
|
|
core: Refactor code to multiply s16 by volume
move code to function pa_mult_s16_volume() in sample-util.h
use 64 bit integers on 64 bit platforms (it's faster)
on i5, 2.5GHz (64-bit)
Running suite(s): Mult-s16
32 bit mult: 1272300 usec (avg: 12723, min = 12533, max = 18749, stddev = 620.48).
64 bit mult: 852241 usec (avg: 8522.41, min = 8420, max = 9148, stddev = 109.388).
100%: Checks: 1, Failures: 0, Errors: 0
on Pentium D, 3.4GHz (32-bit)
Running suite(s): Mult-s16
32 bit mult: 2228504 usec (avg: 22285, min = 18775, max = 29648, stddev = 3865.59).
64 bit mult: 5546861 usec (avg: 55468.6, min = 55028, max = 64924, stddev = 978.981).
100%: Checks: 1, Failures: 0, Errors: 0
on TI DM3730, Cortex-A8, 800MHz (32-bit)
Running suite(s): Mult-s16
32 bit mult: 23708900 usec (avg: 237089, min = 191864, max = 557312, stddev = 77503.6).
64 bit mult: 22190039 usec (avg: 221900, min = 177978, max = 480469, stddev = 68520.5).
100%: Checks: 1, Failures: 0, Errors: 0
there is a test program called mult-s16-test which checks that the functions compute the
same results, and compares runtime
Signed-off-by: Peter Meerwald <pmeerw@pmeerw.net>
2013-02-13 17:27:05 +01:00
|
|
|
static inline int32_t pa_mult_s16_volume(int16_t v, int32_t cv) {
|
2016-01-15 16:09:49 +05:30
|
|
|
#ifdef HAVE_FAST_64BIT_OPERATIONS
|
core: Refactor code to multiply s16 by volume
move code to function pa_mult_s16_volume() in sample-util.h
use 64 bit integers on 64 bit platforms (it's faster)
on i5, 2.5GHz (64-bit)
Running suite(s): Mult-s16
32 bit mult: 1272300 usec (avg: 12723, min = 12533, max = 18749, stddev = 620.48).
64 bit mult: 852241 usec (avg: 8522.41, min = 8420, max = 9148, stddev = 109.388).
100%: Checks: 1, Failures: 0, Errors: 0
on Pentium D, 3.4GHz (32-bit)
Running suite(s): Mult-s16
32 bit mult: 2228504 usec (avg: 22285, min = 18775, max = 29648, stddev = 3865.59).
64 bit mult: 5546861 usec (avg: 55468.6, min = 55028, max = 64924, stddev = 978.981).
100%: Checks: 1, Failures: 0, Errors: 0
on TI DM3730, Cortex-A8, 800MHz (32-bit)
Running suite(s): Mult-s16
32 bit mult: 23708900 usec (avg: 237089, min = 191864, max = 557312, stddev = 77503.6).
64 bit mult: 22190039 usec (avg: 221900, min = 177978, max = 480469, stddev = 68520.5).
100%: Checks: 1, Failures: 0, Errors: 0
there is a test program called mult-s16-test which checks that the functions compute the
same results, and compares runtime
Signed-off-by: Peter Meerwald <pmeerw@pmeerw.net>
2013-02-13 17:27:05 +01:00
|
|
|
/* Multiply with 64 bit integers on 64 bit platforms */
|
|
|
|
|
return (v * (int64_t) cv) >> 16;
|
|
|
|
|
#else
|
|
|
|
|
/* Multiplying the 32 bit volume factor with the
|
|
|
|
|
* 16 bit sample might result in an 48 bit value. We
|
|
|
|
|
* want to do without 64 bit integers and hence do
|
|
|
|
|
* the multiplication independently for the HI and
|
|
|
|
|
* LO part of the volume. */
|
|
|
|
|
|
|
|
|
|
int32_t hi = cv >> 16;
|
|
|
|
|
int32_t lo = cv & 0xFFFF;
|
|
|
|
|
return ((v * lo) >> 16) + (v * hi);
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2008-08-28 17:53:01 +02:00
|
|
|
pa_usec_t pa_bytes_to_usec_round_up(uint64_t length, const pa_sample_spec *spec);
|
|
|
|
|
size_t pa_usec_to_bytes_round_up(pa_usec_t t, const pa_sample_spec *spec);
|
|
|
|
|
|
2009-01-08 21:12:03 +01:00
|
|
|
void pa_memchunk_dump_to_file(pa_memchunk *c, const char *fn);
|
|
|
|
|
|
2009-01-12 19:48:44 +01:00
|
|
|
void pa_memchunk_sine(pa_memchunk *c, pa_mempool *pool, unsigned rate, unsigned freq);
|
|
|
|
|
|
2012-07-09 18:01:32 +02:00
|
|
|
typedef void (*pa_do_volume_func_t) (void *samples, const void *volumes, unsigned channels, unsigned length);
|
2009-08-11 17:10:44 +02:00
|
|
|
|
|
|
|
|
pa_do_volume_func_t pa_get_volume_func(pa_sample_format_t f);
|
|
|
|
|
void pa_set_volume_func(pa_sample_format_t f, pa_do_volume_func_t func);
|
|
|
|
|
|
2009-09-09 04:27:16 +02:00
|
|
|
size_t pa_convert_size(size_t size, const pa_sample_spec *from, const pa_sample_spec *to);
|
|
|
|
|
|
2009-06-17 03:08:34 +02:00
|
|
|
#define PA_CHANNEL_POSITION_MASK_LEFT \
|
|
|
|
|
(PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_FRONT_LEFT) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_REAR_LEFT) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_FRONT_LEFT_OF_CENTER) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_SIDE_LEFT) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_TOP_FRONT_LEFT) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_TOP_REAR_LEFT)) \
|
|
|
|
|
|
|
|
|
|
#define PA_CHANNEL_POSITION_MASK_RIGHT \
|
|
|
|
|
(PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_FRONT_RIGHT) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_REAR_RIGHT) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_FRONT_RIGHT_OF_CENTER) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_SIDE_RIGHT) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_TOP_FRONT_RIGHT) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_TOP_REAR_RIGHT))
|
|
|
|
|
|
|
|
|
|
#define PA_CHANNEL_POSITION_MASK_CENTER \
|
|
|
|
|
(PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_FRONT_CENTER) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_REAR_CENTER) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_TOP_CENTER) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_TOP_FRONT_CENTER) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_TOP_REAR_CENTER))
|
|
|
|
|
|
|
|
|
|
#define PA_CHANNEL_POSITION_MASK_FRONT \
|
|
|
|
|
(PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_FRONT_LEFT) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_FRONT_RIGHT) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_FRONT_CENTER) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_FRONT_LEFT_OF_CENTER) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_FRONT_RIGHT_OF_CENTER) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_TOP_FRONT_LEFT) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_TOP_FRONT_RIGHT) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_TOP_FRONT_CENTER))
|
|
|
|
|
|
|
|
|
|
#define PA_CHANNEL_POSITION_MASK_REAR \
|
|
|
|
|
(PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_REAR_LEFT) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_REAR_RIGHT) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_REAR_CENTER) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_TOP_REAR_LEFT) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_TOP_REAR_RIGHT) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_TOP_REAR_CENTER))
|
|
|
|
|
|
2015-10-27 09:54:39 +01:00
|
|
|
#define PA_CHANNEL_POSITION_MASK_LFE \
|
|
|
|
|
PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_LFE)
|
|
|
|
|
|
|
|
|
|
#define PA_CHANNEL_POSITION_MASK_HFE \
|
|
|
|
|
(PA_CHANNEL_POSITION_MASK_REAR | PA_CHANNEL_POSITION_MASK_FRONT \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK_LEFT | PA_CHANNEL_POSITION_MASK_RIGHT \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK_CENTER)
|
|
|
|
|
|
2009-06-17 03:08:34 +02:00
|
|
|
#define PA_CHANNEL_POSITION_MASK_SIDE_OR_TOP_CENTER \
|
|
|
|
|
(PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_SIDE_LEFT) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_SIDE_RIGHT) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_TOP_CENTER))
|
|
|
|
|
|
|
|
|
|
#define PA_CHANNEL_POSITION_MASK_TOP \
|
|
|
|
|
(PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_TOP_CENTER) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_TOP_FRONT_LEFT) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_TOP_FRONT_RIGHT) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_TOP_FRONT_CENTER) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_TOP_REAR_LEFT) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_TOP_REAR_RIGHT) \
|
|
|
|
|
| PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_TOP_REAR_CENTER))
|
|
|
|
|
|
|
|
|
|
#define PA_CHANNEL_POSITION_MASK_ALL \
|
|
|
|
|
((pa_channel_position_mask_t) (PA_CHANNEL_POSITION_MASK(PA_CHANNEL_POSITION_MAX)-1))
|
|
|
|
|
|
2004-06-23 23:17:30 +00:00
|
|
|
#endif
|