xref: /aosp_15_r20/system/media/audio_utils/echo_reference.c (revision b9df5ad1c9ac98a7fefaac271a55f7ae3db05414)
1*b9df5ad1SAndroid Build Coastguard Worker /*
2*b9df5ad1SAndroid Build Coastguard Worker ** Copyright 2011, The Android Open-Source Project
3*b9df5ad1SAndroid Build Coastguard Worker **
4*b9df5ad1SAndroid Build Coastguard Worker ** Licensed under the Apache License, Version 2.0 (the "License");
5*b9df5ad1SAndroid Build Coastguard Worker ** you may not use this file except in compliance with the License.
6*b9df5ad1SAndroid Build Coastguard Worker ** You may obtain a copy of the License at
7*b9df5ad1SAndroid Build Coastguard Worker **
8*b9df5ad1SAndroid Build Coastguard Worker **     http://www.apache.org/licenses/LICENSE-2.0
9*b9df5ad1SAndroid Build Coastguard Worker **
10*b9df5ad1SAndroid Build Coastguard Worker ** Unless required by applicable law or agreed to in writing, software
11*b9df5ad1SAndroid Build Coastguard Worker ** distributed under the License is distributed on an "AS IS" BASIS,
12*b9df5ad1SAndroid Build Coastguard Worker ** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*b9df5ad1SAndroid Build Coastguard Worker ** See the License for the specific language governing permissions and
14*b9df5ad1SAndroid Build Coastguard Worker ** limitations under the License.
15*b9df5ad1SAndroid Build Coastguard Worker */
16*b9df5ad1SAndroid Build Coastguard Worker 
17*b9df5ad1SAndroid Build Coastguard Worker //#define LOG_NDEBUG 0
18*b9df5ad1SAndroid Build Coastguard Worker #define LOG_TAG "echo_reference"
19*b9df5ad1SAndroid Build Coastguard Worker 
20*b9df5ad1SAndroid Build Coastguard Worker #include <errno.h>
21*b9df5ad1SAndroid Build Coastguard Worker #include <inttypes.h>
22*b9df5ad1SAndroid Build Coastguard Worker #include <pthread.h>
23*b9df5ad1SAndroid Build Coastguard Worker #include <stddef.h>
24*b9df5ad1SAndroid Build Coastguard Worker #include <stdlib.h>
25*b9df5ad1SAndroid Build Coastguard Worker 
26*b9df5ad1SAndroid Build Coastguard Worker #include <log/log.h>
27*b9df5ad1SAndroid Build Coastguard Worker #include <system/audio.h>
28*b9df5ad1SAndroid Build Coastguard Worker #include <audio_utils/resampler.h>
29*b9df5ad1SAndroid Build Coastguard Worker #include <audio_utils/echo_reference.h>
30*b9df5ad1SAndroid Build Coastguard Worker 
31*b9df5ad1SAndroid Build Coastguard Worker // echo reference state: bit field indicating if read, write or both are active.
32*b9df5ad1SAndroid Build Coastguard Worker enum state {
33*b9df5ad1SAndroid Build Coastguard Worker     ECHOREF_IDLE = 0x00,        // idle
34*b9df5ad1SAndroid Build Coastguard Worker     ECHOREF_READING = 0x01,     // reading is active
35*b9df5ad1SAndroid Build Coastguard Worker     ECHOREF_WRITING = 0x02      // writing is active
36*b9df5ad1SAndroid Build Coastguard Worker };
37*b9df5ad1SAndroid Build Coastguard Worker 
38*b9df5ad1SAndroid Build Coastguard Worker struct echo_reference {
39*b9df5ad1SAndroid Build Coastguard Worker     struct echo_reference_itfe itfe;
40*b9df5ad1SAndroid Build Coastguard Worker     int status;                     // init status
41*b9df5ad1SAndroid Build Coastguard Worker     uint32_t state;                 // active state: reading, writing or both
42*b9df5ad1SAndroid Build Coastguard Worker     audio_format_t rd_format;       // read sample format
43*b9df5ad1SAndroid Build Coastguard Worker     uint32_t rd_channel_count;      // read number of channels
44*b9df5ad1SAndroid Build Coastguard Worker     uint32_t rd_sampling_rate;      // read sampling rate in Hz
45*b9df5ad1SAndroid Build Coastguard Worker     size_t rd_frame_size;           // read frame size (bytes per sample)
46*b9df5ad1SAndroid Build Coastguard Worker     audio_format_t wr_format;       // write sample format
47*b9df5ad1SAndroid Build Coastguard Worker     uint32_t wr_channel_count;      // write number of channels
48*b9df5ad1SAndroid Build Coastguard Worker     uint32_t wr_sampling_rate;      // write sampling rate in Hz
49*b9df5ad1SAndroid Build Coastguard Worker     size_t wr_frame_size;           // write frame size (bytes per sample)
50*b9df5ad1SAndroid Build Coastguard Worker     void *buffer;                   // main buffer
51*b9df5ad1SAndroid Build Coastguard Worker     size_t buf_size;                // main buffer size in frames
52*b9df5ad1SAndroid Build Coastguard Worker     size_t frames_in;               // number of frames in main buffer
53*b9df5ad1SAndroid Build Coastguard Worker     void *wr_buf;                   // buffer for input conversions
54*b9df5ad1SAndroid Build Coastguard Worker     size_t wr_buf_size;             // size of conversion buffer in frames
55*b9df5ad1SAndroid Build Coastguard Worker     size_t wr_frames_in;            // number of frames in conversion buffer
56*b9df5ad1SAndroid Build Coastguard Worker     size_t wr_curr_frame_size;      // number of frames given to current write() function
57*b9df5ad1SAndroid Build Coastguard Worker     void *wr_src_buf;               // resampler input buf (either wr_buf or buffer used by write())
58*b9df5ad1SAndroid Build Coastguard Worker     struct timespec wr_render_time; // latest render time indicated by write()
59*b9df5ad1SAndroid Build Coastguard Worker                                     // default ALSA gettimeofday() format
60*b9df5ad1SAndroid Build Coastguard Worker     int32_t  playback_delay;        // playback buffer delay indicated by last write()
61*b9df5ad1SAndroid Build Coastguard Worker     int16_t prev_delta_sign;        // sign of previous delay difference:
62*b9df5ad1SAndroid Build Coastguard Worker                                     //  1: positive, -1: negative, 0: unknown
63*b9df5ad1SAndroid Build Coastguard Worker     uint16_t delta_count;           // number of consecutive delay differences with same sign
64*b9df5ad1SAndroid Build Coastguard Worker     pthread_mutex_t lock;                      // mutex protecting read/write concurrency
65*b9df5ad1SAndroid Build Coastguard Worker     pthread_cond_t cond;                       // condition signaled when data is ready to read
66*b9df5ad1SAndroid Build Coastguard Worker     struct resampler_itfe *resampler;          // input resampler
67*b9df5ad1SAndroid Build Coastguard Worker     struct resampler_buffer_provider provider; // resampler buffer provider
68*b9df5ad1SAndroid Build Coastguard Worker };
69*b9df5ad1SAndroid Build Coastguard Worker 
70*b9df5ad1SAndroid Build Coastguard Worker 
echo_reference_get_next_buffer(struct resampler_buffer_provider * buffer_provider,struct resampler_buffer * buffer)71*b9df5ad1SAndroid Build Coastguard Worker int echo_reference_get_next_buffer(struct resampler_buffer_provider *buffer_provider,
72*b9df5ad1SAndroid Build Coastguard Worker                                    struct resampler_buffer* buffer)
73*b9df5ad1SAndroid Build Coastguard Worker {
74*b9df5ad1SAndroid Build Coastguard Worker     struct echo_reference *er;
75*b9df5ad1SAndroid Build Coastguard Worker 
76*b9df5ad1SAndroid Build Coastguard Worker     if (buffer_provider == NULL) {
77*b9df5ad1SAndroid Build Coastguard Worker         return -EINVAL;
78*b9df5ad1SAndroid Build Coastguard Worker     }
79*b9df5ad1SAndroid Build Coastguard Worker 
80*b9df5ad1SAndroid Build Coastguard Worker     er = (struct echo_reference *)((char *)buffer_provider -
81*b9df5ad1SAndroid Build Coastguard Worker                                       offsetof(struct echo_reference, provider));
82*b9df5ad1SAndroid Build Coastguard Worker 
83*b9df5ad1SAndroid Build Coastguard Worker     if (er->wr_src_buf == NULL || er->wr_frames_in == 0) {
84*b9df5ad1SAndroid Build Coastguard Worker         buffer->raw = NULL;
85*b9df5ad1SAndroid Build Coastguard Worker         buffer->frame_count = 0;
86*b9df5ad1SAndroid Build Coastguard Worker         return -ENODATA;
87*b9df5ad1SAndroid Build Coastguard Worker     }
88*b9df5ad1SAndroid Build Coastguard Worker 
89*b9df5ad1SAndroid Build Coastguard Worker     buffer->frame_count = (buffer->frame_count > er->wr_frames_in) ?
90*b9df5ad1SAndroid Build Coastguard Worker             er->wr_frames_in : buffer->frame_count;
91*b9df5ad1SAndroid Build Coastguard Worker     // this is er->rd_channel_count here as we resample after stereo to mono conversion if any
92*b9df5ad1SAndroid Build Coastguard Worker     buffer->i16 = (int16_t *)er->wr_src_buf + (er->wr_curr_frame_size - er->wr_frames_in) *
93*b9df5ad1SAndroid Build Coastguard Worker             er->rd_channel_count;
94*b9df5ad1SAndroid Build Coastguard Worker 
95*b9df5ad1SAndroid Build Coastguard Worker     return 0;
96*b9df5ad1SAndroid Build Coastguard Worker }
97*b9df5ad1SAndroid Build Coastguard Worker 
echo_reference_release_buffer(struct resampler_buffer_provider * buffer_provider,struct resampler_buffer * buffer)98*b9df5ad1SAndroid Build Coastguard Worker void echo_reference_release_buffer(struct resampler_buffer_provider *buffer_provider,
99*b9df5ad1SAndroid Build Coastguard Worker                                   struct resampler_buffer* buffer)
100*b9df5ad1SAndroid Build Coastguard Worker {
101*b9df5ad1SAndroid Build Coastguard Worker     struct echo_reference *er;
102*b9df5ad1SAndroid Build Coastguard Worker 
103*b9df5ad1SAndroid Build Coastguard Worker     if (buffer_provider == NULL) {
104*b9df5ad1SAndroid Build Coastguard Worker         return;
105*b9df5ad1SAndroid Build Coastguard Worker     }
106*b9df5ad1SAndroid Build Coastguard Worker 
107*b9df5ad1SAndroid Build Coastguard Worker     er = (struct echo_reference *)((char *)buffer_provider -
108*b9df5ad1SAndroid Build Coastguard Worker                                       offsetof(struct echo_reference, provider));
109*b9df5ad1SAndroid Build Coastguard Worker 
110*b9df5ad1SAndroid Build Coastguard Worker     er->wr_frames_in -= buffer->frame_count;
111*b9df5ad1SAndroid Build Coastguard Worker }
112*b9df5ad1SAndroid Build Coastguard Worker 
echo_reference_reset_l(struct echo_reference * er)113*b9df5ad1SAndroid Build Coastguard Worker static void echo_reference_reset_l(struct echo_reference *er)
114*b9df5ad1SAndroid Build Coastguard Worker {
115*b9df5ad1SAndroid Build Coastguard Worker     ALOGV("echo_reference_reset_l()");
116*b9df5ad1SAndroid Build Coastguard Worker     free(er->buffer);
117*b9df5ad1SAndroid Build Coastguard Worker     er->buffer = NULL;
118*b9df5ad1SAndroid Build Coastguard Worker     er->buf_size = 0;
119*b9df5ad1SAndroid Build Coastguard Worker     er->frames_in = 0;
120*b9df5ad1SAndroid Build Coastguard Worker     free(er->wr_buf);
121*b9df5ad1SAndroid Build Coastguard Worker     er->wr_buf = NULL;
122*b9df5ad1SAndroid Build Coastguard Worker     er->wr_buf_size = 0;
123*b9df5ad1SAndroid Build Coastguard Worker     er->wr_render_time.tv_sec = 0;
124*b9df5ad1SAndroid Build Coastguard Worker     er->wr_render_time.tv_nsec = 0;
125*b9df5ad1SAndroid Build Coastguard Worker     er->delta_count = 0;
126*b9df5ad1SAndroid Build Coastguard Worker     er->prev_delta_sign = 0;
127*b9df5ad1SAndroid Build Coastguard Worker }
128*b9df5ad1SAndroid Build Coastguard Worker 
129*b9df5ad1SAndroid Build Coastguard Worker /* additional space in resampler buffer allowing for extra samples to be returned
130*b9df5ad1SAndroid Build Coastguard Worker  * by speex resampler when sample rates ratio is not an integer.
131*b9df5ad1SAndroid Build Coastguard Worker  */
132*b9df5ad1SAndroid Build Coastguard Worker #define RESAMPLER_HEADROOM_SAMPLES   10
133*b9df5ad1SAndroid Build Coastguard Worker 
echo_reference_write(struct echo_reference_itfe * echo_reference,struct echo_reference_buffer * buffer)134*b9df5ad1SAndroid Build Coastguard Worker static int echo_reference_write(struct echo_reference_itfe *echo_reference,
135*b9df5ad1SAndroid Build Coastguard Worker                          struct echo_reference_buffer *buffer)
136*b9df5ad1SAndroid Build Coastguard Worker {
137*b9df5ad1SAndroid Build Coastguard Worker     struct echo_reference *er = (struct echo_reference *)echo_reference;
138*b9df5ad1SAndroid Build Coastguard Worker     int status = 0;
139*b9df5ad1SAndroid Build Coastguard Worker 
140*b9df5ad1SAndroid Build Coastguard Worker     if (er == NULL) {
141*b9df5ad1SAndroid Build Coastguard Worker         return -EINVAL;
142*b9df5ad1SAndroid Build Coastguard Worker     }
143*b9df5ad1SAndroid Build Coastguard Worker 
144*b9df5ad1SAndroid Build Coastguard Worker     pthread_mutex_lock(&er->lock);
145*b9df5ad1SAndroid Build Coastguard Worker 
146*b9df5ad1SAndroid Build Coastguard Worker     if (buffer == NULL) {
147*b9df5ad1SAndroid Build Coastguard Worker         ALOGV("echo_reference_write() stop write");
148*b9df5ad1SAndroid Build Coastguard Worker         er->state &= ~ECHOREF_WRITING;
149*b9df5ad1SAndroid Build Coastguard Worker         echo_reference_reset_l(er);
150*b9df5ad1SAndroid Build Coastguard Worker         goto exit;
151*b9df5ad1SAndroid Build Coastguard Worker     }
152*b9df5ad1SAndroid Build Coastguard Worker 
153*b9df5ad1SAndroid Build Coastguard Worker     ALOGV("echo_reference_write() START trying to write %zu frames", buffer->frame_count);
154*b9df5ad1SAndroid Build Coastguard Worker     ALOGV("echo_reference_write() playbackTimestamp:[%d].[%d], er->playback_delay:[%d]",
155*b9df5ad1SAndroid Build Coastguard Worker             (int)buffer->time_stamp.tv_sec,
156*b9df5ad1SAndroid Build Coastguard Worker             (int)buffer->time_stamp.tv_nsec, er->playback_delay);
157*b9df5ad1SAndroid Build Coastguard Worker 
158*b9df5ad1SAndroid Build Coastguard Worker     //ALOGV("echo_reference_write() %d frames", buffer->frame_count);
159*b9df5ad1SAndroid Build Coastguard Worker     // discard writes until a valid time stamp is provided.
160*b9df5ad1SAndroid Build Coastguard Worker 
161*b9df5ad1SAndroid Build Coastguard Worker     if ((buffer->time_stamp.tv_sec == 0) && (buffer->time_stamp.tv_nsec == 0) &&
162*b9df5ad1SAndroid Build Coastguard Worker         (er->wr_render_time.tv_sec == 0) && (er->wr_render_time.tv_nsec == 0)) {
163*b9df5ad1SAndroid Build Coastguard Worker         goto exit;
164*b9df5ad1SAndroid Build Coastguard Worker     }
165*b9df5ad1SAndroid Build Coastguard Worker 
166*b9df5ad1SAndroid Build Coastguard Worker     if ((er->state & ECHOREF_WRITING) == 0) {
167*b9df5ad1SAndroid Build Coastguard Worker         ALOGV("echo_reference_write() start write");
168*b9df5ad1SAndroid Build Coastguard Worker         if (er->resampler != NULL) {
169*b9df5ad1SAndroid Build Coastguard Worker             er->resampler->reset(er->resampler);
170*b9df5ad1SAndroid Build Coastguard Worker         }
171*b9df5ad1SAndroid Build Coastguard Worker         er->state |= ECHOREF_WRITING;
172*b9df5ad1SAndroid Build Coastguard Worker     }
173*b9df5ad1SAndroid Build Coastguard Worker 
174*b9df5ad1SAndroid Build Coastguard Worker     if ((er->state & ECHOREF_READING) == 0) {
175*b9df5ad1SAndroid Build Coastguard Worker         goto exit;
176*b9df5ad1SAndroid Build Coastguard Worker     }
177*b9df5ad1SAndroid Build Coastguard Worker 
178*b9df5ad1SAndroid Build Coastguard Worker     er->wr_render_time.tv_sec  = buffer->time_stamp.tv_sec;
179*b9df5ad1SAndroid Build Coastguard Worker     er->wr_render_time.tv_nsec = buffer->time_stamp.tv_nsec;
180*b9df5ad1SAndroid Build Coastguard Worker 
181*b9df5ad1SAndroid Build Coastguard Worker     er->playback_delay = buffer->delay_ns;
182*b9df5ad1SAndroid Build Coastguard Worker 
183*b9df5ad1SAndroid Build Coastguard Worker     // this will be used in the get_next_buffer, to support variable input buffer sizes
184*b9df5ad1SAndroid Build Coastguard Worker     er->wr_curr_frame_size = buffer->frame_count;
185*b9df5ad1SAndroid Build Coastguard Worker 
186*b9df5ad1SAndroid Build Coastguard Worker     void *srcBuf;
187*b9df5ad1SAndroid Build Coastguard Worker     size_t inFrames;
188*b9df5ad1SAndroid Build Coastguard Worker     // do stereo to mono and down sampling if necessary
189*b9df5ad1SAndroid Build Coastguard Worker     if (er->rd_channel_count != er->wr_channel_count ||
190*b9df5ad1SAndroid Build Coastguard Worker             er->rd_sampling_rate != er->wr_sampling_rate) {
191*b9df5ad1SAndroid Build Coastguard Worker         size_t wrBufSize = buffer->frame_count;
192*b9df5ad1SAndroid Build Coastguard Worker 
193*b9df5ad1SAndroid Build Coastguard Worker         inFrames = buffer->frame_count;
194*b9df5ad1SAndroid Build Coastguard Worker 
195*b9df5ad1SAndroid Build Coastguard Worker         if (er->rd_sampling_rate != er->wr_sampling_rate) {
196*b9df5ad1SAndroid Build Coastguard Worker             inFrames = (buffer->frame_count * er->rd_sampling_rate) / er->wr_sampling_rate +
197*b9df5ad1SAndroid Build Coastguard Worker                                                     RESAMPLER_HEADROOM_SAMPLES;
198*b9df5ad1SAndroid Build Coastguard Worker             // wr_buf is not only used as resampler output but also for stereo to mono conversion
199*b9df5ad1SAndroid Build Coastguard Worker             // output so buffer size is driven by both write and read sample rates
200*b9df5ad1SAndroid Build Coastguard Worker             if (inFrames > wrBufSize) {
201*b9df5ad1SAndroid Build Coastguard Worker                 wrBufSize = inFrames;
202*b9df5ad1SAndroid Build Coastguard Worker             }
203*b9df5ad1SAndroid Build Coastguard Worker         }
204*b9df5ad1SAndroid Build Coastguard Worker 
205*b9df5ad1SAndroid Build Coastguard Worker         if (er->wr_buf_size < wrBufSize) {
206*b9df5ad1SAndroid Build Coastguard Worker             ALOGV("echo_reference_write() increasing write buffer size from %zu to %zu",
207*b9df5ad1SAndroid Build Coastguard Worker                     er->wr_buf_size, wrBufSize);
208*b9df5ad1SAndroid Build Coastguard Worker             er->wr_buf_size = wrBufSize;
209*b9df5ad1SAndroid Build Coastguard Worker             void *new_buf = realloc(er->wr_buf, er->wr_buf_size * er->rd_frame_size);
210*b9df5ad1SAndroid Build Coastguard Worker             if (new_buf == NULL) {
211*b9df5ad1SAndroid Build Coastguard Worker                 status = -ENOMEM;
212*b9df5ad1SAndroid Build Coastguard Worker                 goto exit;
213*b9df5ad1SAndroid Build Coastguard Worker             } else {
214*b9df5ad1SAndroid Build Coastguard Worker                 er->wr_buf = new_buf;
215*b9df5ad1SAndroid Build Coastguard Worker             }
216*b9df5ad1SAndroid Build Coastguard Worker         }
217*b9df5ad1SAndroid Build Coastguard Worker 
218*b9df5ad1SAndroid Build Coastguard Worker         if (er->rd_channel_count != er->wr_channel_count) {
219*b9df5ad1SAndroid Build Coastguard Worker             // must be stereo to mono
220*b9df5ad1SAndroid Build Coastguard Worker             int16_t *src16 = (int16_t *)buffer->raw;
221*b9df5ad1SAndroid Build Coastguard Worker             int16_t *dst16 = (int16_t *)er->wr_buf;
222*b9df5ad1SAndroid Build Coastguard Worker             size_t frames = buffer->frame_count;
223*b9df5ad1SAndroid Build Coastguard Worker             while (frames--) {
224*b9df5ad1SAndroid Build Coastguard Worker                 *dst16++ = (int16_t)(((int32_t)*src16 + (int32_t)*(src16 + 1)) >> 1);
225*b9df5ad1SAndroid Build Coastguard Worker                 src16 += 2;
226*b9df5ad1SAndroid Build Coastguard Worker             }
227*b9df5ad1SAndroid Build Coastguard Worker         }
228*b9df5ad1SAndroid Build Coastguard Worker         if (er->wr_sampling_rate != er->rd_sampling_rate) {
229*b9df5ad1SAndroid Build Coastguard Worker             if (er->resampler == NULL) {
230*b9df5ad1SAndroid Build Coastguard Worker                 int rc;
231*b9df5ad1SAndroid Build Coastguard Worker                 ALOGV("echo_reference_write() new ReSampler(%d, %d)",
232*b9df5ad1SAndroid Build Coastguard Worker                       er->wr_sampling_rate, er->rd_sampling_rate);
233*b9df5ad1SAndroid Build Coastguard Worker                 er->provider.get_next_buffer = echo_reference_get_next_buffer;
234*b9df5ad1SAndroid Build Coastguard Worker                 er->provider.release_buffer = echo_reference_release_buffer;
235*b9df5ad1SAndroid Build Coastguard Worker                 rc = create_resampler(er->wr_sampling_rate,
236*b9df5ad1SAndroid Build Coastguard Worker                                  er->rd_sampling_rate,
237*b9df5ad1SAndroid Build Coastguard Worker                                  er->rd_channel_count,
238*b9df5ad1SAndroid Build Coastguard Worker                                  RESAMPLER_QUALITY_DEFAULT,
239*b9df5ad1SAndroid Build Coastguard Worker                                  &er->provider,
240*b9df5ad1SAndroid Build Coastguard Worker                                  &er->resampler);
241*b9df5ad1SAndroid Build Coastguard Worker                 if (rc != 0) {
242*b9df5ad1SAndroid Build Coastguard Worker                     er->resampler = NULL;
243*b9df5ad1SAndroid Build Coastguard Worker                     ALOGV("echo_reference_write() failure to create resampler %d", rc);
244*b9df5ad1SAndroid Build Coastguard Worker                     status = -ENODEV;
245*b9df5ad1SAndroid Build Coastguard Worker                     goto exit;
246*b9df5ad1SAndroid Build Coastguard Worker                 }
247*b9df5ad1SAndroid Build Coastguard Worker             }
248*b9df5ad1SAndroid Build Coastguard Worker             // er->wr_src_buf and er->wr_frames_in are used by getNexBuffer() called by the
249*b9df5ad1SAndroid Build Coastguard Worker             // resampler to get new frames
250*b9df5ad1SAndroid Build Coastguard Worker             if (er->rd_channel_count != er->wr_channel_count) {
251*b9df5ad1SAndroid Build Coastguard Worker                 er->wr_src_buf = er->wr_buf;
252*b9df5ad1SAndroid Build Coastguard Worker             } else {
253*b9df5ad1SAndroid Build Coastguard Worker                 er->wr_src_buf = buffer->raw;
254*b9df5ad1SAndroid Build Coastguard Worker             }
255*b9df5ad1SAndroid Build Coastguard Worker             er->wr_frames_in = buffer->frame_count;
256*b9df5ad1SAndroid Build Coastguard Worker             // inFrames is always more than we need here to get frames remaining from previous runs
257*b9df5ad1SAndroid Build Coastguard Worker             // inFrames is updated by resample() with the number of frames produced
258*b9df5ad1SAndroid Build Coastguard Worker             ALOGV("echo_reference_write() ReSampling(%d, %d)",
259*b9df5ad1SAndroid Build Coastguard Worker                   er->wr_sampling_rate, er->rd_sampling_rate);
260*b9df5ad1SAndroid Build Coastguard Worker             er->resampler->resample_from_provider(er->resampler,
261*b9df5ad1SAndroid Build Coastguard Worker                                                      (int16_t *)er->wr_buf, &inFrames);
262*b9df5ad1SAndroid Build Coastguard Worker             ALOGV_IF(er->wr_frames_in != 0,
263*b9df5ad1SAndroid Build Coastguard Worker                     "echo_reference_write() er->wr_frames_in not 0 (%zu) after resampler",
264*b9df5ad1SAndroid Build Coastguard Worker                     er->wr_frames_in);
265*b9df5ad1SAndroid Build Coastguard Worker         }
266*b9df5ad1SAndroid Build Coastguard Worker         srcBuf = er->wr_buf;
267*b9df5ad1SAndroid Build Coastguard Worker     } else {
268*b9df5ad1SAndroid Build Coastguard Worker         inFrames = buffer->frame_count;
269*b9df5ad1SAndroid Build Coastguard Worker         srcBuf = buffer->raw;
270*b9df5ad1SAndroid Build Coastguard Worker     }
271*b9df5ad1SAndroid Build Coastguard Worker 
272*b9df5ad1SAndroid Build Coastguard Worker     if (er->frames_in + inFrames > er->buf_size) {
273*b9df5ad1SAndroid Build Coastguard Worker         ALOGV("echo_reference_write() increasing buffer size from %zu to %zu",
274*b9df5ad1SAndroid Build Coastguard Worker                 er->buf_size, er->frames_in + inFrames);
275*b9df5ad1SAndroid Build Coastguard Worker         er->buf_size = er->frames_in + inFrames;
276*b9df5ad1SAndroid Build Coastguard Worker         void *new_buf = realloc(er->buffer, er->buf_size * er->rd_frame_size);
277*b9df5ad1SAndroid Build Coastguard Worker         if (new_buf == NULL) {
278*b9df5ad1SAndroid Build Coastguard Worker             status = -ENOMEM;
279*b9df5ad1SAndroid Build Coastguard Worker             goto exit;
280*b9df5ad1SAndroid Build Coastguard Worker         } else {
281*b9df5ad1SAndroid Build Coastguard Worker             er->buffer = new_buf;
282*b9df5ad1SAndroid Build Coastguard Worker         }
283*b9df5ad1SAndroid Build Coastguard Worker     }
284*b9df5ad1SAndroid Build Coastguard Worker     memcpy((char *)er->buffer + er->frames_in * er->rd_frame_size,
285*b9df5ad1SAndroid Build Coastguard Worker            srcBuf,
286*b9df5ad1SAndroid Build Coastguard Worker            inFrames * er->rd_frame_size);
287*b9df5ad1SAndroid Build Coastguard Worker     er->frames_in += inFrames;
288*b9df5ad1SAndroid Build Coastguard Worker 
289*b9df5ad1SAndroid Build Coastguard Worker     ALOGV("echo_reference_write() frames written:[%zu], frames total:[%zu] buffer size:[%zu]\n"
290*b9df5ad1SAndroid Build Coastguard Worker           "                       er->wr_render_time:[%d].[%d], er->playback_delay:[%d]",
291*b9df5ad1SAndroid Build Coastguard Worker           inFrames, er->frames_in, er->buf_size,
292*b9df5ad1SAndroid Build Coastguard Worker           (int)er->wr_render_time.tv_sec, (int)er->wr_render_time.tv_nsec, er->playback_delay);
293*b9df5ad1SAndroid Build Coastguard Worker 
294*b9df5ad1SAndroid Build Coastguard Worker     pthread_cond_signal(&er->cond);
295*b9df5ad1SAndroid Build Coastguard Worker exit:
296*b9df5ad1SAndroid Build Coastguard Worker     pthread_mutex_unlock(&er->lock);
297*b9df5ad1SAndroid Build Coastguard Worker     ALOGV("echo_reference_write() END");
298*b9df5ad1SAndroid Build Coastguard Worker     return status;
299*b9df5ad1SAndroid Build Coastguard Worker }
300*b9df5ad1SAndroid Build Coastguard Worker 
301*b9df5ad1SAndroid Build Coastguard Worker // delay jump threshold to update ref buffer: 6 samples at 8kHz in nsecs
302*b9df5ad1SAndroid Build Coastguard Worker #define MIN_DELAY_DELTA_NS (375000*2)
303*b9df5ad1SAndroid Build Coastguard Worker // number of consecutive delta with same sign between expected and actual delay before adjusting
304*b9df5ad1SAndroid Build Coastguard Worker // the buffer
305*b9df5ad1SAndroid Build Coastguard Worker #define MIN_DELTA_NUM 4
306*b9df5ad1SAndroid Build Coastguard Worker 
307*b9df5ad1SAndroid Build Coastguard Worker 
echo_reference_read(struct echo_reference_itfe * echo_reference,struct echo_reference_buffer * buffer)308*b9df5ad1SAndroid Build Coastguard Worker static int echo_reference_read(struct echo_reference_itfe *echo_reference,
309*b9df5ad1SAndroid Build Coastguard Worker                          struct echo_reference_buffer *buffer)
310*b9df5ad1SAndroid Build Coastguard Worker {
311*b9df5ad1SAndroid Build Coastguard Worker     struct echo_reference *er = (struct echo_reference *)echo_reference;
312*b9df5ad1SAndroid Build Coastguard Worker     int status = 0;
313*b9df5ad1SAndroid Build Coastguard Worker 
314*b9df5ad1SAndroid Build Coastguard Worker     if (er == NULL) {
315*b9df5ad1SAndroid Build Coastguard Worker         return -EINVAL;
316*b9df5ad1SAndroid Build Coastguard Worker     }
317*b9df5ad1SAndroid Build Coastguard Worker 
318*b9df5ad1SAndroid Build Coastguard Worker     pthread_mutex_lock(&er->lock);
319*b9df5ad1SAndroid Build Coastguard Worker 
320*b9df5ad1SAndroid Build Coastguard Worker     if (buffer == NULL) {
321*b9df5ad1SAndroid Build Coastguard Worker         ALOGV("echo_reference_read() stop read");
322*b9df5ad1SAndroid Build Coastguard Worker         er->state &= ~ECHOREF_READING;
323*b9df5ad1SAndroid Build Coastguard Worker         goto exit;
324*b9df5ad1SAndroid Build Coastguard Worker     }
325*b9df5ad1SAndroid Build Coastguard Worker 
326*b9df5ad1SAndroid Build Coastguard Worker     ALOGV("echo_reference_read() START, delayCapture:[%d], "
327*b9df5ad1SAndroid Build Coastguard Worker             "er->frames_in:[%zu],buffer->frame_count:[%zu]",
328*b9df5ad1SAndroid Build Coastguard Worker     buffer->delay_ns, er->frames_in, buffer->frame_count);
329*b9df5ad1SAndroid Build Coastguard Worker 
330*b9df5ad1SAndroid Build Coastguard Worker     if ((er->state & ECHOREF_READING) == 0) {
331*b9df5ad1SAndroid Build Coastguard Worker         ALOGV("echo_reference_read() start read");
332*b9df5ad1SAndroid Build Coastguard Worker         echo_reference_reset_l(er);
333*b9df5ad1SAndroid Build Coastguard Worker         er->state |= ECHOREF_READING;
334*b9df5ad1SAndroid Build Coastguard Worker     }
335*b9df5ad1SAndroid Build Coastguard Worker 
336*b9df5ad1SAndroid Build Coastguard Worker     if ((er->state & ECHOREF_WRITING) == 0) {
337*b9df5ad1SAndroid Build Coastguard Worker         memset(buffer->raw, 0, er->rd_frame_size * buffer->frame_count);
338*b9df5ad1SAndroid Build Coastguard Worker         buffer->delay_ns = 0;
339*b9df5ad1SAndroid Build Coastguard Worker         goto exit;
340*b9df5ad1SAndroid Build Coastguard Worker     }
341*b9df5ad1SAndroid Build Coastguard Worker 
342*b9df5ad1SAndroid Build Coastguard Worker //    ALOGV("echo_reference_read() %d frames", buffer->frame_count);
343*b9df5ad1SAndroid Build Coastguard Worker 
344*b9df5ad1SAndroid Build Coastguard Worker     // allow some time for new frames to arrive if not enough frames are ready for read
345*b9df5ad1SAndroid Build Coastguard Worker     if (er->frames_in < buffer->frame_count) {
346*b9df5ad1SAndroid Build Coastguard Worker         uint32_t timeoutMs = (uint32_t)((1000 * buffer->frame_count) / er->rd_sampling_rate / 2);
347*b9df5ad1SAndroid Build Coastguard Worker         struct timespec ts = {0, 0};
348*b9df5ad1SAndroid Build Coastguard Worker 
349*b9df5ad1SAndroid Build Coastguard Worker         clock_gettime(CLOCK_REALTIME, &ts);
350*b9df5ad1SAndroid Build Coastguard Worker 
351*b9df5ad1SAndroid Build Coastguard Worker         ts.tv_sec  += timeoutMs/1000;
352*b9df5ad1SAndroid Build Coastguard Worker         ts.tv_nsec += (timeoutMs%1000) * 1000000;
353*b9df5ad1SAndroid Build Coastguard Worker         if (ts.tv_nsec >= 1000000000) {
354*b9df5ad1SAndroid Build Coastguard Worker             ts.tv_nsec -= 1000000000;
355*b9df5ad1SAndroid Build Coastguard Worker             ts.tv_sec  += 1;
356*b9df5ad1SAndroid Build Coastguard Worker         }
357*b9df5ad1SAndroid Build Coastguard Worker 
358*b9df5ad1SAndroid Build Coastguard Worker         pthread_cond_timedwait(&er->cond, &er->lock, &ts);
359*b9df5ad1SAndroid Build Coastguard Worker 
360*b9df5ad1SAndroid Build Coastguard Worker         ALOGV_IF((er->frames_in < buffer->frame_count),
361*b9df5ad1SAndroid Build Coastguard Worker                  "echo_reference_read() waited %d ms but still not enough frames"
362*b9df5ad1SAndroid Build Coastguard Worker                  " er->frames_in: %zu, buffer->frame_count = %zu",
363*b9df5ad1SAndroid Build Coastguard Worker                  timeoutMs, er->frames_in, buffer->frame_count);
364*b9df5ad1SAndroid Build Coastguard Worker     }
365*b9df5ad1SAndroid Build Coastguard Worker 
366*b9df5ad1SAndroid Build Coastguard Worker     int64_t timeDiff;
367*b9df5ad1SAndroid Build Coastguard Worker     struct timespec tmp;
368*b9df5ad1SAndroid Build Coastguard Worker 
369*b9df5ad1SAndroid Build Coastguard Worker     if ((er->wr_render_time.tv_sec == 0 && er->wr_render_time.tv_nsec == 0) ||
370*b9df5ad1SAndroid Build Coastguard Worker         (buffer->time_stamp.tv_sec == 0 && buffer->time_stamp.tv_nsec == 0)) {
371*b9df5ad1SAndroid Build Coastguard Worker         ALOGV("echo_reference_read(): NEW:timestamp is zero---------setting timeDiff = 0, "
372*b9df5ad1SAndroid Build Coastguard Worker              "not updating delay this time");
373*b9df5ad1SAndroid Build Coastguard Worker         timeDiff = 0;
374*b9df5ad1SAndroid Build Coastguard Worker     } else {
375*b9df5ad1SAndroid Build Coastguard Worker         if (buffer->time_stamp.tv_nsec < er->wr_render_time.tv_nsec) {
376*b9df5ad1SAndroid Build Coastguard Worker             tmp.tv_sec = buffer->time_stamp.tv_sec - er->wr_render_time.tv_sec - 1;
377*b9df5ad1SAndroid Build Coastguard Worker             tmp.tv_nsec = 1000000000 + buffer->time_stamp.tv_nsec - er->wr_render_time.tv_nsec;
378*b9df5ad1SAndroid Build Coastguard Worker         } else {
379*b9df5ad1SAndroid Build Coastguard Worker             tmp.tv_sec = buffer->time_stamp.tv_sec - er->wr_render_time.tv_sec;
380*b9df5ad1SAndroid Build Coastguard Worker             tmp.tv_nsec = buffer->time_stamp.tv_nsec - er->wr_render_time.tv_nsec;
381*b9df5ad1SAndroid Build Coastguard Worker         }
382*b9df5ad1SAndroid Build Coastguard Worker         timeDiff = (((int64_t)tmp.tv_sec * 1000000000 + tmp.tv_nsec));
383*b9df5ad1SAndroid Build Coastguard Worker 
384*b9df5ad1SAndroid Build Coastguard Worker         int64_t expectedDelayNs =  er->playback_delay + buffer->delay_ns - timeDiff;
385*b9df5ad1SAndroid Build Coastguard Worker 
386*b9df5ad1SAndroid Build Coastguard Worker         if (er->resampler != NULL) {
387*b9df5ad1SAndroid Build Coastguard Worker             // Resampler already compensates part of the delay
388*b9df5ad1SAndroid Build Coastguard Worker             int32_t rsmp_delay = er->resampler->delay_ns(er->resampler);
389*b9df5ad1SAndroid Build Coastguard Worker             expectedDelayNs -= rsmp_delay;
390*b9df5ad1SAndroid Build Coastguard Worker         }
391*b9df5ad1SAndroid Build Coastguard Worker 
392*b9df5ad1SAndroid Build Coastguard Worker         ALOGV("echo_reference_read(): expectedDelayNs[%" PRId64 "] = "
393*b9df5ad1SAndroid Build Coastguard Worker                 "er->playback_delay[%d] + delayCapture[%d"
394*b9df5ad1SAndroid Build Coastguard Worker                 "] - timeDiff[%" PRId64 "]",
395*b9df5ad1SAndroid Build Coastguard Worker                 expectedDelayNs, er->playback_delay, buffer->delay_ns, timeDiff);
396*b9df5ad1SAndroid Build Coastguard Worker 
397*b9df5ad1SAndroid Build Coastguard Worker         if (expectedDelayNs > 0) {
398*b9df5ad1SAndroid Build Coastguard Worker             int64_t delayNs = ((int64_t)er->frames_in * 1000000000) / er->rd_sampling_rate;
399*b9df5ad1SAndroid Build Coastguard Worker 
400*b9df5ad1SAndroid Build Coastguard Worker             int64_t  deltaNs = delayNs - expectedDelayNs;
401*b9df5ad1SAndroid Build Coastguard Worker 
402*b9df5ad1SAndroid Build Coastguard Worker             ALOGV("echo_reference_read(): EchoPathDelayDeviation between reference and DMA [%"
403*b9df5ad1SAndroid Build Coastguard Worker                     PRId64 "]", deltaNs);
404*b9df5ad1SAndroid Build Coastguard Worker             if (llabs(deltaNs) >= MIN_DELAY_DELTA_NS) {
405*b9df5ad1SAndroid Build Coastguard Worker                 // smooth the variation and update the reference buffer only
406*b9df5ad1SAndroid Build Coastguard Worker                 // if a deviation in the same direction is observed for more than MIN_DELTA_NUM
407*b9df5ad1SAndroid Build Coastguard Worker                 // consecutive reads.
408*b9df5ad1SAndroid Build Coastguard Worker                 int16_t delay_sign = (deltaNs >= 0) ? 1 : -1;
409*b9df5ad1SAndroid Build Coastguard Worker                 if (delay_sign == er->prev_delta_sign) {
410*b9df5ad1SAndroid Build Coastguard Worker                     er->delta_count++;
411*b9df5ad1SAndroid Build Coastguard Worker                 } else {
412*b9df5ad1SAndroid Build Coastguard Worker                     er->delta_count = 1;
413*b9df5ad1SAndroid Build Coastguard Worker                 }
414*b9df5ad1SAndroid Build Coastguard Worker                 er->prev_delta_sign = delay_sign;
415*b9df5ad1SAndroid Build Coastguard Worker 
416*b9df5ad1SAndroid Build Coastguard Worker                 if (er->delta_count > MIN_DELTA_NUM) {
417*b9df5ad1SAndroid Build Coastguard Worker                     size_t previousFrameIn = er->frames_in;
418*b9df5ad1SAndroid Build Coastguard Worker                     er->frames_in = (size_t)((expectedDelayNs * er->rd_sampling_rate)/1000000000);
419*b9df5ad1SAndroid Build Coastguard Worker                     int offset = er->frames_in - previousFrameIn;
420*b9df5ad1SAndroid Build Coastguard Worker 
421*b9df5ad1SAndroid Build Coastguard Worker                     ALOGV("echo_reference_read(): deltaNs ENOUGH and %s: "
422*b9df5ad1SAndroid Build Coastguard Worker                             "er->frames_in: %zu, previousFrameIn = %zu",
423*b9df5ad1SAndroid Build Coastguard Worker                          delay_sign ? "positive" : "negative", er->frames_in, previousFrameIn);
424*b9df5ad1SAndroid Build Coastguard Worker 
425*b9df5ad1SAndroid Build Coastguard Worker                     if (deltaNs < 0) {
426*b9df5ad1SAndroid Build Coastguard Worker                         // Less data available in the reference buffer than expected
427*b9df5ad1SAndroid Build Coastguard Worker                         if (er->frames_in > er->buf_size) {
428*b9df5ad1SAndroid Build Coastguard Worker                             er->buf_size = er->frames_in;
429*b9df5ad1SAndroid Build Coastguard Worker                             ALOGV("echo_reference_read(): increasing buffer size to %zu",
430*b9df5ad1SAndroid Build Coastguard Worker                                   er->buf_size);
431*b9df5ad1SAndroid Build Coastguard Worker                             void *new_buf = realloc(er->buffer, er->buf_size * er->rd_frame_size);
432*b9df5ad1SAndroid Build Coastguard Worker                             if (new_buf == NULL) {
433*b9df5ad1SAndroid Build Coastguard Worker                                 status = -ENOMEM;
434*b9df5ad1SAndroid Build Coastguard Worker                                 goto exit;
435*b9df5ad1SAndroid Build Coastguard Worker                             } else {
436*b9df5ad1SAndroid Build Coastguard Worker                                 er->buffer = new_buf;
437*b9df5ad1SAndroid Build Coastguard Worker                             }
438*b9df5ad1SAndroid Build Coastguard Worker                         }
439*b9df5ad1SAndroid Build Coastguard Worker 
440*b9df5ad1SAndroid Build Coastguard Worker                         if (offset > 0) {
441*b9df5ad1SAndroid Build Coastguard Worker                             memset((char *)er->buffer + previousFrameIn * er->rd_frame_size,
442*b9df5ad1SAndroid Build Coastguard Worker                                    0, offset * er->rd_frame_size);
443*b9df5ad1SAndroid Build Coastguard Worker                             ALOGV("echo_reference_read(): pushing ref buffer by [%d]", offset);
444*b9df5ad1SAndroid Build Coastguard Worker                         }
445*b9df5ad1SAndroid Build Coastguard Worker                     } else {
446*b9df5ad1SAndroid Build Coastguard Worker                         // More data available in the reference buffer than expected
447*b9df5ad1SAndroid Build Coastguard Worker                         offset = -offset;
448*b9df5ad1SAndroid Build Coastguard Worker                         if (offset > 0) {
449*b9df5ad1SAndroid Build Coastguard Worker                             memcpy(er->buffer, (char *)er->buffer + (offset * er->rd_frame_size),
450*b9df5ad1SAndroid Build Coastguard Worker                                    er->frames_in * er->rd_frame_size);
451*b9df5ad1SAndroid Build Coastguard Worker                             ALOGV("echo_reference_read(): shifting ref buffer by [%zu]",
452*b9df5ad1SAndroid Build Coastguard Worker                                   er->frames_in);
453*b9df5ad1SAndroid Build Coastguard Worker                         }
454*b9df5ad1SAndroid Build Coastguard Worker                     }
455*b9df5ad1SAndroid Build Coastguard Worker                 }
456*b9df5ad1SAndroid Build Coastguard Worker             } else {
457*b9df5ad1SAndroid Build Coastguard Worker                 er->delta_count = 0;
458*b9df5ad1SAndroid Build Coastguard Worker                 er->prev_delta_sign = 0;
459*b9df5ad1SAndroid Build Coastguard Worker                 ALOGV("echo_reference_read(): Constant EchoPathDelay - difference "
460*b9df5ad1SAndroid Build Coastguard Worker                         "between reference and DMA %" PRId64, deltaNs);
461*b9df5ad1SAndroid Build Coastguard Worker             }
462*b9df5ad1SAndroid Build Coastguard Worker         } else {
463*b9df5ad1SAndroid Build Coastguard Worker             ALOGV("echo_reference_read(): NEGATIVE expectedDelayNs[%" PRId64
464*b9df5ad1SAndroid Build Coastguard Worker                  "] = er->playback_delay[%d] + delayCapture[%d"
465*b9df5ad1SAndroid Build Coastguard Worker                  "] - timeDiff[%" PRId64 "]",
466*b9df5ad1SAndroid Build Coastguard Worker                  expectedDelayNs, er->playback_delay, buffer->delay_ns, timeDiff);
467*b9df5ad1SAndroid Build Coastguard Worker         }
468*b9df5ad1SAndroid Build Coastguard Worker     }
469*b9df5ad1SAndroid Build Coastguard Worker 
470*b9df5ad1SAndroid Build Coastguard Worker     if (er->frames_in < buffer->frame_count) {
471*b9df5ad1SAndroid Build Coastguard Worker         if (buffer->frame_count > er->buf_size) {
472*b9df5ad1SAndroid Build Coastguard Worker             er->buf_size = buffer->frame_count;
473*b9df5ad1SAndroid Build Coastguard Worker             ALOGV("echo_reference_read(): increasing buffer size to %zu", er->buf_size);
474*b9df5ad1SAndroid Build Coastguard Worker             void *new_buf  = realloc(er->buffer, er->buf_size * er->rd_frame_size);
475*b9df5ad1SAndroid Build Coastguard Worker             if (new_buf == NULL) {
476*b9df5ad1SAndroid Build Coastguard Worker                 status = -ENOMEM;
477*b9df5ad1SAndroid Build Coastguard Worker                 goto exit;
478*b9df5ad1SAndroid Build Coastguard Worker             } else {
479*b9df5ad1SAndroid Build Coastguard Worker                 er->buffer = new_buf;
480*b9df5ad1SAndroid Build Coastguard Worker             }
481*b9df5ad1SAndroid Build Coastguard Worker         }
482*b9df5ad1SAndroid Build Coastguard Worker         // filling up the reference buffer with 0s to match the expected delay.
483*b9df5ad1SAndroid Build Coastguard Worker         memset((char *)er->buffer + er->frames_in * er->rd_frame_size,
484*b9df5ad1SAndroid Build Coastguard Worker             0, (buffer->frame_count - er->frames_in) * er->rd_frame_size);
485*b9df5ad1SAndroid Build Coastguard Worker         er->frames_in = buffer->frame_count;
486*b9df5ad1SAndroid Build Coastguard Worker     }
487*b9df5ad1SAndroid Build Coastguard Worker 
488*b9df5ad1SAndroid Build Coastguard Worker     memcpy(buffer->raw,
489*b9df5ad1SAndroid Build Coastguard Worker            (char *)er->buffer,
490*b9df5ad1SAndroid Build Coastguard Worker            buffer->frame_count * er->rd_frame_size);
491*b9df5ad1SAndroid Build Coastguard Worker 
492*b9df5ad1SAndroid Build Coastguard Worker     er->frames_in -= buffer->frame_count;
493*b9df5ad1SAndroid Build Coastguard Worker     memcpy(er->buffer,
494*b9df5ad1SAndroid Build Coastguard Worker            (char *)er->buffer + buffer->frame_count * er->rd_frame_size,
495*b9df5ad1SAndroid Build Coastguard Worker            er->frames_in * er->rd_frame_size);
496*b9df5ad1SAndroid Build Coastguard Worker 
497*b9df5ad1SAndroid Build Coastguard Worker     // As the reference buffer is now time aligned to the microphone signal there is a zero delay
498*b9df5ad1SAndroid Build Coastguard Worker     buffer->delay_ns = 0;
499*b9df5ad1SAndroid Build Coastguard Worker 
500*b9df5ad1SAndroid Build Coastguard Worker     ALOGV("echo_reference_read() END %zu frames, total frames in %zu",
501*b9df5ad1SAndroid Build Coastguard Worker           buffer->frame_count, er->frames_in);
502*b9df5ad1SAndroid Build Coastguard Worker 
503*b9df5ad1SAndroid Build Coastguard Worker     pthread_cond_signal(&er->cond);
504*b9df5ad1SAndroid Build Coastguard Worker 
505*b9df5ad1SAndroid Build Coastguard Worker exit:
506*b9df5ad1SAndroid Build Coastguard Worker     pthread_mutex_unlock(&er->lock);
507*b9df5ad1SAndroid Build Coastguard Worker     return status;
508*b9df5ad1SAndroid Build Coastguard Worker }
509*b9df5ad1SAndroid Build Coastguard Worker 
510*b9df5ad1SAndroid Build Coastguard Worker 
create_echo_reference(audio_format_t rdFormat,uint32_t rdChannelCount,uint32_t rdSamplingRate,audio_format_t wrFormat,uint32_t wrChannelCount,uint32_t wrSamplingRate,struct echo_reference_itfe ** echo_reference)511*b9df5ad1SAndroid Build Coastguard Worker int create_echo_reference(audio_format_t rdFormat,
512*b9df5ad1SAndroid Build Coastguard Worker                             uint32_t rdChannelCount,
513*b9df5ad1SAndroid Build Coastguard Worker                             uint32_t rdSamplingRate,
514*b9df5ad1SAndroid Build Coastguard Worker                             audio_format_t wrFormat,
515*b9df5ad1SAndroid Build Coastguard Worker                             uint32_t wrChannelCount,
516*b9df5ad1SAndroid Build Coastguard Worker                             uint32_t wrSamplingRate,
517*b9df5ad1SAndroid Build Coastguard Worker                             struct echo_reference_itfe **echo_reference)
518*b9df5ad1SAndroid Build Coastguard Worker {
519*b9df5ad1SAndroid Build Coastguard Worker     struct echo_reference *er;
520*b9df5ad1SAndroid Build Coastguard Worker 
521*b9df5ad1SAndroid Build Coastguard Worker     ALOGV("create_echo_reference()");
522*b9df5ad1SAndroid Build Coastguard Worker 
523*b9df5ad1SAndroid Build Coastguard Worker     if (echo_reference == NULL) {
524*b9df5ad1SAndroid Build Coastguard Worker         return -EINVAL;
525*b9df5ad1SAndroid Build Coastguard Worker     }
526*b9df5ad1SAndroid Build Coastguard Worker 
527*b9df5ad1SAndroid Build Coastguard Worker     *echo_reference = NULL;
528*b9df5ad1SAndroid Build Coastguard Worker 
529*b9df5ad1SAndroid Build Coastguard Worker     if (rdFormat != AUDIO_FORMAT_PCM_16_BIT ||
530*b9df5ad1SAndroid Build Coastguard Worker             rdFormat != wrFormat) {
531*b9df5ad1SAndroid Build Coastguard Worker         ALOGW("create_echo_reference bad format rd %d, wr %d", rdFormat, wrFormat);
532*b9df5ad1SAndroid Build Coastguard Worker         return -EINVAL;
533*b9df5ad1SAndroid Build Coastguard Worker     }
534*b9df5ad1SAndroid Build Coastguard Worker     if ((rdChannelCount != 1 && rdChannelCount != 2) ||
535*b9df5ad1SAndroid Build Coastguard Worker             wrChannelCount != 2) {
536*b9df5ad1SAndroid Build Coastguard Worker         ALOGW("create_echo_reference bad channel count rd %d, wr %d", rdChannelCount,
537*b9df5ad1SAndroid Build Coastguard Worker                 wrChannelCount);
538*b9df5ad1SAndroid Build Coastguard Worker         return -EINVAL;
539*b9df5ad1SAndroid Build Coastguard Worker     }
540*b9df5ad1SAndroid Build Coastguard Worker 
541*b9df5ad1SAndroid Build Coastguard Worker     er = (struct echo_reference *)calloc(1, sizeof(struct echo_reference));
542*b9df5ad1SAndroid Build Coastguard Worker 
543*b9df5ad1SAndroid Build Coastguard Worker     er->itfe.read = echo_reference_read;
544*b9df5ad1SAndroid Build Coastguard Worker     er->itfe.write = echo_reference_write;
545*b9df5ad1SAndroid Build Coastguard Worker 
546*b9df5ad1SAndroid Build Coastguard Worker     er->state = ECHOREF_IDLE;
547*b9df5ad1SAndroid Build Coastguard Worker     er->rd_format = rdFormat;
548*b9df5ad1SAndroid Build Coastguard Worker     er->rd_channel_count = rdChannelCount;
549*b9df5ad1SAndroid Build Coastguard Worker     er->rd_sampling_rate = rdSamplingRate;
550*b9df5ad1SAndroid Build Coastguard Worker     er->wr_format = wrFormat;
551*b9df5ad1SAndroid Build Coastguard Worker     er->wr_channel_count = wrChannelCount;
552*b9df5ad1SAndroid Build Coastguard Worker     er->wr_sampling_rate = wrSamplingRate;
553*b9df5ad1SAndroid Build Coastguard Worker     er->rd_frame_size = audio_bytes_per_sample(rdFormat) * rdChannelCount;
554*b9df5ad1SAndroid Build Coastguard Worker     er->wr_frame_size = audio_bytes_per_sample(wrFormat) * wrChannelCount;
555*b9df5ad1SAndroid Build Coastguard Worker     *echo_reference = &er->itfe;
556*b9df5ad1SAndroid Build Coastguard Worker     return 0;
557*b9df5ad1SAndroid Build Coastguard Worker }
558*b9df5ad1SAndroid Build Coastguard Worker 
release_echo_reference(struct echo_reference_itfe * echo_reference)559*b9df5ad1SAndroid Build Coastguard Worker void release_echo_reference(struct echo_reference_itfe *echo_reference) {
560*b9df5ad1SAndroid Build Coastguard Worker     struct echo_reference *er = (struct echo_reference *)echo_reference;
561*b9df5ad1SAndroid Build Coastguard Worker 
562*b9df5ad1SAndroid Build Coastguard Worker     if (er == NULL) {
563*b9df5ad1SAndroid Build Coastguard Worker         return;
564*b9df5ad1SAndroid Build Coastguard Worker     }
565*b9df5ad1SAndroid Build Coastguard Worker 
566*b9df5ad1SAndroid Build Coastguard Worker     ALOGV("EchoReference dstor");
567*b9df5ad1SAndroid Build Coastguard Worker     echo_reference_reset_l(er);
568*b9df5ad1SAndroid Build Coastguard Worker     if (er->resampler != NULL) {
569*b9df5ad1SAndroid Build Coastguard Worker         release_resampler(er->resampler);
570*b9df5ad1SAndroid Build Coastguard Worker     }
571*b9df5ad1SAndroid Build Coastguard Worker     free(er);
572*b9df5ad1SAndroid Build Coastguard Worker }
573