blob: 0b82657311ffae211d96014b81b53ca668b61707 [file] [log] [blame]
Eric Laurentb3184d72011-08-17 18:36:09 -07001/*
2** Copyright 2011, The Android Open-Source Project
3**
4** Licensed under the Apache License, Version 2.0 (the "License");
5** you may not use this file except in compliance with the License.
6** You may obtain a copy of the License at
7**
8** http://www.apache.org/licenses/LICENSE-2.0
9**
10** Unless required by applicable law or agreed to in writing, software
11** distributed under the License is distributed on an "AS IS" BASIS,
12** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13** See the License for the specific language governing permissions and
14** limitations under the License.
15*/
16
17//#define LOG_NDEBUG 0
18#define LOG_TAG "echo_reference"
19
20#include <errno.h>
21#include <stdlib.h>
22#include <pthread.h>
23#include <cutils/log.h>
24#include <system/audio.h>
25#include <audio_utils/resampler.h>
26#include <audio_utils/echo_reference.h>
27
28// echo reference state: bit field indicating if read, write or both are active.
29enum state {
30 ECHOREF_IDLE = 0x00, // idle
31 ECHOREF_READING = 0x01, // reading is active
32 ECHOREF_WRITING = 0x02 // writing is active
33};
34
35struct echo_reference {
36 struct echo_reference_itfe itfe;
37 int status; // init status
38 uint32_t state; // active state: reading, writing or both
39 audio_format_t rd_format; // read sample format
40 uint32_t rd_channel_count; // read number of channels
41 uint32_t rd_sampling_rate; // read sampling rate in Hz
42 size_t rd_frame_size; // read frame size (bytes per sample)
43 audio_format_t wr_format; // write sample format
44 uint32_t wr_channel_count; // write number of channels
45 uint32_t wr_sampling_rate; // write sampling rate in Hz
46 size_t wr_frame_size; // write frame size (bytes per sample)
47 void *buffer; // main buffer
48 size_t buf_size; // main buffer size in frames
49 size_t frames_in; // number of frames in main buffer
50 void *wr_buf; // buffer for input conversions
51 size_t wr_buf_size; // size of conversion buffer in frames
52 size_t wr_frames_in; // number of frames in conversion buffer
Eric Laurentf06cdb52012-03-30 09:39:46 -070053 size_t wr_curr_frame_size; // number of frames given to current write() function
Eric Laurentb3184d72011-08-17 18:36:09 -070054 void *wr_src_buf; // resampler input buf (either wr_buf or buffer used by write())
55 struct timespec wr_render_time; // latest render time indicated by write()
56 // default ALSA gettimeofday() format
57 int32_t playback_delay; // playback buffer delay indicated by last write()
Eric Laurentf06cdb52012-03-30 09:39:46 -070058 int16_t prev_delta_sign; // sign of previous delay difference:
59 // 1: positive, -1: negative, 0: unknown
60 uint16_t delta_count; // number of consecutive delay differences with same sign
Eric Laurentb3184d72011-08-17 18:36:09 -070061 pthread_mutex_t lock; // mutex protecting read/write concurrency
62 pthread_cond_t cond; // condition signaled when data is ready to read
Eric Laurent33e8f782012-02-06 13:57:31 -080063 struct resampler_itfe *resampler; // input resampler
Eric Laurentb3184d72011-08-17 18:36:09 -070064 struct resampler_buffer_provider provider; // resampler buffer provider
65};
66
67
Eric Laurentb3184d72011-08-17 18:36:09 -070068int echo_reference_get_next_buffer(struct resampler_buffer_provider *buffer_provider,
69 struct resampler_buffer* buffer)
70{
71 struct echo_reference *er;
72
73 if (buffer_provider == NULL) {
74 return -EINVAL;
75 }
76
Eric Laurentc9d5dff2011-08-31 19:18:26 -070077 er = (struct echo_reference *)((char *)buffer_provider -
78 offsetof(struct echo_reference, provider));
Eric Laurentb3184d72011-08-17 18:36:09 -070079
80 if (er->wr_src_buf == NULL || er->wr_frames_in == 0) {
81 buffer->raw = NULL;
82 buffer->frame_count = 0;
83 return -ENODATA;
84 }
85
Glenn Kasten548d9cf2012-11-02 07:59:08 -070086 buffer->frame_count = (buffer->frame_count > er->wr_frames_in) ?
87 er->wr_frames_in : buffer->frame_count;
Eric Laurentb3184d72011-08-17 18:36:09 -070088 // this is er->rd_channel_count here as we resample after stereo to mono conversion if any
Glenn Kasten548d9cf2012-11-02 07:59:08 -070089 buffer->i16 = (int16_t *)er->wr_src_buf + (er->wr_curr_frame_size - er->wr_frames_in) *
90 er->rd_channel_count;
Eric Laurentb3184d72011-08-17 18:36:09 -070091
92 return 0;
93}
94
95void echo_reference_release_buffer(struct resampler_buffer_provider *buffer_provider,
96 struct resampler_buffer* buffer)
97{
98 struct echo_reference *er;
99
100 if (buffer_provider == NULL) {
101 return;
102 }
103
Eric Laurentc9d5dff2011-08-31 19:18:26 -0700104 er = (struct echo_reference *)((char *)buffer_provider -
105 offsetof(struct echo_reference, provider));
Eric Laurentb3184d72011-08-17 18:36:09 -0700106
107 er->wr_frames_in -= buffer->frame_count;
108}
109
110static void echo_reference_reset_l(struct echo_reference *er)
111{
Steve Block473d4a52011-10-26 11:12:08 +0100112 ALOGV("echo_reference_reset_l()");
Eric Laurentb3184d72011-08-17 18:36:09 -0700113 free(er->buffer);
114 er->buffer = NULL;
115 er->buf_size = 0;
116 er->frames_in = 0;
117 free(er->wr_buf);
118 er->wr_buf = NULL;
119 er->wr_buf_size = 0;
120 er->wr_render_time.tv_sec = 0;
121 er->wr_render_time.tv_nsec = 0;
Eric Laurentf06cdb52012-03-30 09:39:46 -0700122 er->delta_count = 0;
123 er->prev_delta_sign = 0;
Eric Laurentb3184d72011-08-17 18:36:09 -0700124}
125
Eric Laurent33e8f782012-02-06 13:57:31 -0800126/* additional space in resampler buffer allowing for extra samples to be returned
127 * by speex resampler when sample rates ratio is not an integer.
128 */
129#define RESAMPLER_HEADROOM_SAMPLES 10
130
Eric Laurentb3184d72011-08-17 18:36:09 -0700131static int echo_reference_write(struct echo_reference_itfe *echo_reference,
132 struct echo_reference_buffer *buffer)
133{
134 struct echo_reference *er = (struct echo_reference *)echo_reference;
135 int status = 0;
136
137 if (er == NULL) {
138 return -EINVAL;
139 }
140
141 pthread_mutex_lock(&er->lock);
142
143 if (buffer == NULL) {
Steve Block473d4a52011-10-26 11:12:08 +0100144 ALOGV("echo_reference_write() stop write");
Eric Laurentb3184d72011-08-17 18:36:09 -0700145 er->state &= ~ECHOREF_WRITING;
146 echo_reference_reset_l(er);
147 goto exit;
148 }
149
Steve Block473d4a52011-10-26 11:12:08 +0100150 ALOGV("echo_reference_write() START trying to write %d frames", buffer->frame_count);
151 ALOGV("echo_reference_write() playbackTimestamp:[%d].[%d], er->playback_delay:[%d]",
Eric Laurentb3184d72011-08-17 18:36:09 -0700152 (int)buffer->time_stamp.tv_sec,
153 (int)buffer->time_stamp.tv_nsec, er->playback_delay);
154
Steve Block473d4a52011-10-26 11:12:08 +0100155 //ALOGV("echo_reference_write() %d frames", buffer->frame_count);
Eric Laurentb3184d72011-08-17 18:36:09 -0700156 // discard writes until a valid time stamp is provided.
157
158 if ((buffer->time_stamp.tv_sec == 0) && (buffer->time_stamp.tv_nsec == 0) &&
159 (er->wr_render_time.tv_sec == 0) && (er->wr_render_time.tv_nsec == 0)) {
160 goto exit;
161 }
162
163 if ((er->state & ECHOREF_WRITING) == 0) {
Steve Block473d4a52011-10-26 11:12:08 +0100164 ALOGV("echo_reference_write() start write");
Eric Laurent33e8f782012-02-06 13:57:31 -0800165 if (er->resampler != NULL) {
166 er->resampler->reset(er->resampler);
Eric Laurentb3184d72011-08-17 18:36:09 -0700167 }
168 er->state |= ECHOREF_WRITING;
169 }
170
171 if ((er->state & ECHOREF_READING) == 0) {
172 goto exit;
173 }
174
175 er->wr_render_time.tv_sec = buffer->time_stamp.tv_sec;
176 er->wr_render_time.tv_nsec = buffer->time_stamp.tv_nsec;
177
178 er->playback_delay = buffer->delay_ns;
179
Eric Laurentf06cdb52012-03-30 09:39:46 -0700180 // this will be used in the get_next_buffer, to support variable input buffer sizes
181 er->wr_curr_frame_size = buffer->frame_count;
182
Eric Laurentb3184d72011-08-17 18:36:09 -0700183 void *srcBuf;
184 size_t inFrames;
185 // do stereo to mono and down sampling if necessary
186 if (er->rd_channel_count != er->wr_channel_count ||
187 er->rd_sampling_rate != er->wr_sampling_rate) {
Eric Laurentc164cb82012-02-17 13:40:29 -0800188 size_t wrBufSize = buffer->frame_count;
Eric Laurent33e8f782012-02-06 13:57:31 -0800189
Eric Laurentc164cb82012-02-17 13:40:29 -0800190 inFrames = buffer->frame_count;
191
192 if (er->rd_sampling_rate != er->wr_sampling_rate) {
193 inFrames = (buffer->frame_count * er->rd_sampling_rate) / er->wr_sampling_rate +
194 RESAMPLER_HEADROOM_SAMPLES;
195 // wr_buf is not only used as resampler output but also for stereo to mono conversion
196 // output so buffer size is driven by both write and read sample rates
197 if (inFrames > wrBufSize) {
198 wrBufSize = inFrames;
199 }
200 }
201
202 if (er->wr_buf_size < wrBufSize) {
203 ALOGV("echo_reference_write() increasing write buffer size from %d to %d",
204 er->wr_buf_size, wrBufSize);
205 er->wr_buf_size = wrBufSize;
Eric Laurentb3184d72011-08-17 18:36:09 -0700206 er->wr_buf = realloc(er->wr_buf, er->wr_buf_size * er->rd_frame_size);
207 }
208
Eric Laurentb3184d72011-08-17 18:36:09 -0700209 if (er->rd_channel_count != er->wr_channel_count) {
210 // must be stereo to mono
211 int16_t *src16 = (int16_t *)buffer->raw;
212 int16_t *dst16 = (int16_t *)er->wr_buf;
213 size_t frames = buffer->frame_count;
214 while (frames--) {
215 *dst16++ = (int16_t)(((int32_t)*src16 + (int32_t)*(src16 + 1)) >> 1);
216 src16 += 2;
217 }
218 }
219 if (er->wr_sampling_rate != er->rd_sampling_rate) {
Eric Laurent33e8f782012-02-06 13:57:31 -0800220 if (er->resampler == NULL) {
Eric Laurentb3184d72011-08-17 18:36:09 -0700221 int rc;
Steve Block473d4a52011-10-26 11:12:08 +0100222 ALOGV("echo_reference_write() new ReSampler(%d, %d)",
Eric Laurentb3184d72011-08-17 18:36:09 -0700223 er->wr_sampling_rate, er->rd_sampling_rate);
224 er->provider.get_next_buffer = echo_reference_get_next_buffer;
225 er->provider.release_buffer = echo_reference_release_buffer;
226 rc = create_resampler(er->wr_sampling_rate,
227 er->rd_sampling_rate,
228 er->rd_channel_count,
Eric Laurentf06cdb52012-03-30 09:39:46 -0700229 RESAMPLER_QUALITY_DEFAULT,
Eric Laurentb3184d72011-08-17 18:36:09 -0700230 &er->provider,
Eric Laurent33e8f782012-02-06 13:57:31 -0800231 &er->resampler);
Eric Laurentb3184d72011-08-17 18:36:09 -0700232 if (rc != 0) {
Eric Laurent33e8f782012-02-06 13:57:31 -0800233 er->resampler = NULL;
Steve Block473d4a52011-10-26 11:12:08 +0100234 ALOGV("echo_reference_write() failure to create resampler %d", rc);
Eric Laurentb3184d72011-08-17 18:36:09 -0700235 status = -ENODEV;
236 goto exit;
237 }
238 }
Glenn Kasten548d9cf2012-11-02 07:59:08 -0700239 // er->wr_src_buf and er->wr_frames_in are used by getNexBuffer() called by the
240 // resampler to get new frames
Eric Laurentb3184d72011-08-17 18:36:09 -0700241 if (er->rd_channel_count != er->wr_channel_count) {
242 er->wr_src_buf = er->wr_buf;
243 } else {
244 er->wr_src_buf = buffer->raw;
245 }
246 er->wr_frames_in = buffer->frame_count;
247 // inFrames is always more than we need here to get frames remaining from previous runs
248 // inFrames is updated by resample() with the number of frames produced
Steve Block473d4a52011-10-26 11:12:08 +0100249 ALOGV("echo_reference_write() ReSampling(%d, %d)",
Eric Laurentb3184d72011-08-17 18:36:09 -0700250 er->wr_sampling_rate, er->rd_sampling_rate);
Eric Laurent33e8f782012-02-06 13:57:31 -0800251 er->resampler->resample_from_provider(er->resampler,
Eric Laurentb3184d72011-08-17 18:36:09 -0700252 (int16_t *)er->wr_buf, &inFrames);
Eric Laurentf06cdb52012-03-30 09:39:46 -0700253 ALOGV_IF(er->wr_frames_in != 0,
Eric Laurentb3184d72011-08-17 18:36:09 -0700254 "echo_reference_write() er->wr_frames_in not 0 (%d) after resampler",
255 er->wr_frames_in);
256 }
257 srcBuf = er->wr_buf;
258 } else {
259 inFrames = buffer->frame_count;
260 srcBuf = buffer->raw;
261 }
262
263 if (er->frames_in + inFrames > er->buf_size) {
Steve Block473d4a52011-10-26 11:12:08 +0100264 ALOGV("echo_reference_write() increasing buffer size from %d to %d",
Eric Laurentb3184d72011-08-17 18:36:09 -0700265 er->buf_size, er->frames_in + inFrames);
266 er->buf_size = er->frames_in + inFrames;
267 er->buffer = realloc(er->buffer, er->buf_size * er->rd_frame_size);
268 }
269 memcpy((char *)er->buffer + er->frames_in * er->rd_frame_size,
270 srcBuf,
271 inFrames * er->rd_frame_size);
272 er->frames_in += inFrames;
273
Eric Laurentf06cdb52012-03-30 09:39:46 -0700274 ALOGV("echo_reference_write() frames written:[%d], frames total:[%d] buffer size:[%d]\n"
275 " er->wr_render_time:[%d].[%d], er->playback_delay:[%d]",
276 inFrames, er->frames_in, er->buf_size,
277 (int)er->wr_render_time.tv_sec, (int)er->wr_render_time.tv_nsec, er->playback_delay);
Eric Laurentb3184d72011-08-17 18:36:09 -0700278
279 pthread_cond_signal(&er->cond);
280exit:
281 pthread_mutex_unlock(&er->lock);
Steve Block473d4a52011-10-26 11:12:08 +0100282 ALOGV("echo_reference_write() END");
Eric Laurentb3184d72011-08-17 18:36:09 -0700283 return status;
284}
285
Eric Laurentf06cdb52012-03-30 09:39:46 -0700286// delay jump threshold to update ref buffer: 6 samples at 8kHz in nsecs
287#define MIN_DELAY_DELTA_NS (375000*2)
288// number of consecutive delta with same sign between expected and actual delay before adjusting
289// the buffer
290#define MIN_DELTA_NUM 4
Eric Laurentb3184d72011-08-17 18:36:09 -0700291
292
293static int echo_reference_read(struct echo_reference_itfe *echo_reference,
294 struct echo_reference_buffer *buffer)
295{
296 struct echo_reference *er = (struct echo_reference *)echo_reference;
297
298 if (er == NULL) {
299 return -EINVAL;
300 }
301
302 pthread_mutex_lock(&er->lock);
303
304 if (buffer == NULL) {
Eric Laurentf06cdb52012-03-30 09:39:46 -0700305 ALOGV("echo_reference_read() stop read");
Eric Laurentb3184d72011-08-17 18:36:09 -0700306 er->state &= ~ECHOREF_READING;
307 goto exit;
308 }
309
Eric Laurentf06cdb52012-03-30 09:39:46 -0700310 ALOGV("echo_reference_read() START, delayCapture:[%d], "
311 "er->frames_in:[%d],buffer->frame_count:[%d]",
Eric Laurentb3184d72011-08-17 18:36:09 -0700312 buffer->delay_ns, er->frames_in, buffer->frame_count);
313
314 if ((er->state & ECHOREF_READING) == 0) {
Eric Laurentf06cdb52012-03-30 09:39:46 -0700315 ALOGV("echo_reference_read() start read");
Eric Laurentb3184d72011-08-17 18:36:09 -0700316 echo_reference_reset_l(er);
317 er->state |= ECHOREF_READING;
318 }
319
320 if ((er->state & ECHOREF_WRITING) == 0) {
321 memset(buffer->raw, 0, er->rd_frame_size * buffer->frame_count);
322 buffer->delay_ns = 0;
323 goto exit;
324 }
325
Eric Laurentf06cdb52012-03-30 09:39:46 -0700326// ALOGV("echo_reference_read() %d frames", buffer->frame_count);
Eric Laurentb3184d72011-08-17 18:36:09 -0700327
328 // allow some time for new frames to arrive if not enough frames are ready for read
329 if (er->frames_in < buffer->frame_count) {
330 uint32_t timeoutMs = (uint32_t)((1000 * buffer->frame_count) / er->rd_sampling_rate / 2);
Colin Cross31c4e5c2014-01-24 18:20:15 -0800331 struct timespec ts = {0, 0};
Eric Laurentb3184d72011-08-17 18:36:09 -0700332
Colin Cross31c4e5c2014-01-24 18:20:15 -0800333#ifndef HAVE_PTHREAD_COND_TIMEDWAIT_RELATIVE
334 clock_gettime(CLOCK_REALTIME, &ts);
335#endif
336
337 ts.tv_sec += timeoutMs/1000;
338 ts.tv_nsec += (timeoutMs%1000) * 1000000;
339 if (ts.tv_nsec >= 1000000000) {
340 ts.tv_nsec -= 1000000000;
341 ts.tv_sec += 1;
342 }
343
344#ifdef HAVE_PTHREAD_COND_TIMEDWAIT_RELATIVE
Eric Laurentb3184d72011-08-17 18:36:09 -0700345 pthread_cond_timedwait_relative_np(&er->cond, &er->lock, &ts);
Colin Cross31c4e5c2014-01-24 18:20:15 -0800346#else
347 pthread_cond_timedwait(&er->cond, &er->lock, &ts);
348#endif
Eric Laurentb3184d72011-08-17 18:36:09 -0700349
Eric Laurentf06cdb52012-03-30 09:39:46 -0700350 ALOGV_IF((er->frames_in < buffer->frame_count),
351 "echo_reference_read() waited %d ms but still not enough frames"\
Eric Laurentb3184d72011-08-17 18:36:09 -0700352 " er->frames_in: %d, buffer->frame_count = %d",
Eric Laurentf06cdb52012-03-30 09:39:46 -0700353 timeoutMs, er->frames_in, buffer->frame_count);
Eric Laurentb3184d72011-08-17 18:36:09 -0700354 }
355
356 int64_t timeDiff;
357 struct timespec tmp;
358
359 if ((er->wr_render_time.tv_sec == 0 && er->wr_render_time.tv_nsec == 0) ||
360 (buffer->time_stamp.tv_sec == 0 && buffer->time_stamp.tv_nsec == 0)) {
Eric Laurentf06cdb52012-03-30 09:39:46 -0700361 ALOGV("echo_reference_read(): NEW:timestamp is zero---------setting timeDiff = 0, "\
Eric Laurentb3184d72011-08-17 18:36:09 -0700362 "not updating delay this time");
363 timeDiff = 0;
364 } else {
365 if (buffer->time_stamp.tv_nsec < er->wr_render_time.tv_nsec) {
366 tmp.tv_sec = buffer->time_stamp.tv_sec - er->wr_render_time.tv_sec - 1;
367 tmp.tv_nsec = 1000000000 + buffer->time_stamp.tv_nsec - er->wr_render_time.tv_nsec;
368 } else {
369 tmp.tv_sec = buffer->time_stamp.tv_sec - er->wr_render_time.tv_sec;
370 tmp.tv_nsec = buffer->time_stamp.tv_nsec - er->wr_render_time.tv_nsec;
371 }
372 timeDiff = (((int64_t)tmp.tv_sec * 1000000000 + tmp.tv_nsec));
373
374 int64_t expectedDelayNs = er->playback_delay + buffer->delay_ns - timeDiff;
375
Eric Laurentf06cdb52012-03-30 09:39:46 -0700376 if (er->resampler != NULL) {
377 // Resampler already compensates part of the delay
378 int32_t rsmp_delay = er->resampler->delay_ns(er->resampler);
379 expectedDelayNs -= rsmp_delay;
380 }
381
382 ALOGV("echo_reference_read(): expectedDelayNs[%lld] = "
383 "er->playback_delay[%d] + delayCapture[%d] - timeDiff[%lld]",
384 expectedDelayNs, er->playback_delay, buffer->delay_ns, timeDiff);
Eric Laurentb3184d72011-08-17 18:36:09 -0700385
386 if (expectedDelayNs > 0) {
387 int64_t delayNs = ((int64_t)er->frames_in * 1000000000) / er->rd_sampling_rate;
388
Eric Laurentf06cdb52012-03-30 09:39:46 -0700389 int64_t deltaNs = delayNs - expectedDelayNs;
Eric Laurentb3184d72011-08-17 18:36:09 -0700390
Glenn Kasten548d9cf2012-11-02 07:59:08 -0700391 ALOGV("echo_reference_read(): EchoPathDelayDeviation between reference and DMA [%lld]",
392 deltaNs);
Eric Laurentf06cdb52012-03-30 09:39:46 -0700393 if (abs(deltaNs) >= MIN_DELAY_DELTA_NS) {
394 // smooth the variation and update the reference buffer only
395 // if a deviation in the same direction is observed for more than MIN_DELTA_NUM
396 // consecutive reads.
397 int16_t delay_sign = (deltaNs >= 0) ? 1 : -1;
398 if (delay_sign == er->prev_delta_sign) {
399 er->delta_count++;
Eric Laurentb3184d72011-08-17 18:36:09 -0700400 } else {
Eric Laurentf06cdb52012-03-30 09:39:46 -0700401 er->delta_count = 1;
402 }
403 er->prev_delta_sign = delay_sign;
Eric Laurentb3184d72011-08-17 18:36:09 -0700404
Eric Laurentf06cdb52012-03-30 09:39:46 -0700405 if (er->delta_count > MIN_DELTA_NUM) {
406 size_t previousFrameIn = er->frames_in;
407 er->frames_in = (size_t)((expectedDelayNs * er->rd_sampling_rate)/1000000000);
408 int offset = er->frames_in - previousFrameIn;
Eric Laurentb3184d72011-08-17 18:36:09 -0700409
Eric Laurentf06cdb52012-03-30 09:39:46 -0700410 ALOGV("echo_reference_read(): deltaNs ENOUGH and %s: "
411 "er->frames_in: %d, previousFrameIn = %d",
412 delay_sign ? "positive" : "negative", er->frames_in, previousFrameIn);
413
414 if (deltaNs < 0) {
415 // Less data available in the reference buffer than expected
416 if (er->frames_in > er->buf_size) {
417 er->buf_size = er->frames_in;
418 er->buffer = realloc(er->buffer, er->buf_size * er->rd_frame_size);
419 ALOGV("echo_reference_read(): increasing buffer size to %d",
420 er->buf_size);
Eric Laurentb3184d72011-08-17 18:36:09 -0700421 }
Eric Laurentf06cdb52012-03-30 09:39:46 -0700422
423 if (offset > 0) {
424 memset((char *)er->buffer + previousFrameIn * er->rd_frame_size,
425 0, offset * er->rd_frame_size);
426 ALOGV("echo_reference_read(): pushing ref buffer by [%d]", offset);
427 }
Eric Laurentb3184d72011-08-17 18:36:09 -0700428 } else {
Eric Laurentf06cdb52012-03-30 09:39:46 -0700429 // More data available in the reference buffer than expected
430 offset = -offset;
Eric Laurentb3184d72011-08-17 18:36:09 -0700431 if (offset > 0) {
432 memcpy(er->buffer, (char *)er->buffer + (offset * er->rd_frame_size),
Eric Laurentf06cdb52012-03-30 09:39:46 -0700433 er->frames_in * er->rd_frame_size);
434 ALOGV("echo_reference_read(): shifting ref buffer by [%d]",
435 er->frames_in);
Eric Laurentb3184d72011-08-17 18:36:09 -0700436 }
437 }
Eric Laurentb3184d72011-08-17 18:36:09 -0700438 }
439 } else {
Eric Laurentf06cdb52012-03-30 09:39:46 -0700440 er->delta_count = 0;
441 er->prev_delta_sign = 0;
442 ALOGV("echo_reference_read(): Constant EchoPathDelay - difference "
443 "between reference and DMA %lld", deltaNs);
Eric Laurentb3184d72011-08-17 18:36:09 -0700444 }
445 } else {
Eric Laurentf06cdb52012-03-30 09:39:46 -0700446 ALOGV("echo_reference_read(): NEGATIVE expectedDelayNs[%lld] = "\
Eric Laurentb3184d72011-08-17 18:36:09 -0700447 "er->playback_delay[%d] + delayCapture[%d] - timeDiff[%lld]",
448 expectedDelayNs, er->playback_delay, buffer->delay_ns, timeDiff);
449 }
450 }
451
Eric Laurentf06cdb52012-03-30 09:39:46 -0700452 if (er->frames_in < buffer->frame_count) {
453 if (buffer->frame_count > er->buf_size) {
454 er->buf_size = buffer->frame_count;
455 er->buffer = realloc(er->buffer, er->buf_size * er->rd_frame_size);
456 ALOGV("echo_reference_read(): increasing buffer size to %d", er->buf_size);
457 }
458 // filling up the reference buffer with 0s to match the expected delay.
459 memset((char *)er->buffer + er->frames_in * er->rd_frame_size,
460 0, (buffer->frame_count - er->frames_in) * er->rd_frame_size);
461 er->frames_in = buffer->frame_count;
462 }
463
Eric Laurentb3184d72011-08-17 18:36:09 -0700464 memcpy(buffer->raw,
465 (char *)er->buffer,
466 buffer->frame_count * er->rd_frame_size);
467
468 er->frames_in -= buffer->frame_count;
469 memcpy(er->buffer,
470 (char *)er->buffer + buffer->frame_count * er->rd_frame_size,
471 er->frames_in * er->rd_frame_size);
472
473 // As the reference buffer is now time aligned to the microphone signal there is a zero delay
474 buffer->delay_ns = 0;
475
Eric Laurentf06cdb52012-03-30 09:39:46 -0700476 ALOGV("echo_reference_read() END %d frames, total frames in %d",
Eric Laurentb3184d72011-08-17 18:36:09 -0700477 buffer->frame_count, er->frames_in);
478
479 pthread_cond_signal(&er->cond);
480
481exit:
482 pthread_mutex_unlock(&er->lock);
483 return 0;
484}
485
486
487int create_echo_reference(audio_format_t rdFormat,
488 uint32_t rdChannelCount,
489 uint32_t rdSamplingRate,
490 audio_format_t wrFormat,
491 uint32_t wrChannelCount,
492 uint32_t wrSamplingRate,
493 struct echo_reference_itfe **echo_reference)
494{
495 struct echo_reference *er;
496
Steve Block473d4a52011-10-26 11:12:08 +0100497 ALOGV("create_echo_reference()");
Eric Laurentb3184d72011-08-17 18:36:09 -0700498
499 if (echo_reference == NULL) {
500 return -EINVAL;
501 }
502
503 *echo_reference = NULL;
504
505 if (rdFormat != AUDIO_FORMAT_PCM_16_BIT ||
506 rdFormat != wrFormat) {
Steve Blockd0e47292012-01-06 10:23:51 +0000507 ALOGW("create_echo_reference bad format rd %d, wr %d", rdFormat, wrFormat);
Eric Laurentb3184d72011-08-17 18:36:09 -0700508 return -EINVAL;
509 }
510 if ((rdChannelCount != 1 && rdChannelCount != 2) ||
511 wrChannelCount != 2) {
Glenn Kasten548d9cf2012-11-02 07:59:08 -0700512 ALOGW("create_echo_reference bad channel count rd %d, wr %d", rdChannelCount,
513 wrChannelCount);
Eric Laurentb3184d72011-08-17 18:36:09 -0700514 return -EINVAL;
515 }
516
Eric Laurentb3184d72011-08-17 18:36:09 -0700517 er = (struct echo_reference *)calloc(1, sizeof(struct echo_reference));
518
519 er->itfe.read = echo_reference_read;
520 er->itfe.write = echo_reference_write;
521
522 er->state = ECHOREF_IDLE;
523 er->rd_format = rdFormat;
524 er->rd_channel_count = rdChannelCount;
525 er->rd_sampling_rate = rdSamplingRate;
526 er->wr_format = wrFormat;
527 er->wr_channel_count = wrChannelCount;
528 er->wr_sampling_rate = wrSamplingRate;
529 er->rd_frame_size = audio_bytes_per_sample(rdFormat) * rdChannelCount;
530 er->wr_frame_size = audio_bytes_per_sample(wrFormat) * wrChannelCount;
531 *echo_reference = &er->itfe;
532 return 0;
533}
534
535void release_echo_reference(struct echo_reference_itfe *echo_reference) {
536 struct echo_reference *er = (struct echo_reference *)echo_reference;
537
538 if (er == NULL) {
539 return;
540 }
541
Steve Block473d4a52011-10-26 11:12:08 +0100542 ALOGV("EchoReference dstor");
Eric Laurentb3184d72011-08-17 18:36:09 -0700543 echo_reference_reset_l(er);
Eric Laurent33e8f782012-02-06 13:57:31 -0800544 if (er->resampler != NULL) {
545 release_resampler(er->resampler);
Eric Laurentb3184d72011-08-17 18:36:09 -0700546 }
547 free(er);
548}
549