1 /*
2  * Copyright (C) 2011 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #define LOG_TAG "legacy_audio_hw_hal"
18 //#define LOG_NDEBUG 0
19 
20 #include <stdint.h>
21 
22 #include <hardware/hardware.h>
23 #include <system/audio.h>
24 #include <hardware/audio.h>
25 
26 #include <hardware_legacy/AudioHardwareInterface.h>
27 #include <hardware_legacy/AudioSystemLegacy.h>
28 
29 namespace android_audio_legacy {
30 
31 class AudioHardwareInterface;
32 
33 extern "C" {
34 
35 struct legacy_audio_module {
36     struct audio_module module;
37 };
38 
39 struct legacy_audio_device {
40     struct audio_hw_device device;
41 
42     AudioHardwareInterface *hwif;
43 };
44 
45 struct legacy_stream_out {
46     struct audio_stream_out stream;
47 
48     AudioStreamOut *legacy_out;
49 };
50 
51 struct legacy_stream_in {
52     struct audio_stream_in stream;
53 
54     AudioStreamIn *legacy_in;
55 };
56 
57 
58 enum {
59     HAL_API_REV_1_0,
60     HAL_API_REV_2_0,
61     HAL_API_REV_NUM
62 } hal_api_rev;
63 
64 static uint32_t audio_device_conv_table[][HAL_API_REV_NUM] =
65 {
66     /* output devices */
67     { AudioSystem::DEVICE_OUT_EARPIECE, AUDIO_DEVICE_OUT_EARPIECE },
68     { AudioSystem::DEVICE_OUT_SPEAKER, AUDIO_DEVICE_OUT_SPEAKER },
69     { AudioSystem::DEVICE_OUT_WIRED_HEADSET, AUDIO_DEVICE_OUT_WIRED_HEADSET },
70     { AudioSystem::DEVICE_OUT_WIRED_HEADPHONE, AUDIO_DEVICE_OUT_WIRED_HEADPHONE },
71     { AudioSystem::DEVICE_OUT_BLUETOOTH_SCO, AUDIO_DEVICE_OUT_BLUETOOTH_SCO },
72     { AudioSystem::DEVICE_OUT_BLUETOOTH_SCO_HEADSET, AUDIO_DEVICE_OUT_BLUETOOTH_SCO_HEADSET },
73     { AudioSystem::DEVICE_OUT_BLUETOOTH_SCO_CARKIT, AUDIO_DEVICE_OUT_BLUETOOTH_SCO_CARKIT },
74     { AudioSystem::DEVICE_OUT_BLUETOOTH_A2DP, AUDIO_DEVICE_OUT_BLUETOOTH_A2DP },
75     { AudioSystem::DEVICE_OUT_BLUETOOTH_A2DP_HEADPHONES, AUDIO_DEVICE_OUT_BLUETOOTH_A2DP_HEADPHONES },
76     { AudioSystem::DEVICE_OUT_BLUETOOTH_A2DP_SPEAKER, AUDIO_DEVICE_OUT_BLUETOOTH_A2DP_SPEAKER },
77     { AudioSystem::DEVICE_OUT_AUX_DIGITAL, AUDIO_DEVICE_OUT_AUX_DIGITAL },
78     { AudioSystem::DEVICE_OUT_ANLG_DOCK_HEADSET, AUDIO_DEVICE_OUT_ANLG_DOCK_HEADSET },
79     { AudioSystem::DEVICE_OUT_DGTL_DOCK_HEADSET, AUDIO_DEVICE_OUT_DGTL_DOCK_HEADSET },
80     { AudioSystem::DEVICE_OUT_DEFAULT, AUDIO_DEVICE_OUT_DEFAULT },
81     /* input devices */
82     { AudioSystem::DEVICE_IN_COMMUNICATION, AUDIO_DEVICE_IN_COMMUNICATION },
83     { AudioSystem::DEVICE_IN_AMBIENT, AUDIO_DEVICE_IN_AMBIENT },
84     { AudioSystem::DEVICE_IN_BUILTIN_MIC, AUDIO_DEVICE_IN_BUILTIN_MIC },
85     { AudioSystem::DEVICE_IN_BLUETOOTH_SCO_HEADSET, AUDIO_DEVICE_IN_BLUETOOTH_SCO_HEADSET },
86     { AudioSystem::DEVICE_IN_WIRED_HEADSET, AUDIO_DEVICE_IN_WIRED_HEADSET },
87     { AudioSystem::DEVICE_IN_AUX_DIGITAL, AUDIO_DEVICE_IN_AUX_DIGITAL },
88     { AudioSystem::DEVICE_IN_VOICE_CALL, AUDIO_DEVICE_IN_VOICE_CALL },
89     { AudioSystem::DEVICE_IN_BACK_MIC, AUDIO_DEVICE_IN_BACK_MIC },
90     { AudioSystem::DEVICE_IN_DEFAULT, AUDIO_DEVICE_IN_DEFAULT },
91 };
92 
convert_audio_device(uint32_t from_device,int from_rev,int to_rev)93 static audio_devices_t convert_audio_device(uint32_t from_device, int from_rev, int to_rev)
94 {
95     const uint32_t k_num_devices = sizeof(audio_device_conv_table)/sizeof(uint32_t)/HAL_API_REV_NUM;
96     audio_devices_t to_device = AUDIO_DEVICE_NONE;
97     uint32_t in_bit = 0;
98 
99     if (from_rev != HAL_API_REV_1_0) {
100         in_bit = from_device & AUDIO_DEVICE_BIT_IN;
101         from_device &= ~AUDIO_DEVICE_BIT_IN;
102     }
103 
104     while (from_device) {
105         uint32_t i = 31 - __builtin_clz(from_device);
106         uint32_t cur_device = (1 << i) | in_bit;
107 
108         for (i = 0; i < k_num_devices; i++) {
109             if (audio_device_conv_table[i][from_rev] == cur_device) {
110                 to_device = (audio_devices_t)(to_device | audio_device_conv_table[i][to_rev]);
111                 break;
112             }
113         }
114         from_device &= ~cur_device;
115     }
116     return to_device;
117 }
118 
119 
120 /** audio_stream_out implementation **/
out_get_sample_rate(const struct audio_stream * stream)121 static uint32_t out_get_sample_rate(const struct audio_stream *stream)
122 {
123     const struct legacy_stream_out *out =
124         reinterpret_cast<const struct legacy_stream_out *>(stream);
125     return out->legacy_out->sampleRate();
126 }
127 
out_set_sample_rate(struct audio_stream * stream,uint32_t rate)128 static int out_set_sample_rate(struct audio_stream *stream, uint32_t rate)
129 {
130     struct legacy_stream_out *out =
131         reinterpret_cast<struct legacy_stream_out *>(stream);
132 
133     ALOGE("(%s:%d) %s: Implement me!", __FILE__, __LINE__, __func__);
134     /* TODO: implement this */
135     return 0;
136 }
137 
out_get_buffer_size(const struct audio_stream * stream)138 static size_t out_get_buffer_size(const struct audio_stream *stream)
139 {
140     const struct legacy_stream_out *out =
141         reinterpret_cast<const struct legacy_stream_out *>(stream);
142     return out->legacy_out->bufferSize();
143 }
144 
out_get_channels(const struct audio_stream * stream)145 static audio_channel_mask_t out_get_channels(const struct audio_stream *stream)
146 {
147     const struct legacy_stream_out *out =
148         reinterpret_cast<const struct legacy_stream_out *>(stream);
149     return (audio_channel_mask_t) out->legacy_out->channels();
150 }
151 
out_get_format(const struct audio_stream * stream)152 static audio_format_t out_get_format(const struct audio_stream *stream)
153 {
154     const struct legacy_stream_out *out =
155         reinterpret_cast<const struct legacy_stream_out *>(stream);
156     // legacy API, don't change return type
157     return (audio_format_t) out->legacy_out->format();
158 }
159 
out_set_format(struct audio_stream * stream,audio_format_t format)160 static int out_set_format(struct audio_stream *stream, audio_format_t format)
161 {
162     struct legacy_stream_out *out =
163         reinterpret_cast<struct legacy_stream_out *>(stream);
164     ALOGE("(%s:%d) %s: Implement me!", __FILE__, __LINE__, __func__);
165     /* TODO: implement me */
166     return 0;
167 }
168 
out_standby(struct audio_stream * stream)169 static int out_standby(struct audio_stream *stream)
170 {
171     struct legacy_stream_out *out =
172         reinterpret_cast<struct legacy_stream_out *>(stream);
173     return out->legacy_out->standby();
174 }
175 
out_dump(const struct audio_stream * stream,int fd)176 static int out_dump(const struct audio_stream *stream, int fd)
177 {
178     const struct legacy_stream_out *out =
179         reinterpret_cast<const struct legacy_stream_out *>(stream);
180     Vector<String16> args;
181     return out->legacy_out->dump(fd, args);
182 }
183 
out_set_parameters(struct audio_stream * stream,const char * kvpairs)184 static int out_set_parameters(struct audio_stream *stream, const char *kvpairs)
185 {
186     struct legacy_stream_out *out =
187         reinterpret_cast<struct legacy_stream_out *>(stream);
188     int val;
189     String8 s8 = String8(kvpairs);
190     AudioParameter parms = AudioParameter(String8(kvpairs));
191 
192     if (parms.getInt(String8(AUDIO_PARAMETER_STREAM_ROUTING), val) == NO_ERROR) {
193         val = convert_audio_device(val, HAL_API_REV_2_0, HAL_API_REV_1_0);
194         parms.remove(String8(AUDIO_PARAMETER_STREAM_ROUTING));
195         parms.addInt(String8(AUDIO_PARAMETER_STREAM_ROUTING), val);
196         s8 = parms.toString();
197     }
198 
199     return out->legacy_out->setParameters(s8);
200 }
201 
out_get_parameters(const struct audio_stream * stream,const char * keys)202 static char * out_get_parameters(const struct audio_stream *stream, const char *keys)
203 {
204     const struct legacy_stream_out *out =
205         reinterpret_cast<const struct legacy_stream_out *>(stream);
206     String8 s8;
207     int val;
208 
209     s8 = out->legacy_out->getParameters(String8(keys));
210 
211     AudioParameter parms = AudioParameter(s8);
212     if (parms.getInt(String8(AUDIO_PARAMETER_STREAM_ROUTING), val) == NO_ERROR) {
213         val = convert_audio_device(val, HAL_API_REV_1_0, HAL_API_REV_2_0);
214         parms.remove(String8(AUDIO_PARAMETER_STREAM_ROUTING));
215         parms.addInt(String8(AUDIO_PARAMETER_STREAM_ROUTING), val);
216         s8 = parms.toString();
217     }
218 
219     return strdup(s8.c_str());
220 }
221 
out_get_latency(const struct audio_stream_out * stream)222 static uint32_t out_get_latency(const struct audio_stream_out *stream)
223 {
224     const struct legacy_stream_out *out =
225         reinterpret_cast<const struct legacy_stream_out *>(stream);
226     return out->legacy_out->latency();
227 }
228 
out_set_volume(struct audio_stream_out * stream,float left,float right)229 static int out_set_volume(struct audio_stream_out *stream, float left,
230                           float right)
231 {
232     struct legacy_stream_out *out =
233         reinterpret_cast<struct legacy_stream_out *>(stream);
234     return out->legacy_out->setVolume(left, right);
235 }
236 
out_write(struct audio_stream_out * stream,const void * buffer,size_t bytes)237 static ssize_t out_write(struct audio_stream_out *stream, const void* buffer,
238                          size_t bytes)
239 {
240     struct legacy_stream_out *out =
241         reinterpret_cast<struct legacy_stream_out *>(stream);
242     return out->legacy_out->write(buffer, bytes);
243 }
244 
out_get_render_position(const struct audio_stream_out * stream,uint32_t * dsp_frames)245 static int out_get_render_position(const struct audio_stream_out *stream,
246                                    uint32_t *dsp_frames)
247 {
248     const struct legacy_stream_out *out =
249         reinterpret_cast<const struct legacy_stream_out *>(stream);
250     return out->legacy_out->getRenderPosition(dsp_frames);
251 }
252 
out_get_next_write_timestamp(const struct audio_stream_out * stream,int64_t * timestamp)253 static int out_get_next_write_timestamp(const struct audio_stream_out *stream,
254                                         int64_t *timestamp)
255 {
256     const struct legacy_stream_out *out =
257         reinterpret_cast<const struct legacy_stream_out *>(stream);
258     return out->legacy_out->getNextWriteTimestamp(timestamp);
259 }
260 
out_add_audio_effect(const struct audio_stream * stream,effect_handle_t effect)261 static int out_add_audio_effect(const struct audio_stream *stream, effect_handle_t effect)
262 {
263     return 0;
264 }
265 
out_remove_audio_effect(const struct audio_stream * stream,effect_handle_t effect)266 static int out_remove_audio_effect(const struct audio_stream *stream, effect_handle_t effect)
267 {
268     return 0;
269 }
270 
271 /** audio_stream_in implementation **/
in_get_sample_rate(const struct audio_stream * stream)272 static uint32_t in_get_sample_rate(const struct audio_stream *stream)
273 {
274     const struct legacy_stream_in *in =
275         reinterpret_cast<const struct legacy_stream_in *>(stream);
276     return in->legacy_in->sampleRate();
277 }
278 
in_set_sample_rate(struct audio_stream * stream,uint32_t rate)279 static int in_set_sample_rate(struct audio_stream *stream, uint32_t rate)
280 {
281     struct legacy_stream_in *in =
282         reinterpret_cast<struct legacy_stream_in *>(stream);
283 
284     ALOGE("(%s:%d) %s: Implement me!", __FILE__, __LINE__, __func__);
285     /* TODO: implement this */
286     return 0;
287 }
288 
in_get_buffer_size(const struct audio_stream * stream)289 static size_t in_get_buffer_size(const struct audio_stream *stream)
290 {
291     const struct legacy_stream_in *in =
292         reinterpret_cast<const struct legacy_stream_in *>(stream);
293     return in->legacy_in->bufferSize();
294 }
295 
in_get_channels(const struct audio_stream * stream)296 static audio_channel_mask_t in_get_channels(const struct audio_stream *stream)
297 {
298     const struct legacy_stream_in *in =
299         reinterpret_cast<const struct legacy_stream_in *>(stream);
300     return (audio_channel_mask_t) in->legacy_in->channels();
301 }
302 
in_get_format(const struct audio_stream * stream)303 static audio_format_t in_get_format(const struct audio_stream *stream)
304 {
305     const struct legacy_stream_in *in =
306         reinterpret_cast<const struct legacy_stream_in *>(stream);
307     // legacy API, don't change return type
308     return (audio_format_t) in->legacy_in->format();
309 }
310 
in_set_format(struct audio_stream * stream,audio_format_t format)311 static int in_set_format(struct audio_stream *stream, audio_format_t format)
312 {
313     struct legacy_stream_in *in =
314         reinterpret_cast<struct legacy_stream_in *>(stream);
315     ALOGE("(%s:%d) %s: Implement me!", __FILE__, __LINE__, __func__);
316     /* TODO: implement me */
317     return 0;
318 }
319 
in_standby(struct audio_stream * stream)320 static int in_standby(struct audio_stream *stream)
321 {
322     struct legacy_stream_in *in = reinterpret_cast<struct legacy_stream_in *>(stream);
323     return in->legacy_in->standby();
324 }
325 
in_dump(const struct audio_stream * stream,int fd)326 static int in_dump(const struct audio_stream *stream, int fd)
327 {
328     const struct legacy_stream_in *in =
329         reinterpret_cast<const struct legacy_stream_in *>(stream);
330     Vector<String16> args;
331     return in->legacy_in->dump(fd, args);
332 }
333 
in_set_parameters(struct audio_stream * stream,const char * kvpairs)334 static int in_set_parameters(struct audio_stream *stream, const char *kvpairs)
335 {
336     struct legacy_stream_in *in =
337         reinterpret_cast<struct legacy_stream_in *>(stream);
338     int val;
339     AudioParameter parms = AudioParameter(String8(kvpairs));
340     String8 s8 = String8(kvpairs);
341 
342     if (parms.getInt(String8(AUDIO_PARAMETER_STREAM_ROUTING), val) == NO_ERROR) {
343         val = convert_audio_device(val, HAL_API_REV_2_0, HAL_API_REV_1_0);
344         parms.remove(String8(AUDIO_PARAMETER_STREAM_ROUTING));
345         parms.addInt(String8(AUDIO_PARAMETER_STREAM_ROUTING), val);
346         s8 = parms.toString();
347     }
348 
349     return in->legacy_in->setParameters(s8);
350 }
351 
in_get_parameters(const struct audio_stream * stream,const char * keys)352 static char * in_get_parameters(const struct audio_stream *stream,
353                                 const char *keys)
354 {
355     const struct legacy_stream_in *in =
356         reinterpret_cast<const struct legacy_stream_in *>(stream);
357     String8 s8;
358     int val;
359 
360     s8 = in->legacy_in->getParameters(String8(keys));
361 
362     AudioParameter parms = AudioParameter(s8);
363     if (parms.getInt(String8(AUDIO_PARAMETER_STREAM_ROUTING), val) == NO_ERROR) {
364         val = convert_audio_device(val, HAL_API_REV_1_0, HAL_API_REV_2_0);
365         parms.remove(String8(AUDIO_PARAMETER_STREAM_ROUTING));
366         parms.addInt(String8(AUDIO_PARAMETER_STREAM_ROUTING), val);
367         s8 = parms.toString();
368     }
369 
370     return strdup(s8.c_str());
371 }
372 
in_set_gain(struct audio_stream_in * stream,float gain)373 static int in_set_gain(struct audio_stream_in *stream, float gain)
374 {
375     struct legacy_stream_in *in =
376         reinterpret_cast<struct legacy_stream_in *>(stream);
377     return in->legacy_in->setGain(gain);
378 }
379 
in_read(struct audio_stream_in * stream,void * buffer,size_t bytes)380 static ssize_t in_read(struct audio_stream_in *stream, void* buffer,
381                        size_t bytes)
382 {
383     struct legacy_stream_in *in =
384         reinterpret_cast<struct legacy_stream_in *>(stream);
385     return in->legacy_in->read(buffer, bytes);
386 }
387 
in_get_input_frames_lost(struct audio_stream_in * stream)388 static uint32_t in_get_input_frames_lost(struct audio_stream_in *stream)
389 {
390     struct legacy_stream_in *in =
391         reinterpret_cast<struct legacy_stream_in *>(stream);
392     return in->legacy_in->getInputFramesLost();
393 }
394 
in_add_audio_effect(const struct audio_stream * stream,effect_handle_t effect)395 static int in_add_audio_effect(const struct audio_stream *stream, effect_handle_t effect)
396 {
397     const struct legacy_stream_in *in =
398         reinterpret_cast<const struct legacy_stream_in *>(stream);
399     return in->legacy_in->addAudioEffect(effect);
400 }
401 
in_remove_audio_effect(const struct audio_stream * stream,effect_handle_t effect)402 static int in_remove_audio_effect(const struct audio_stream *stream, effect_handle_t effect)
403 {
404     const struct legacy_stream_in *in =
405         reinterpret_cast<const struct legacy_stream_in *>(stream);
406     return in->legacy_in->removeAudioEffect(effect);
407 }
408 
409 /** audio_hw_device implementation **/
to_ladev(struct audio_hw_device * dev)410 static inline struct legacy_audio_device * to_ladev(struct audio_hw_device *dev)
411 {
412     return reinterpret_cast<struct legacy_audio_device *>(dev);
413 }
414 
to_cladev(const struct audio_hw_device * dev)415 static inline const struct legacy_audio_device * to_cladev(const struct audio_hw_device *dev)
416 {
417     return reinterpret_cast<const struct legacy_audio_device *>(dev);
418 }
419 
adev_init_check(const struct audio_hw_device * dev)420 static int adev_init_check(const struct audio_hw_device *dev)
421 {
422     const struct legacy_audio_device *ladev = to_cladev(dev);
423 
424     return ladev->hwif->initCheck();
425 }
426 
adev_set_voice_volume(struct audio_hw_device * dev,float volume)427 static int adev_set_voice_volume(struct audio_hw_device *dev, float volume)
428 {
429     struct legacy_audio_device *ladev = to_ladev(dev);
430     return ladev->hwif->setVoiceVolume(volume);
431 }
432 
adev_set_master_volume(struct audio_hw_device * dev,float volume)433 static int adev_set_master_volume(struct audio_hw_device *dev, float volume)
434 {
435     struct legacy_audio_device *ladev = to_ladev(dev);
436     return ladev->hwif->setMasterVolume(volume);
437 }
438 
adev_get_master_volume(struct audio_hw_device * dev,float * volume)439 static int adev_get_master_volume(struct audio_hw_device *dev, float* volume)
440 {
441     struct legacy_audio_device *ladev = to_ladev(dev);
442     return ladev->hwif->getMasterVolume(volume);
443 }
444 
adev_set_mode(struct audio_hw_device * dev,audio_mode_t mode)445 static int adev_set_mode(struct audio_hw_device *dev, audio_mode_t mode)
446 {
447     struct legacy_audio_device *ladev = to_ladev(dev);
448     // as this is the legacy API, don't change it to use audio_mode_t instead of int
449     return ladev->hwif->setMode((int) mode);
450 }
451 
adev_set_mic_mute(struct audio_hw_device * dev,bool state)452 static int adev_set_mic_mute(struct audio_hw_device *dev, bool state)
453 {
454     struct legacy_audio_device *ladev = to_ladev(dev);
455     return ladev->hwif->setMicMute(state);
456 }
457 
adev_get_mic_mute(const struct audio_hw_device * dev,bool * state)458 static int adev_get_mic_mute(const struct audio_hw_device *dev, bool *state)
459 {
460     const struct legacy_audio_device *ladev = to_cladev(dev);
461     return ladev->hwif->getMicMute(state);
462 }
463 
adev_set_parameters(struct audio_hw_device * dev,const char * kvpairs)464 static int adev_set_parameters(struct audio_hw_device *dev, const char *kvpairs)
465 {
466     struct legacy_audio_device *ladev = to_ladev(dev);
467     return ladev->hwif->setParameters(String8(kvpairs));
468 }
469 
adev_get_parameters(const struct audio_hw_device * dev,const char * keys)470 static char * adev_get_parameters(const struct audio_hw_device *dev,
471                                   const char *keys)
472 {
473     const struct legacy_audio_device *ladev = to_cladev(dev);
474     String8 s8;
475 
476     s8 = ladev->hwif->getParameters(String8(keys));
477     return strdup(s8.c_str());
478 }
479 
adev_get_input_buffer_size(const struct audio_hw_device * dev,const struct audio_config * config)480 static size_t adev_get_input_buffer_size(const struct audio_hw_device *dev,
481                                          const struct audio_config *config)
482 {
483     const struct legacy_audio_device *ladev = to_cladev(dev);
484     return ladev->hwif->getInputBufferSize(config->sample_rate, (int) config->format,
485                                            audio_channel_count_from_in_mask(config->channel_mask));
486 }
487 
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)488 static int adev_open_output_stream(struct audio_hw_device *dev,
489                                    audio_io_handle_t handle,
490                                    audio_devices_t devices,
491                                    audio_output_flags_t flags,
492                                    struct audio_config *config,
493                                    struct audio_stream_out **stream_out,
494                                    const char *address __unused)
495 {
496     struct legacy_audio_device *ladev = to_ladev(dev);
497     status_t status;
498     struct legacy_stream_out *out;
499     int ret;
500 
501     out = (struct legacy_stream_out *)calloc(1, sizeof(*out));
502     if (!out)
503         return -ENOMEM;
504 
505     devices = convert_audio_device(devices, HAL_API_REV_2_0, HAL_API_REV_1_0);
506 
507     uint32_t raw_channel_mask = config->channel_mask;
508     out->legacy_out = ladev->hwif->openOutputStreamWithFlags(devices, flags,
509                                                     (int *) &config->format,
510                                                     &raw_channel_mask,
511                                                     &config->sample_rate, &status);
512     if (!out->legacy_out) {
513         ret = status;
514         goto err_open;
515     }
516     config->channel_mask = (audio_channel_mask_t)raw_channel_mask;
517 
518     out->stream.common.get_sample_rate = out_get_sample_rate;
519     out->stream.common.set_sample_rate = out_set_sample_rate;
520     out->stream.common.get_buffer_size = out_get_buffer_size;
521     out->stream.common.get_channels = out_get_channels;
522     out->stream.common.get_format = out_get_format;
523     out->stream.common.set_format = out_set_format;
524     out->stream.common.standby = out_standby;
525     out->stream.common.dump = out_dump;
526     out->stream.common.set_parameters = out_set_parameters;
527     out->stream.common.get_parameters = out_get_parameters;
528     out->stream.common.add_audio_effect = out_add_audio_effect;
529     out->stream.common.remove_audio_effect = out_remove_audio_effect;
530     out->stream.get_latency = out_get_latency;
531     out->stream.set_volume = out_set_volume;
532     out->stream.write = out_write;
533     out->stream.get_render_position = out_get_render_position;
534     out->stream.get_next_write_timestamp = out_get_next_write_timestamp;
535 
536     *stream_out = &out->stream;
537     return 0;
538 
539 err_open:
540     free(out);
541     *stream_out = NULL;
542     return ret;
543 }
544 
adev_close_output_stream(struct audio_hw_device * dev,struct audio_stream_out * stream)545 static void adev_close_output_stream(struct audio_hw_device *dev,
546                                      struct audio_stream_out* stream)
547 {
548     struct legacy_audio_device *ladev = to_ladev(dev);
549     struct legacy_stream_out *out = reinterpret_cast<struct legacy_stream_out *>(stream);
550 
551     ladev->hwif->closeOutputStream(out->legacy_out);
552     free(out);
553 }
554 
555 /** This method creates and opens the audio hardware input stream */
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 __unused)556 static int adev_open_input_stream(struct audio_hw_device *dev,
557                                   audio_io_handle_t handle,
558                                   audio_devices_t devices,
559                                   struct audio_config *config,
560                                   struct audio_stream_in **stream_in,
561                                   audio_input_flags_t flags __unused,
562                                   const char *address __unused,
563                                   audio_source_t source __unused)
564 {
565     struct legacy_audio_device *ladev = to_ladev(dev);
566     status_t status;
567     struct legacy_stream_in *in;
568     int ret;
569 
570     in = (struct legacy_stream_in *)calloc(1, sizeof(*in));
571     if (!in)
572         return -ENOMEM;
573 
574     devices = convert_audio_device(devices, HAL_API_REV_2_0, HAL_API_REV_1_0);
575 
576     uint32_t raw_channel_mask = config->channel_mask;
577     in->legacy_in = ladev->hwif->openInputStream(devices, (int *) &config->format,
578                                                  &raw_channel_mask, &config->sample_rate,
579                                                  &status, (AudioSystem::audio_in_acoustics)0);
580     if (!in->legacy_in) {
581         ret = status;
582         goto err_open;
583     }
584     config->channel_mask = (audio_channel_mask_t)raw_channel_mask;
585 
586     in->stream.common.get_sample_rate = in_get_sample_rate;
587     in->stream.common.set_sample_rate = in_set_sample_rate;
588     in->stream.common.get_buffer_size = in_get_buffer_size;
589     in->stream.common.get_channels = in_get_channels;
590     in->stream.common.get_format = in_get_format;
591     in->stream.common.set_format = in_set_format;
592     in->stream.common.standby = in_standby;
593     in->stream.common.dump = in_dump;
594     in->stream.common.set_parameters = in_set_parameters;
595     in->stream.common.get_parameters = in_get_parameters;
596     in->stream.common.add_audio_effect = in_add_audio_effect;
597     in->stream.common.remove_audio_effect = in_remove_audio_effect;
598     in->stream.set_gain = in_set_gain;
599     in->stream.read = in_read;
600     in->stream.get_input_frames_lost = in_get_input_frames_lost;
601 
602     *stream_in = &in->stream;
603     return 0;
604 
605 err_open:
606     free(in);
607     *stream_in = NULL;
608     return ret;
609 }
610 
adev_close_input_stream(struct audio_hw_device * dev,struct audio_stream_in * stream)611 static void adev_close_input_stream(struct audio_hw_device *dev,
612                                struct audio_stream_in *stream)
613 {
614     struct legacy_audio_device *ladev = to_ladev(dev);
615     struct legacy_stream_in *in =
616         reinterpret_cast<struct legacy_stream_in *>(stream);
617 
618     ladev->hwif->closeInputStream(in->legacy_in);
619     free(in);
620 }
621 
adev_dump(const struct audio_hw_device * dev,int fd)622 static int adev_dump(const struct audio_hw_device *dev, int fd)
623 {
624     const struct legacy_audio_device *ladev = to_cladev(dev);
625     Vector<String16> args;
626 
627     return ladev->hwif->dumpState(fd, args);
628 }
629 
legacy_adev_close(hw_device_t * device)630 static int legacy_adev_close(hw_device_t* device)
631 {
632     struct audio_hw_device *hwdev =
633                         reinterpret_cast<struct audio_hw_device *>(device);
634     struct legacy_audio_device *ladev = to_ladev(hwdev);
635 
636     if (!ladev)
637         return 0;
638 
639     if (ladev->hwif)
640         delete ladev->hwif;
641 
642     free(ladev);
643     return 0;
644 }
645 
legacy_adev_open(const hw_module_t * module,const char * name,hw_device_t ** device)646 static int legacy_adev_open(const hw_module_t* module, const char* name,
647                             hw_device_t** device)
648 {
649     struct legacy_audio_device *ladev;
650     int ret;
651 
652     if (strcmp(name, AUDIO_HARDWARE_INTERFACE) != 0)
653         return -EINVAL;
654 
655     ladev = (struct legacy_audio_device *)calloc(1, sizeof(*ladev));
656     if (!ladev)
657         return -ENOMEM;
658 
659     ladev->device.common.tag = HARDWARE_DEVICE_TAG;
660     ladev->device.common.version = AUDIO_DEVICE_API_VERSION_2_0;
661     ladev->device.common.module = const_cast<hw_module_t*>(module);
662     ladev->device.common.close = legacy_adev_close;
663 
664     ladev->device.init_check = adev_init_check;
665     ladev->device.set_voice_volume = adev_set_voice_volume;
666     ladev->device.set_master_volume = adev_set_master_volume;
667     ladev->device.get_master_volume = adev_get_master_volume;
668     ladev->device.set_mode = adev_set_mode;
669     ladev->device.set_mic_mute = adev_set_mic_mute;
670     ladev->device.get_mic_mute = adev_get_mic_mute;
671     ladev->device.set_parameters = adev_set_parameters;
672     ladev->device.get_parameters = adev_get_parameters;
673     ladev->device.get_input_buffer_size = adev_get_input_buffer_size;
674     ladev->device.open_output_stream = adev_open_output_stream;
675     ladev->device.close_output_stream = adev_close_output_stream;
676     ladev->device.open_input_stream = adev_open_input_stream;
677     ladev->device.close_input_stream = adev_close_input_stream;
678     ladev->device.dump = adev_dump;
679 
680     ladev->hwif = createAudioHardware();
681     if (!ladev->hwif) {
682         ret = -EIO;
683         goto err_create_audio_hw;
684     }
685 
686     *device = &ladev->device.common;
687 
688     return 0;
689 
690 err_create_audio_hw:
691     free(ladev);
692     return ret;
693 }
694 
695 static struct hw_module_methods_t legacy_audio_module_methods = {
696         open: legacy_adev_open
697 };
698 
699 struct legacy_audio_module HAL_MODULE_INFO_SYM = {
700     module: {
701         common: {
702             tag: HARDWARE_MODULE_TAG,
703             module_api_version: AUDIO_MODULE_API_VERSION_0_1,
704             hal_api_version: HARDWARE_HAL_API_VERSION,
705             id: AUDIO_HARDWARE_MODULE_ID,
706             name: "LEGACY Audio HW HAL",
707             author: "The Android Open Source Project",
708             methods: &legacy_audio_module_methods,
709             dso : NULL,
710             reserved : {0},
711         },
712     },
713 };
714 
715 }; // extern "C"
716 
717 }; // namespace android_audio_legacy
718