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