1 /***
2 This file is part of PulseAudio.
3
4 Copyright 2004-2006 Lennart Poettering
5
6 PulseAudio is free software; you can redistribute it and/or modify
7 it under the terms of the GNU Lesser General Public License as published
8 by the Free Software Foundation; either version 2.1 of the License,
9 or (at your option) any later version.
10
11 PulseAudio is distributed in the hope that it will be useful, but
12 WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 General Public License for more details.
15
16 You should have received a copy of the GNU Lesser General Public License
17 along with PulseAudio; if not, see <http://www.gnu.org/licenses/>.
18 ***/
19
20 #ifdef HAVE_CONFIG_H
21 #include <config.h>
22 #endif
23
24 #include <speex/speex_resampler.h>
25 #include <math.h>
26
27 #include <pulsecore/once.h>
28 #include <pulsecore/resampler.h>
29
pa_speex_is_fixed_point(void)30 bool pa_speex_is_fixed_point(void) {
31 static bool result = false;
32 PA_ONCE_BEGIN {
33 float f_out = -1.0f, f_in = 1.0f;
34 spx_uint32_t in_len = 1, out_len = 1;
35 SpeexResamplerState *s;
36
37 pa_assert_se(s = speex_resampler_init(1, 1, 1,
38 SPEEX_RESAMPLER_QUALITY_MIN, NULL));
39
40 /* feed one sample that is too soft for fixed-point speex */
41 pa_assert_se(speex_resampler_process_float(s, 0, &f_in, &in_len,
42 &f_out, &out_len) == RESAMPLER_ERR_SUCCESS);
43
44 /* expecting sample has been processed, one sample output */
45 pa_assert_se(in_len == 1 && out_len == 1);
46
47 /* speex compiled with --enable-fixed-point will output 0.0 due to insufficient precision */
48 if (fabsf(f_out) < 0.00001f)
49 result = true;
50
51 speex_resampler_destroy(s);
52 } PA_ONCE_END;
53 return result;
54 }
55
56
speex_resample_float(pa_resampler * r,const pa_memchunk * input,unsigned in_n_frames,pa_memchunk * output,unsigned * out_n_frames)57 static unsigned speex_resample_float(pa_resampler *r, const pa_memchunk *input, unsigned in_n_frames, pa_memchunk *output, unsigned *out_n_frames) {
58 float *in, *out;
59 uint32_t inf = in_n_frames, outf = *out_n_frames;
60 SpeexResamplerState *state;
61
62 pa_assert(r);
63 pa_assert(input);
64 pa_assert(output);
65 pa_assert(out_n_frames);
66
67 state = r->impl.data;
68
69 in = pa_memblock_acquire_chunk(input);
70 out = pa_memblock_acquire_chunk(output);
71
72 /* Strictly speaking, speex resampler expects its input
73 * to be normalized to the [-32768.0 .. 32767.0] range.
74 * This matters if speex has been compiled with --enable-fixed-point,
75 * because such speex will round the samples to the nearest
76 * integer. speex with --enable-fixed-point is therefore incompatible
77 * with PulseAudio's floating-point sample range [-1 .. 1]. speex
78 * without --enable-fixed-point works fine with this range.
79 * Care has been taken to call speex_resample_float() only
80 * for speex compiled without --enable-fixed-point.
81 */
82 pa_assert_se(speex_resampler_process_interleaved_float(state, in, &inf, out, &outf) == 0);
83
84 pa_memblock_release(input->memblock);
85 pa_memblock_release(output->memblock);
86
87 pa_assert(inf == in_n_frames);
88 *out_n_frames = outf;
89
90 return 0;
91 }
92
speex_resample_int(pa_resampler * r,const pa_memchunk * input,unsigned in_n_frames,pa_memchunk * output,unsigned * out_n_frames)93 static unsigned speex_resample_int(pa_resampler *r, const pa_memchunk *input, unsigned in_n_frames, pa_memchunk *output, unsigned *out_n_frames) {
94 int16_t *in, *out;
95 uint32_t inf = in_n_frames, outf = *out_n_frames;
96 SpeexResamplerState *state;
97
98 pa_assert(r);
99 pa_assert(input);
100 pa_assert(output);
101 pa_assert(out_n_frames);
102
103 state = r->impl.data;
104
105 in = pa_memblock_acquire_chunk(input);
106 out = pa_memblock_acquire_chunk(output);
107
108 pa_assert_se(speex_resampler_process_interleaved_int(state, in, &inf, out, &outf) == 0);
109
110 pa_memblock_release(input->memblock);
111 pa_memblock_release(output->memblock);
112
113 pa_assert(inf == in_n_frames);
114 *out_n_frames = outf;
115
116 return 0;
117 }
118
speex_update_rates(pa_resampler * r)119 static void speex_update_rates(pa_resampler *r) {
120 SpeexResamplerState *state;
121 pa_assert(r);
122
123 state = r->impl.data;
124
125 pa_assert_se(speex_resampler_set_rate(state, r->i_ss.rate, r->o_ss.rate) == 0);
126 }
127
speex_reset(pa_resampler * r)128 static void speex_reset(pa_resampler *r) {
129 SpeexResamplerState *state;
130 pa_assert(r);
131
132 state = r->impl.data;
133
134 pa_assert_se(speex_resampler_reset_mem(state) == 0);
135 speex_resampler_skip_zeros(state);
136 }
137
speex_free(pa_resampler * r)138 static void speex_free(pa_resampler *r) {
139 SpeexResamplerState *state;
140 pa_assert(r);
141
142 state = r->impl.data;
143 if (!state)
144 return;
145
146 speex_resampler_destroy(state);
147 }
148
pa_resampler_speex_init(pa_resampler * r)149 int pa_resampler_speex_init(pa_resampler *r) {
150 int q, err;
151 SpeexResamplerState *state;
152
153 pa_assert(r);
154
155 r->impl.free = speex_free;
156 r->impl.update_rates = speex_update_rates;
157 r->impl.reset = speex_reset;
158
159 if (r->method >= PA_RESAMPLER_SPEEX_FIXED_BASE && r->method <= PA_RESAMPLER_SPEEX_FIXED_MAX) {
160
161 q = r->method - PA_RESAMPLER_SPEEX_FIXED_BASE;
162 r->impl.resample = speex_resample_int;
163
164 } else {
165 pa_assert(r->method >= PA_RESAMPLER_SPEEX_FLOAT_BASE && r->method <= PA_RESAMPLER_SPEEX_FLOAT_MAX);
166
167 q = r->method - PA_RESAMPLER_SPEEX_FLOAT_BASE;
168 r->impl.resample = speex_resample_float;
169 }
170
171 pa_log_info("Choosing speex quality setting %i.", q);
172
173 if (!(state = speex_resampler_init(r->work_channels, r->i_ss.rate, r->o_ss.rate, q, &err)))
174 return -1;
175
176 speex_resampler_skip_zeros(state);
177
178 r->impl.data = state;
179
180 return 0;
181 }
182