• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2016 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  * Copied as it is from device/amlogic/generic/hal/audio/
17  */
18 
19 #define LOG_TAG "audio_hw_yukawa"
20 //#define LOG_NDEBUG 0
21 
22 #include <errno.h>
23 #include <inttypes.h>
24 #include <malloc.h>
25 #include <pthread.h>
26 #include <stdint.h>
27 #include <stdlib.h>
28 #include <sys/time.h>
29 #include <unistd.h>
30 #include <string.h>
31 
32 #include <log/log.h>
33 #include <cutils/str_parms.h>
34 #include <cutils/properties.h>
35 
36 #include <hardware/hardware.h>
37 #include <system/audio.h>
38 #include <hardware/audio.h>
39 
40 #include <audio_effects/effect_aec.h>
41 #include <audio_route/audio_route.h>
42 #include <audio_utils/clock.h>
43 #include <audio_utils/echo_reference.h>
44 #include <audio_utils/resampler.h>
45 #include <cutils/properties.h>
46 #include <hardware/audio_alsaops.h>
47 #include <hardware/audio_effect.h>
48 #include <sound/asound.h>
49 #include <tinyalsa/asoundlib.h>
50 
51 #include <sys/ioctl.h>
52 
53 #include "audio_aec.h"
54 #include "audio_hw.h"
55 
56 static int adev_get_mic_mute(const struct audio_hw_device* dev, bool* state);
57 static int adev_get_microphones(const struct audio_hw_device* dev,
58                                 struct audio_microphone_characteristic_t* mic_array,
59                                 size_t* mic_count);
60 static size_t out_get_buffer_size(const struct audio_stream* stream);
61 
is_aec_input(const struct alsa_stream_in * in)62 static bool is_aec_input(const struct alsa_stream_in* in) {
63     /* If AEC is in the app, only configure based on ECHO_REFERENCE spec.
64      * If AEC is in the HAL, configure using the given mic stream. */
65     bool aec_input = true;
66 #if !defined(AEC_HAL)
67     aec_input = (in->source == AUDIO_SOURCE_ECHO_REFERENCE);
68 #endif
69     return aec_input;
70 }
71 
get_audio_output_port(audio_devices_t devices)72 static int get_audio_output_port(audio_devices_t devices) {
73     /* Only HDMI out for now #FIXME */
74     return PORT_HDMI;
75 }
76 
get_audio_card(int direction,int port)77 static int get_audio_card(int direction, int port) {
78     struct pcm_params* params = NULL;
79     int card = 0;
80 
81     while (!params && card < 8) {
82 	/* Find the first input/output device that works */
83         params = pcm_params_get(card, port, direction);
84 	card++;
85     }
86     pcm_params_free(params);
87 
88     return card - 1;
89 }
90 
timestamp_adjust(struct timespec * ts,ssize_t frames,uint32_t sampling_rate)91 static void timestamp_adjust(struct timespec* ts, ssize_t frames, uint32_t sampling_rate) {
92     /* This function assumes the adjustment (in nsec) is less than the max value of long,
93      * which for 32-bit long this is 2^31 * 1e-9 seconds, slightly over 2 seconds.
94      * For 64-bit long it is  9e+9 seconds. */
95     long adj_nsec = (frames / (float) sampling_rate) * 1E9L;
96     ts->tv_nsec += adj_nsec;
97     while (ts->tv_nsec > 1E9L) {
98         ts->tv_sec++;
99         ts->tv_nsec -= 1E9L;
100     }
101     if (ts->tv_nsec < 0) {
102         ts->tv_sec--;
103         ts->tv_nsec += 1E9L;
104     }
105 }
106 
107 /* Helper function to get PCM hardware timestamp.
108  * Only the field 'timestamp' of argument 'ts' is updated. */
get_pcm_timestamp(struct pcm * pcm,uint32_t sample_rate,struct aec_info * info,bool isOutput)109 static int get_pcm_timestamp(struct pcm* pcm, uint32_t sample_rate, struct aec_info* info,
110                              bool isOutput) {
111     int ret = 0;
112     if (pcm_get_htimestamp(pcm, &info->available, &info->timestamp) < 0) {
113         ALOGE("Error getting PCM timestamp!");
114         info->timestamp.tv_sec = 0;
115         info->timestamp.tv_nsec = 0;
116         return -EINVAL;
117     }
118     ssize_t frames;
119     if (isOutput) {
120         frames = pcm_get_buffer_size(pcm) - info->available;
121     } else {
122         frames = -info->available; /* rewind timestamp */
123     }
124     timestamp_adjust(&info->timestamp, frames, sample_rate);
125     return ret;
126 }
127 
read_filter_from_file(const char * filename,int16_t * filter,int max_length)128 static int read_filter_from_file(const char* filename, int16_t* filter, int max_length) {
129     FILE* fp = fopen(filename, "r");
130     if (fp == NULL) {
131         ALOGI("%s: File %s not found.", __func__, filename);
132         return 0;
133     }
134     int num_taps = 0;
135     char* line = NULL;
136     size_t len = 0;
137     while (!feof(fp)) {
138         size_t size = getline(&line, &len, fp);
139         if ((line[0] == '#') || (size < 2)) {
140             continue;
141         }
142         int n = sscanf(line, "%" SCNd16 "\n", &filter[num_taps++]);
143         if (n < 1) {
144             ALOGE("Could not find coefficient %d! Exiting...", num_taps - 1);
145             return 0;
146         }
147         ALOGV("Coeff %d : %" PRId16, num_taps, filter[num_taps - 1]);
148         if (num_taps == max_length) {
149             ALOGI("%s: max tap length %d reached.", __func__, max_length);
150             break;
151         }
152     }
153     free(line);
154     fclose(fp);
155     return num_taps;
156 }
157 
out_set_eq(struct alsa_stream_out * out)158 static void out_set_eq(struct alsa_stream_out* out) {
159     out->speaker_eq = NULL;
160     int16_t* speaker_eq_coeffs = (int16_t*)calloc(SPEAKER_MAX_EQ_LENGTH, sizeof(int16_t));
161     if (speaker_eq_coeffs == NULL) {
162         ALOGE("%s: Failed to allocate speaker EQ", __func__);
163         return;
164     }
165     int num_taps = read_filter_from_file(SPEAKER_EQ_FILE, speaker_eq_coeffs, SPEAKER_MAX_EQ_LENGTH);
166     if (num_taps == 0) {
167         ALOGI("%s: Empty filter file or 0 taps set.", __func__);
168         free(speaker_eq_coeffs);
169         return;
170     }
171     out->speaker_eq = fir_init(
172             out->config.channels, FIR_SINGLE_FILTER, num_taps,
173             out_get_buffer_size(&out->stream.common) / out->config.channels / sizeof(int16_t),
174             speaker_eq_coeffs);
175     free(speaker_eq_coeffs);
176 }
177 
178 /* must be called with hw device and output stream mutexes locked */
start_output_stream(struct alsa_stream_out * out)179 static int start_output_stream(struct alsa_stream_out *out)
180 {
181     struct alsa_audio_device *adev = out->dev;
182 
183     /* default to low power: will be corrected in out_write if necessary before first write to
184      * tinyalsa.
185      */
186     out->write_threshold = PLAYBACK_PERIOD_COUNT * PLAYBACK_PERIOD_SIZE;
187     out->config.start_threshold = PLAYBACK_PERIOD_START_THRESHOLD * PLAYBACK_PERIOD_SIZE;
188     out->config.avail_min = PLAYBACK_PERIOD_SIZE;
189     out->unavailable = true;
190     unsigned int pcm_retry_count = PCM_OPEN_RETRIES;
191     int out_port = get_audio_output_port(out->devices);
192     int out_card = get_audio_card(PCM_OUT, out_port);
193 
194     while (1) {
195         out->pcm = pcm_open(out_card, out_port, PCM_OUT | PCM_MONOTONIC, &out->config);
196         if ((out->pcm != NULL) && pcm_is_ready(out->pcm)) {
197             break;
198         } else {
199             ALOGE("cannot open pcm_out driver: %s", pcm_get_error(out->pcm));
200             if (out->pcm != NULL) {
201                 pcm_close(out->pcm);
202                 out->pcm = NULL;
203             }
204             if (--pcm_retry_count == 0) {
205                 ALOGE("Failed to open pcm_out after %d tries", PCM_OPEN_RETRIES);
206                 return -ENODEV;
207             }
208             usleep(PCM_OPEN_WAIT_TIME_MS * 1000);
209         }
210     }
211     out->unavailable = false;
212     adev->active_output = out;
213     return 0;
214 }
215 
out_get_sample_rate(const struct audio_stream * stream)216 static uint32_t out_get_sample_rate(const struct audio_stream *stream)
217 {
218     struct alsa_stream_out *out = (struct alsa_stream_out *)stream;
219     return out->config.rate;
220 }
221 
out_set_sample_rate(struct audio_stream * stream,uint32_t rate)222 static int out_set_sample_rate(struct audio_stream *stream, uint32_t rate)
223 {
224     ALOGV("out_set_sample_rate: %d", 0);
225     return -ENOSYS;
226 }
227 
out_get_buffer_size(const struct audio_stream * stream)228 static size_t out_get_buffer_size(const struct audio_stream *stream)
229 {
230     ALOGV("out_get_buffer_size: %d", 4096);
231 
232     /* return the closest majoring multiple of 16 frames, as
233      * audioflinger expects audio buffers to be a multiple of 16 frames */
234     size_t size = PLAYBACK_PERIOD_SIZE;
235     size = ((size + 15) / 16) * 16;
236     return size * audio_stream_out_frame_size((struct audio_stream_out *)stream);
237 }
238 
out_get_channels(const struct audio_stream * stream)239 static audio_channel_mask_t out_get_channels(const struct audio_stream *stream)
240 {
241     ALOGV("out_get_channels");
242     struct alsa_stream_out *out = (struct alsa_stream_out *)stream;
243     return audio_channel_out_mask_from_count(out->config.channels);
244 }
245 
out_get_format(const struct audio_stream * stream)246 static audio_format_t out_get_format(const struct audio_stream *stream)
247 {
248     ALOGV("out_get_format");
249     struct alsa_stream_out *out = (struct alsa_stream_out *)stream;
250     return audio_format_from_pcm_format(out->config.format);
251 }
252 
out_set_format(struct audio_stream * stream,audio_format_t format)253 static int out_set_format(struct audio_stream *stream, audio_format_t format)
254 {
255     ALOGV("out_set_format: %d",format);
256     return -ENOSYS;
257 }
258 
do_output_standby(struct alsa_stream_out * out)259 static int do_output_standby(struct alsa_stream_out *out)
260 {
261     struct alsa_audio_device *adev = out->dev;
262 
263     fir_reset(out->speaker_eq);
264 
265     if (!out->standby) {
266         pcm_close(out->pcm);
267         out->pcm = NULL;
268         adev->active_output = NULL;
269         out->standby = 1;
270     }
271     aec_set_spk_running(adev->aec, false);
272     return 0;
273 }
274 
out_standby(struct audio_stream * stream)275 static int out_standby(struct audio_stream *stream)
276 {
277     ALOGV("out_standby");
278     struct alsa_stream_out *out = (struct alsa_stream_out *)stream;
279     int status;
280 
281     pthread_mutex_lock(&out->dev->lock);
282     pthread_mutex_lock(&out->lock);
283     status = do_output_standby(out);
284     pthread_mutex_unlock(&out->lock);
285     pthread_mutex_unlock(&out->dev->lock);
286     return status;
287 }
288 
out_dump(const struct audio_stream * stream,int fd)289 static int out_dump(const struct audio_stream *stream, int fd)
290 {
291     ALOGV("out_dump");
292     return 0;
293 }
294 
out_set_parameters(struct audio_stream * stream,const char * kvpairs)295 static int out_set_parameters(struct audio_stream *stream, const char *kvpairs)
296 {
297     ALOGV("out_set_parameters");
298     struct alsa_stream_out *out = (struct alsa_stream_out *)stream;
299     struct alsa_audio_device *adev = out->dev;
300     struct str_parms *parms;
301     char value[32];
302     int ret, val = 0;
303 
304     parms = str_parms_create_str(kvpairs);
305 
306     ret = str_parms_get_str(parms, AUDIO_PARAMETER_STREAM_ROUTING, value, sizeof(value));
307     if (ret >= 0) {
308         val = atoi(value);
309         pthread_mutex_lock(&adev->lock);
310         pthread_mutex_lock(&out->lock);
311         if (((out->devices & AUDIO_DEVICE_OUT_ALL) != val) && (val != 0)) {
312             out->devices &= ~AUDIO_DEVICE_OUT_ALL;
313             out->devices |= val;
314         }
315         pthread_mutex_unlock(&out->lock);
316         pthread_mutex_unlock(&adev->lock);
317     }
318 
319     str_parms_destroy(parms);
320     return 0;
321 }
322 
out_get_parameters(const struct audio_stream * stream,const char * keys)323 static char * out_get_parameters(const struct audio_stream *stream, const char *keys)
324 {
325     ALOGV("out_get_parameters");
326     return strdup("");
327 }
328 
out_get_latency(const struct audio_stream_out * stream)329 static uint32_t out_get_latency(const struct audio_stream_out *stream)
330 {
331     ALOGV("out_get_latency");
332     struct alsa_stream_out *out = (struct alsa_stream_out *)stream;
333     return (PLAYBACK_PERIOD_SIZE * PLAYBACK_PERIOD_COUNT * 1000) / out->config.rate;
334 }
335 
out_set_volume(struct audio_stream_out * stream,float left,float right)336 static int out_set_volume(struct audio_stream_out *stream, float left,
337         float right)
338 {
339     ALOGV("out_set_volume: Left:%f Right:%f", left, right);
340     return -ENOSYS;
341 }
342 
out_write(struct audio_stream_out * stream,const void * buffer,size_t bytes)343 static ssize_t out_write(struct audio_stream_out *stream, const void* buffer,
344         size_t bytes)
345 {
346     int ret;
347     struct alsa_stream_out *out = (struct alsa_stream_out *)stream;
348     struct alsa_audio_device *adev = out->dev;
349     size_t frame_size = audio_stream_out_frame_size(stream);
350     size_t out_frames = bytes / frame_size;
351 
352     ALOGV("%s: devices: %d, bytes %zu", __func__, out->devices, bytes);
353 
354     /* acquiring hw device mutex systematically is useful if a low priority thread is waiting
355      * on the output stream mutex - e.g. executing select_mode() while holding the hw device
356      * mutex
357      */
358     pthread_mutex_lock(&adev->lock);
359     pthread_mutex_lock(&out->lock);
360     if (out->standby) {
361         ret = start_output_stream(out);
362         if (ret != 0) {
363             pthread_mutex_unlock(&adev->lock);
364             goto exit;
365         }
366         out->standby = 0;
367         aec_set_spk_running(adev->aec, true);
368     }
369 
370     pthread_mutex_unlock(&adev->lock);
371 
372     if (out->speaker_eq != NULL) {
373         fir_process_interleaved(out->speaker_eq, (int16_t*)buffer, (int16_t*)buffer, out_frames);
374     }
375 
376     ret = pcm_write(out->pcm, buffer, out_frames * frame_size);
377     if (ret == 0) {
378         out->frames_written += out_frames;
379 
380         struct aec_info info;
381         get_pcm_timestamp(out->pcm, out->config.rate, &info, true /*isOutput*/);
382         out->timestamp = info.timestamp;
383         info.bytes = out_frames * frame_size;
384         int aec_ret = write_to_reference_fifo(adev->aec, (void *)buffer, &info);
385         if (aec_ret) {
386             ALOGE("AEC: Write to speaker loopback FIFO failed!");
387         }
388     }
389 
390 exit:
391     pthread_mutex_unlock(&out->lock);
392 
393     if (ret != 0) {
394         usleep((int64_t)bytes * 1000000 / audio_stream_out_frame_size(stream) /
395                 out_get_sample_rate(&stream->common));
396     }
397 
398     return bytes;
399 }
400 
out_get_render_position(const struct audio_stream_out * stream,uint32_t * dsp_frames)401 static int out_get_render_position(const struct audio_stream_out *stream,
402         uint32_t *dsp_frames)
403 {
404     ALOGV("out_get_render_position: dsp_frames: %p", dsp_frames);
405     return -ENOSYS;
406 }
407 
out_get_presentation_position(const struct audio_stream_out * stream,uint64_t * frames,struct timespec * timestamp)408 static int out_get_presentation_position(const struct audio_stream_out *stream,
409                                    uint64_t *frames, struct timespec *timestamp)
410 {
411     if (stream == NULL || frames == NULL || timestamp == NULL) {
412         return -EINVAL;
413     }
414     struct alsa_stream_out* out = (struct alsa_stream_out*)stream;
415 
416     *frames = out->frames_written;
417     *timestamp = out->timestamp;
418     ALOGV("%s: frames: %" PRIu64 ", timestamp (nsec): %" PRIu64, __func__, *frames,
419           audio_utils_ns_from_timespec(timestamp));
420 
421     return 0;
422 }
423 
424 
out_add_audio_effect(const struct audio_stream * stream,effect_handle_t effect)425 static int out_add_audio_effect(const struct audio_stream *stream, effect_handle_t effect)
426 {
427     ALOGV("out_add_audio_effect: %p", effect);
428     return 0;
429 }
430 
out_remove_audio_effect(const struct audio_stream * stream,effect_handle_t effect)431 static int out_remove_audio_effect(const struct audio_stream *stream, effect_handle_t effect)
432 {
433     ALOGV("out_remove_audio_effect: %p", effect);
434     return 0;
435 }
436 
out_get_next_write_timestamp(const struct audio_stream_out * stream,int64_t * timestamp)437 static int out_get_next_write_timestamp(const struct audio_stream_out *stream,
438         int64_t *timestamp)
439 {
440     *timestamp = 0;
441     ALOGV("out_get_next_write_timestamp: %ld", (long int)(*timestamp));
442     return -ENOSYS;
443 }
444 
445 /** audio_stream_in implementation **/
446 
447 /* must be called with hw device and input stream mutexes locked */
start_input_stream(struct alsa_stream_in * in)448 static int start_input_stream(struct alsa_stream_in *in)
449 {
450     struct alsa_audio_device *adev = in->dev;
451     in->unavailable = true;
452     unsigned int pcm_retry_count = PCM_OPEN_RETRIES;
453     int in_card = get_audio_card(PCM_IN, PORT_BUILTIN_MIC);
454 
455     while (1) {
456         in->pcm = pcm_open(in_card, PORT_BUILTIN_MIC, PCM_IN | PCM_MONOTONIC, &in->config);
457         if ((in->pcm != NULL) && pcm_is_ready(in->pcm)) {
458             break;
459         } else {
460             ALOGE("cannot open pcm_in driver: %s", pcm_get_error(in->pcm));
461             if (in->pcm != NULL) {
462                 pcm_close(in->pcm);
463                 in->pcm = NULL;
464             }
465             if (--pcm_retry_count == 0) {
466                 ALOGE("Failed to open pcm_in after %d tries", PCM_OPEN_RETRIES);
467                 return -ENODEV;
468             }
469             usleep(PCM_OPEN_WAIT_TIME_MS * 1000);
470         }
471     }
472     in->unavailable = false;
473     adev->active_input = in;
474     return 0;
475 }
476 
get_mic_characteristics(struct audio_microphone_characteristic_t * mic_data,size_t * mic_count)477 static void get_mic_characteristics(struct audio_microphone_characteristic_t* mic_data,
478                                     size_t* mic_count) {
479     *mic_count = 1;
480     memset(mic_data, 0, sizeof(struct audio_microphone_characteristic_t));
481     strlcpy(mic_data->device_id, "builtin_mic", AUDIO_MICROPHONE_ID_MAX_LEN - 1);
482     strlcpy(mic_data->address, "top", AUDIO_DEVICE_MAX_ADDRESS_LEN - 1);
483     memset(mic_data->channel_mapping, AUDIO_MICROPHONE_CHANNEL_MAPPING_UNUSED,
484            sizeof(mic_data->channel_mapping));
485     mic_data->device = AUDIO_DEVICE_IN_BUILTIN_MIC;
486     mic_data->sensitivity = -37.0;
487     mic_data->max_spl = AUDIO_MICROPHONE_SPL_UNKNOWN;
488     mic_data->min_spl = AUDIO_MICROPHONE_SPL_UNKNOWN;
489     mic_data->orientation.x = 0.0f;
490     mic_data->orientation.y = 0.0f;
491     mic_data->orientation.z = 0.0f;
492     mic_data->geometric_location.x = AUDIO_MICROPHONE_COORDINATE_UNKNOWN;
493     mic_data->geometric_location.y = AUDIO_MICROPHONE_COORDINATE_UNKNOWN;
494     mic_data->geometric_location.z = AUDIO_MICROPHONE_COORDINATE_UNKNOWN;
495 }
496 
in_get_sample_rate(const struct audio_stream * stream)497 static uint32_t in_get_sample_rate(const struct audio_stream *stream)
498 {
499     struct alsa_stream_in *in = (struct alsa_stream_in *)stream;
500     return in->config.rate;
501 }
502 
in_set_sample_rate(struct audio_stream * stream,uint32_t rate)503 static int in_set_sample_rate(struct audio_stream *stream, uint32_t rate)
504 {
505     ALOGV("in_set_sample_rate: %d", rate);
506     return -ENOSYS;
507 }
508 
get_input_buffer_size(size_t frames,audio_format_t format,audio_channel_mask_t channel_mask)509 static size_t get_input_buffer_size(size_t frames, audio_format_t format,
510                                     audio_channel_mask_t channel_mask) {
511     /* return the closest majoring multiple of 16 frames, as
512      * audioflinger expects audio buffers to be a multiple of 16 frames */
513     frames = ((frames + 15) / 16) * 16;
514     size_t bytes_per_frame = audio_channel_count_from_in_mask(channel_mask) *
515                             audio_bytes_per_sample(format);
516     size_t buffer_size = frames * bytes_per_frame;
517     return buffer_size;
518 }
519 
in_get_channels(const struct audio_stream * stream)520 static audio_channel_mask_t in_get_channels(const struct audio_stream *stream)
521 {
522     struct alsa_stream_in *in = (struct alsa_stream_in *)stream;
523     ALOGV("in_get_channels: %d", in->config.channels);
524     return audio_channel_in_mask_from_count(in->config.channels);
525 }
526 
in_get_format(const struct audio_stream * stream)527 static audio_format_t in_get_format(const struct audio_stream *stream)
528 {
529     struct alsa_stream_in *in = (struct alsa_stream_in *)stream;
530     ALOGV("in_get_format: %d", in->config.format);
531     return audio_format_from_pcm_format(in->config.format);
532 }
533 
in_set_format(struct audio_stream * stream,audio_format_t format)534 static int in_set_format(struct audio_stream *stream, audio_format_t format)
535 {
536     return -ENOSYS;
537 }
538 
in_get_buffer_size(const struct audio_stream * stream)539 static size_t in_get_buffer_size(const struct audio_stream *stream)
540 {
541     struct alsa_stream_in* in = (struct alsa_stream_in*)stream;
542     size_t frames = CAPTURE_PERIOD_SIZE;
543     if (in->source == AUDIO_SOURCE_ECHO_REFERENCE) {
544         frames = CAPTURE_PERIOD_SIZE * PLAYBACK_CODEC_SAMPLING_RATE / CAPTURE_CODEC_SAMPLING_RATE;
545     }
546 
547     size_t buffer_size =
548             get_input_buffer_size(frames, stream->get_format(stream), stream->get_channels(stream));
549     ALOGV("in_get_buffer_size: %zu", buffer_size);
550     return buffer_size;
551 }
552 
in_get_active_microphones(const struct audio_stream_in * stream,struct audio_microphone_characteristic_t * mic_array,size_t * mic_count)553 static int in_get_active_microphones(const struct audio_stream_in* stream,
554                                      struct audio_microphone_characteristic_t* mic_array,
555                                      size_t* mic_count) {
556     ALOGV("in_get_active_microphones");
557     if ((mic_array == NULL) || (mic_count == NULL)) {
558         return -EINVAL;
559     }
560     struct alsa_stream_in* in = (struct alsa_stream_in*)stream;
561     struct audio_hw_device* dev = (struct audio_hw_device*)in->dev;
562     bool mic_muted = false;
563     adev_get_mic_mute(dev, &mic_muted);
564     if ((in->source == AUDIO_SOURCE_ECHO_REFERENCE) || mic_muted) {
565         *mic_count = 0;
566         return 0;
567     }
568     adev_get_microphones(dev, mic_array, mic_count);
569     return 0;
570 }
571 
do_input_standby(struct alsa_stream_in * in)572 static int do_input_standby(struct alsa_stream_in *in)
573 {
574     struct alsa_audio_device *adev = in->dev;
575 
576     if (!in->standby) {
577         pcm_close(in->pcm);
578         in->pcm = NULL;
579         adev->active_input = NULL;
580         in->standby = true;
581     }
582     return 0;
583 }
584 
in_standby(struct audio_stream * stream)585 static int in_standby(struct audio_stream *stream)
586 {
587     struct alsa_stream_in *in = (struct alsa_stream_in *)stream;
588     int status;
589 
590     pthread_mutex_lock(&in->lock);
591     pthread_mutex_lock(&in->dev->lock);
592     status = do_input_standby(in);
593     pthread_mutex_unlock(&in->dev->lock);
594     pthread_mutex_unlock(&in->lock);
595     return status;
596 }
597 
in_dump(const struct audio_stream * stream,int fd)598 static int in_dump(const struct audio_stream *stream, int fd)
599 {
600     struct alsa_stream_in* in = (struct alsa_stream_in*)stream;
601     if (in->source == AUDIO_SOURCE_ECHO_REFERENCE) {
602         return 0;
603     }
604 
605     struct audio_microphone_characteristic_t mic_array[AUDIO_MICROPHONE_MAX_COUNT];
606     size_t mic_count;
607 
608     get_mic_characteristics(mic_array, &mic_count);
609 
610     dprintf(fd, "  Microphone count: %zd\n", mic_count);
611     size_t idx;
612     for (idx = 0; idx < mic_count; idx++) {
613         dprintf(fd, "  Microphone: %zd\n", idx);
614         dprintf(fd, "    Address: %s\n", mic_array[idx].address);
615         dprintf(fd, "    Device: %d\n", mic_array[idx].device);
616         dprintf(fd, "    Sensitivity (dB): %.2f\n", mic_array[idx].sensitivity);
617     }
618 
619     return 0;
620 }
621 
in_set_parameters(struct audio_stream * stream,const char * kvpairs)622 static int in_set_parameters(struct audio_stream *stream, const char *kvpairs)
623 {
624     return 0;
625 }
626 
in_get_parameters(const struct audio_stream * stream,const char * keys)627 static char * in_get_parameters(const struct audio_stream *stream,
628         const char *keys)
629 {
630     return strdup("");
631 }
632 
in_set_gain(struct audio_stream_in * stream,float gain)633 static int in_set_gain(struct audio_stream_in *stream, float gain)
634 {
635     return 0;
636 }
637 
in_read(struct audio_stream_in * stream,void * buffer,size_t bytes)638 static ssize_t in_read(struct audio_stream_in *stream, void* buffer,
639         size_t bytes)
640 {
641     int ret;
642     struct alsa_stream_in *in = (struct alsa_stream_in *)stream;
643     struct alsa_audio_device *adev = in->dev;
644     size_t frame_size = audio_stream_in_frame_size(stream);
645     size_t in_frames = bytes / frame_size;
646 
647     ALOGV("in_read: stream: %d, bytes %zu", in->source, bytes);
648 
649     /* Special handling for Echo Reference: simply get the reference from FIFO.
650      * The format and sample rate should be specified by arguments to adev_open_input_stream. */
651     if (in->source == AUDIO_SOURCE_ECHO_REFERENCE) {
652         struct aec_info info;
653         info.bytes = bytes;
654 
655         const uint64_t time_increment_nsec = (uint64_t)bytes * NANOS_PER_SECOND /
656                                              audio_stream_in_frame_size(stream) /
657                                              in_get_sample_rate(&stream->common);
658         if (!aec_get_spk_running(adev->aec)) {
659             if (in->timestamp_nsec == 0) {
660                 struct timespec now;
661                 clock_gettime(CLOCK_MONOTONIC, &now);
662                 const uint64_t timestamp_nsec = audio_utils_ns_from_timespec(&now);
663                 in->timestamp_nsec = timestamp_nsec;
664             } else {
665                 in->timestamp_nsec += time_increment_nsec;
666             }
667             memset(buffer, 0, bytes);
668             const uint64_t time_increment_usec = time_increment_nsec / 1000;
669             usleep(time_increment_usec);
670         } else {
671             int ref_ret = get_reference_samples(adev->aec, buffer, &info);
672             if ((ref_ret) || (info.timestamp_usec == 0)) {
673                 memset(buffer, 0, bytes);
674                 in->timestamp_nsec += time_increment_nsec;
675             } else {
676                 in->timestamp_nsec = 1000 * info.timestamp_usec;
677             }
678         }
679         in->frames_read += in_frames;
680 
681 #if DEBUG_AEC
682         FILE* fp_ref = fopen("/data/local/traces/aec_ref.pcm", "a+");
683         if (fp_ref) {
684             fwrite((char*)buffer, 1, bytes, fp_ref);
685             fclose(fp_ref);
686         } else {
687             ALOGE("AEC debug: Could not open file aec_ref.pcm!");
688         }
689         FILE* fp_ref_ts = fopen("/data/local/traces/aec_ref_timestamps.txt", "a+");
690         if (fp_ref_ts) {
691             fprintf(fp_ref_ts, "%" PRIu64 "\n", in->timestamp_nsec);
692             fclose(fp_ref_ts);
693         } else {
694             ALOGE("AEC debug: Could not open file aec_ref_timestamps.txt!");
695         }
696 #endif
697         return info.bytes;
698     }
699 
700     /* Microphone input stream read */
701 
702     /* acquiring hw device mutex systematically is useful if a low priority thread is waiting
703      * on the input stream mutex - e.g. executing select_mode() while holding the hw device
704      * mutex
705      */
706     pthread_mutex_lock(&in->lock);
707     pthread_mutex_lock(&adev->lock);
708     if (in->standby) {
709         ret = start_input_stream(in);
710         if (ret != 0) {
711             pthread_mutex_unlock(&adev->lock);
712             ALOGE("start_input_stream failed with code %d", ret);
713             goto exit;
714         }
715         in->standby = false;
716     }
717 
718     pthread_mutex_unlock(&adev->lock);
719 
720     ret = pcm_read(in->pcm, buffer, in_frames * frame_size);
721     struct aec_info info;
722     get_pcm_timestamp(in->pcm, in->config.rate, &info, false /*isOutput*/);
723     if (ret == 0) {
724         in->frames_read += in_frames;
725         in->timestamp_nsec = audio_utils_ns_from_timespec(&info.timestamp);
726     }
727     else {
728         ALOGE("pcm_read failed with code %d", ret);
729     }
730 
731 exit:
732     pthread_mutex_unlock(&in->lock);
733 
734     bool mic_muted = false;
735     adev_get_mic_mute((struct audio_hw_device*)adev, &mic_muted);
736     if (mic_muted) {
737         memset(buffer, 0, bytes);
738     }
739 
740     if (ret != 0) {
741         usleep((int64_t)bytes * 1000000 / audio_stream_in_frame_size(stream) /
742                 in_get_sample_rate(&stream->common));
743     } else {
744         /* Process AEC if available */
745         /* TODO move to a separate thread */
746         if (!mic_muted) {
747             info.bytes = bytes;
748             int aec_ret = process_aec(adev->aec, buffer, &info);
749             if (aec_ret) {
750                 ALOGE("process_aec returned error code %d", aec_ret);
751             }
752         }
753     }
754 
755 #if DEBUG_AEC && !defined(AEC_HAL)
756     FILE* fp_in = fopen("/data/local/traces/aec_in.pcm", "a+");
757     if (fp_in) {
758         fwrite((char*)buffer, 1, bytes, fp_in);
759         fclose(fp_in);
760     } else {
761         ALOGE("AEC debug: Could not open file aec_in.pcm!");
762     }
763     FILE* fp_mic_ts = fopen("/data/local/traces/aec_in_timestamps.txt", "a+");
764     if (fp_mic_ts) {
765         fprintf(fp_mic_ts, "%" PRIu64 "\n", in->timestamp_nsec);
766         fclose(fp_mic_ts);
767     } else {
768         ALOGE("AEC debug: Could not open file aec_in_timestamps.txt!");
769     }
770 #endif
771 
772     return bytes;
773 }
774 
in_get_capture_position(const struct audio_stream_in * stream,int64_t * frames,int64_t * time)775 static int in_get_capture_position(const struct audio_stream_in* stream, int64_t* frames,
776                                    int64_t* time) {
777     if (stream == NULL || frames == NULL || time == NULL) {
778         return -EINVAL;
779     }
780     struct alsa_stream_in* in = (struct alsa_stream_in*)stream;
781 
782     *frames = in->frames_read;
783     *time = in->timestamp_nsec;
784     ALOGV("%s: source: %d, timestamp (nsec): %" PRIu64, __func__, in->source, *time);
785 
786     return 0;
787 }
788 
in_get_input_frames_lost(struct audio_stream_in * stream)789 static uint32_t in_get_input_frames_lost(struct audio_stream_in *stream)
790 {
791     return 0;
792 }
793 
in_add_audio_effect(const struct audio_stream * stream,effect_handle_t effect)794 static int in_add_audio_effect(const struct audio_stream *stream, effect_handle_t effect)
795 {
796     return 0;
797 }
798 
in_remove_audio_effect(const struct audio_stream * stream,effect_handle_t effect)799 static int in_remove_audio_effect(const struct audio_stream *stream, effect_handle_t effect)
800 {
801     return 0;
802 }
803 
adev_open_output_stream(struct audio_hw_device * dev,audio_io_handle_t handle,audio_devices_t devices,audio_output_flags_t flags,struct audio_config * config,struct audio_stream_out ** stream_out,const char * address __unused)804 static int adev_open_output_stream(struct audio_hw_device *dev,
805         audio_io_handle_t handle,
806         audio_devices_t devices,
807         audio_output_flags_t flags,
808         struct audio_config *config,
809         struct audio_stream_out **stream_out,
810         const char *address __unused)
811 {
812     ALOGV("adev_open_output_stream...");
813 
814     struct alsa_audio_device *ladev = (struct alsa_audio_device *)dev;
815     int out_port = get_audio_output_port(devices);
816     int out_card = get_audio_card(PCM_OUT, out_port);
817     struct pcm_params* params = pcm_params_get(out_card, out_port, PCM_OUT);
818     if (!params) {
819         return -ENOSYS;
820     }
821 
822     struct alsa_stream_out* out =
823             (struct alsa_stream_out*)calloc(1, sizeof(struct alsa_stream_out));
824     if (!out) {
825         return -ENOMEM;
826     }
827 
828     out->stream.common.get_sample_rate = out_get_sample_rate;
829     out->stream.common.set_sample_rate = out_set_sample_rate;
830     out->stream.common.get_buffer_size = out_get_buffer_size;
831     out->stream.common.get_channels = out_get_channels;
832     out->stream.common.get_format = out_get_format;
833     out->stream.common.set_format = out_set_format;
834     out->stream.common.standby = out_standby;
835     out->stream.common.dump = out_dump;
836     out->stream.common.set_parameters = out_set_parameters;
837     out->stream.common.get_parameters = out_get_parameters;
838     out->stream.common.add_audio_effect = out_add_audio_effect;
839     out->stream.common.remove_audio_effect = out_remove_audio_effect;
840     out->stream.get_latency = out_get_latency;
841     out->stream.set_volume = out_set_volume;
842     out->stream.write = out_write;
843     out->stream.get_render_position = out_get_render_position;
844     out->stream.get_next_write_timestamp = out_get_next_write_timestamp;
845     out->stream.get_presentation_position = out_get_presentation_position;
846 
847     out->config.channels = CHANNEL_STEREO;
848     out->config.rate = PLAYBACK_CODEC_SAMPLING_RATE;
849     out->config.format = PCM_FORMAT_S16_LE;
850     out->config.period_size = PLAYBACK_PERIOD_SIZE;
851     out->config.period_count = PLAYBACK_PERIOD_COUNT;
852 
853     if (out->config.rate != config->sample_rate ||
854            audio_channel_count_from_out_mask(config->channel_mask) != CHANNEL_STEREO ||
855                out->config.format !=  pcm_format_from_audio_format(config->format) ) {
856         config->sample_rate = out->config.rate;
857         config->format = audio_format_from_pcm_format(out->config.format);
858         config->channel_mask = audio_channel_out_mask_from_count(CHANNEL_STEREO);
859         goto error_1;
860     }
861 
862     ALOGI("adev_open_output_stream selects channels=%d rate=%d format=%d, devices=%d",
863           out->config.channels, out->config.rate, out->config.format, devices);
864 
865     out->dev = ladev;
866     out->standby = 1;
867     out->unavailable = false;
868     out->devices = devices;
869 
870     config->format = out_get_format(&out->stream.common);
871     config->channel_mask = out_get_channels(&out->stream.common);
872     config->sample_rate = out_get_sample_rate(&out->stream.common);
873 
874     out->speaker_eq = NULL;
875     if (out_port == PORT_INTERNAL_SPEAKER) {
876         out_set_eq(out);
877         if (out->speaker_eq == NULL) {
878             ALOGE("%s: Failed to initialize speaker EQ", __func__);
879         }
880     }
881 
882     int aec_ret = init_aec_reference_config(ladev->aec, out);
883     if (aec_ret) {
884         ALOGE("AEC: Speaker config init failed!");
885         goto error_2;
886     }
887 
888     *stream_out = &out->stream;
889     return 0;
890 
891 error_2:
892     fir_release(out->speaker_eq);
893 error_1:
894     free(out);
895     return -EINVAL;
896 }
897 
adev_close_output_stream(struct audio_hw_device * dev,struct audio_stream_out * stream)898 static void adev_close_output_stream(struct audio_hw_device *dev,
899         struct audio_stream_out *stream)
900 {
901     ALOGV("adev_close_output_stream...");
902     struct alsa_audio_device *adev = (struct alsa_audio_device *)dev;
903     destroy_aec_reference_config(adev->aec);
904     struct alsa_stream_out* out = (struct alsa_stream_out*)stream;
905     fir_release(out->speaker_eq);
906     free(stream);
907 }
908 
adev_set_parameters(struct audio_hw_device * dev,const char * kvpairs)909 static int adev_set_parameters(struct audio_hw_device *dev, const char *kvpairs)
910 {
911     ALOGV("adev_set_parameters");
912     return -ENOSYS;
913 }
914 
adev_get_parameters(const struct audio_hw_device * dev,const char * keys)915 static char * adev_get_parameters(const struct audio_hw_device *dev,
916         const char *keys)
917 {
918     ALOGV("adev_get_parameters");
919     return strdup("");
920 }
921 
adev_get_microphones(const struct audio_hw_device * dev,struct audio_microphone_characteristic_t * mic_array,size_t * mic_count)922 static int adev_get_microphones(const struct audio_hw_device* dev,
923                                 struct audio_microphone_characteristic_t* mic_array,
924                                 size_t* mic_count) {
925     ALOGV("adev_get_microphones");
926     if ((mic_array == NULL) || (mic_count == NULL)) {
927         return -EINVAL;
928     }
929     get_mic_characteristics(mic_array, mic_count);
930     return 0;
931 }
932 
adev_init_check(const struct audio_hw_device * dev)933 static int adev_init_check(const struct audio_hw_device *dev)
934 {
935     ALOGV("adev_init_check");
936     return 0;
937 }
938 
adev_set_voice_volume(struct audio_hw_device * dev,float volume)939 static int adev_set_voice_volume(struct audio_hw_device *dev, float volume)
940 {
941     ALOGV("adev_set_voice_volume: %f", volume);
942     return -ENOSYS;
943 }
944 
adev_set_master_volume(struct audio_hw_device * dev,float volume)945 static int adev_set_master_volume(struct audio_hw_device *dev, float volume)
946 {
947     ALOGV("adev_set_master_volume: %f", volume);
948     return -ENOSYS;
949 }
950 
adev_get_master_volume(struct audio_hw_device * dev,float * volume)951 static int adev_get_master_volume(struct audio_hw_device *dev, float *volume)
952 {
953     ALOGV("adev_get_master_volume: %f", *volume);
954     return -ENOSYS;
955 }
956 
adev_set_master_mute(struct audio_hw_device * dev,bool muted)957 static int adev_set_master_mute(struct audio_hw_device *dev, bool muted)
958 {
959     ALOGV("adev_set_master_mute: %d", muted);
960     return -ENOSYS;
961 }
962 
adev_get_master_mute(struct audio_hw_device * dev,bool * muted)963 static int adev_get_master_mute(struct audio_hw_device *dev, bool *muted)
964 {
965     ALOGV("adev_get_master_mute: %d", *muted);
966     return -ENOSYS;
967 }
968 
adev_set_mode(struct audio_hw_device * dev,audio_mode_t mode)969 static int adev_set_mode(struct audio_hw_device *dev, audio_mode_t mode)
970 {
971     ALOGV("adev_set_mode: %d", mode);
972     return 0;
973 }
974 
adev_set_mic_mute(struct audio_hw_device * dev,bool state)975 static int adev_set_mic_mute(struct audio_hw_device *dev, bool state)
976 {
977     ALOGV("adev_set_mic_mute: %d",state);
978     struct alsa_audio_device *adev = (struct alsa_audio_device *)dev;
979     pthread_mutex_lock(&adev->lock);
980     adev->mic_mute = state;
981     pthread_mutex_unlock(&adev->lock);
982     return 0;
983 }
984 
adev_get_mic_mute(const struct audio_hw_device * dev,bool * state)985 static int adev_get_mic_mute(const struct audio_hw_device *dev, bool *state)
986 {
987     ALOGV("adev_get_mic_mute");
988     struct alsa_audio_device *adev = (struct alsa_audio_device *)dev;
989     pthread_mutex_lock(&adev->lock);
990     *state = adev->mic_mute;
991     pthread_mutex_unlock(&adev->lock);
992     return 0;
993 }
994 
adev_get_input_buffer_size(const struct audio_hw_device * dev,const struct audio_config * config)995 static size_t adev_get_input_buffer_size(const struct audio_hw_device *dev,
996         const struct audio_config *config)
997 {
998     size_t buffer_size =
999             get_input_buffer_size(CAPTURE_PERIOD_SIZE, config->format, config->channel_mask);
1000     ALOGV("adev_get_input_buffer_size: %zu", buffer_size);
1001     return buffer_size;
1002 }
1003 
adev_open_input_stream(struct audio_hw_device * dev,audio_io_handle_t handle,audio_devices_t devices,struct audio_config * config,struct audio_stream_in ** stream_in,audio_input_flags_t flags __unused,const char * address __unused,audio_source_t source)1004 static int adev_open_input_stream(struct audio_hw_device* dev, audio_io_handle_t handle,
1005                                   audio_devices_t devices, struct audio_config* config,
1006                                   struct audio_stream_in** stream_in,
1007                                   audio_input_flags_t flags __unused, const char* address __unused,
1008                                   audio_source_t source) {
1009     ALOGV("adev_open_input_stream...");
1010 
1011     struct alsa_audio_device *ladev = (struct alsa_audio_device *)dev;
1012 
1013     int in_card = get_audio_card(PCM_IN, PORT_BUILTIN_MIC);
1014     struct pcm_params* params = pcm_params_get(in_card, PORT_BUILTIN_MIC, PCM_IN);
1015     if (!params) {
1016         return -ENOSYS;
1017     }
1018 
1019     struct alsa_stream_in* in = (struct alsa_stream_in*)calloc(1, sizeof(struct alsa_stream_in));
1020     if (!in) {
1021         return -ENOMEM;
1022     }
1023 
1024     in->stream.common.get_sample_rate = in_get_sample_rate;
1025     in->stream.common.set_sample_rate = in_set_sample_rate;
1026     in->stream.common.get_buffer_size = in_get_buffer_size;
1027     in->stream.common.get_channels = in_get_channels;
1028     in->stream.common.get_format = in_get_format;
1029     in->stream.common.set_format = in_set_format;
1030     in->stream.common.standby = in_standby;
1031     in->stream.common.dump = in_dump;
1032     in->stream.common.set_parameters = in_set_parameters;
1033     in->stream.common.get_parameters = in_get_parameters;
1034     in->stream.common.add_audio_effect = in_add_audio_effect;
1035     in->stream.common.remove_audio_effect = in_remove_audio_effect;
1036     in->stream.set_gain = in_set_gain;
1037     in->stream.read = in_read;
1038     in->stream.get_input_frames_lost = in_get_input_frames_lost;
1039     in->stream.get_capture_position = in_get_capture_position;
1040     in->stream.get_active_microphones = in_get_active_microphones;
1041 
1042     in->config.channels = CHANNEL_STEREO;
1043     if (source == AUDIO_SOURCE_ECHO_REFERENCE) {
1044         in->config.rate = PLAYBACK_CODEC_SAMPLING_RATE;
1045     } else {
1046         in->config.rate = CAPTURE_CODEC_SAMPLING_RATE;
1047     }
1048     in->config.format = PCM_FORMAT_S32_LE;
1049     in->config.period_size = CAPTURE_PERIOD_SIZE;
1050     in->config.period_count = CAPTURE_PERIOD_COUNT;
1051 
1052     if (in->config.rate != config->sample_rate ||
1053            audio_channel_count_from_in_mask(config->channel_mask) != CHANNEL_STEREO ||
1054                in->config.format !=  pcm_format_from_audio_format(config->format) ) {
1055         config->format = in_get_format(&in->stream.common);
1056         config->channel_mask = in_get_channels(&in->stream.common);
1057         config->sample_rate = in_get_sample_rate(&in->stream.common);
1058         goto error_1;
1059     }
1060 
1061     ALOGI("adev_open_input_stream selects channels=%d rate=%d format=%d source=%d",
1062           in->config.channels, in->config.rate, in->config.format, source);
1063 
1064     in->dev = ladev;
1065     in->standby = true;
1066     in->unavailable = false;
1067     in->source = source;
1068     in->devices = devices;
1069 
1070     if (is_aec_input(in)) {
1071         int aec_ret = init_aec_mic_config(ladev->aec, in);
1072         if (aec_ret) {
1073             ALOGE("AEC: Mic config init failed!");
1074             goto error_1;
1075         }
1076     }
1077 
1078 #if DEBUG_AEC
1079     remove("/data/local/traces/aec_ref.pcm");
1080     remove("/data/local/traces/aec_in.pcm");
1081     remove("/data/local/traces/aec_ref_timestamps.txt");
1082     remove("/data/local/traces/aec_in_timestamps.txt");
1083 #endif
1084 
1085     *stream_in = &in->stream;
1086     return 0;
1087 
1088 error_1:
1089     free(in);
1090     return -EINVAL;
1091 }
1092 
adev_close_input_stream(struct audio_hw_device * dev,struct audio_stream_in * stream)1093 static void adev_close_input_stream(struct audio_hw_device *dev,
1094         struct audio_stream_in *stream)
1095 {
1096     ALOGV("adev_close_input_stream...");
1097     struct alsa_stream_in* in = (struct alsa_stream_in*)stream;
1098     if (is_aec_input(in)) {
1099         destroy_aec_mic_config(in->dev->aec);
1100     }
1101     free(stream);
1102     return;
1103 }
1104 
adev_dump(const audio_hw_device_t * device,int fd)1105 static int adev_dump(const audio_hw_device_t *device, int fd)
1106 {
1107     ALOGV("adev_dump");
1108     return 0;
1109 }
1110 
adev_close(hw_device_t * device)1111 static int adev_close(hw_device_t *device)
1112 {
1113     ALOGV("adev_close");
1114 
1115     struct alsa_audio_device *adev = (struct alsa_audio_device *)device;
1116     release_aec(adev->aec);
1117     audio_route_free(adev->audio_route);
1118     mixer_close(adev->mixer);
1119     free(device);
1120     return 0;
1121 }
1122 
adev_open(const hw_module_t * module,const char * name,hw_device_t ** device)1123 static int adev_open(const hw_module_t* module, const char* name,
1124         hw_device_t** device)
1125 {
1126     char vendor_hw[PROPERTY_VALUE_MAX] = {0};
1127     // Prefix for the hdmi path, the board name is the suffix
1128     char path_name[256] = "hdmi_";
1129     ALOGV("adev_open: %s", name);
1130 
1131     if (strcmp(name, AUDIO_HARDWARE_INTERFACE) != 0) {
1132         return -EINVAL;
1133     }
1134 
1135     struct alsa_audio_device* adev = calloc(1, sizeof(struct alsa_audio_device));
1136     if (!adev) {
1137         return -ENOMEM;
1138     }
1139 
1140     adev->hw_device.common.tag = HARDWARE_DEVICE_TAG;
1141     adev->hw_device.common.version = AUDIO_DEVICE_API_VERSION_2_0;
1142     adev->hw_device.common.module = (struct hw_module_t *) module;
1143     adev->hw_device.common.close = adev_close;
1144     adev->hw_device.init_check = adev_init_check;
1145     adev->hw_device.set_voice_volume = adev_set_voice_volume;
1146     adev->hw_device.set_master_volume = adev_set_master_volume;
1147     adev->hw_device.get_master_volume = adev_get_master_volume;
1148     adev->hw_device.set_master_mute = adev_set_master_mute;
1149     adev->hw_device.get_master_mute = adev_get_master_mute;
1150     adev->hw_device.set_mode = adev_set_mode;
1151     adev->hw_device.set_mic_mute = adev_set_mic_mute;
1152     adev->hw_device.get_mic_mute = adev_get_mic_mute;
1153     adev->hw_device.set_parameters = adev_set_parameters;
1154     adev->hw_device.get_parameters = adev_get_parameters;
1155     adev->hw_device.get_input_buffer_size = adev_get_input_buffer_size;
1156     adev->hw_device.open_output_stream = adev_open_output_stream;
1157     adev->hw_device.close_output_stream = adev_close_output_stream;
1158     adev->hw_device.open_input_stream = adev_open_input_stream;
1159     adev->hw_device.close_input_stream = adev_close_input_stream;
1160     adev->hw_device.dump = adev_dump;
1161     adev->hw_device.get_microphones = adev_get_microphones;
1162 
1163     *device = &adev->hw_device.common;
1164 
1165     int out_card = get_audio_card(PCM_OUT, 0);
1166     adev->mixer = mixer_open(out_card);
1167     if (!adev->mixer) {
1168         ALOGE("Unable to open the mixer, aborting.");
1169         goto error_1;
1170     }
1171 
1172     adev->audio_route = audio_route_init(out_card, MIXER_XML_PATH);
1173     if (!adev->audio_route) {
1174         ALOGE("%s: Failed to init audio route controls, aborting.", __func__);
1175         goto error_2;
1176     }
1177 
1178     /*
1179      * To support both the db845c and rb5 we need to used the right mixer path
1180      * we do this by checking the hardware name. Which is set at boot time.
1181      */
1182     property_get("vendor.hw", vendor_hw, "db845c");
1183     strlcat(path_name, vendor_hw, 256);
1184     ALOGV("%s: Using mixer path: %s", __func__, path_name);
1185     audio_route_apply_and_update_path(adev->audio_route, path_name);
1186 
1187     pthread_mutex_lock(&adev->lock);
1188     if (init_aec(CAPTURE_CODEC_SAMPLING_RATE, NUM_AEC_REFERENCE_CHANNELS,
1189                     CHANNEL_STEREO, &adev->aec)) {
1190         pthread_mutex_unlock(&adev->lock);
1191         goto error_3;
1192     }
1193     pthread_mutex_unlock(&adev->lock);
1194 
1195     return 0;
1196 
1197 error_3:
1198     audio_route_free(adev->audio_route);
1199 error_2:
1200     mixer_close(adev->mixer);
1201 error_1:
1202     free(adev);
1203     return -EINVAL;
1204 }
1205 
1206 static struct hw_module_methods_t hal_module_methods = {
1207     .open = adev_open,
1208 };
1209 
1210 struct audio_module HAL_MODULE_INFO_SYM = {
1211     .common = {
1212         .tag = HARDWARE_MODULE_TAG,
1213         .module_api_version = AUDIO_MODULE_API_VERSION_0_1,
1214         .hal_api_version = HARDWARE_HAL_API_VERSION,
1215         .id = AUDIO_HARDWARE_MODULE_ID,
1216         .name = "Yukawa audio HW HAL",
1217         .author = "The Android Open Source Project",
1218         .methods = &hal_module_methods,
1219     },
1220 };
1221