blob: ef6b1534ed9b07c5887fdc3e8b0625965d32ab29 [file] [log] [blame]
Weiyin Jiangaa80acd2016-09-21 16:42:11 +08001/*
Weiyin Jiang82e40942017-01-10 16:07:34 +08002 * Copyright (c) 2016-2017, The Linux Foundation. All rights reserved.
Weiyin Jiangaa80acd2016-09-21 16:42:11 +08003 *
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions are
6 * met:
7 * * Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * * Redistributions in binary form must reproduce the above
10 * copyright notice, this list of conditions and the following
11 * disclaimer in the documentation and/or other materials provided
12 * with the distribution.
13 * * Neither the name of The Linux Foundation nor the names of its
14 * contributors may be used to endorse or promote products derived
15 * from this software without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED "AS IS" AND ANY EXPRESS OR IMPLIED
18 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
19 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT
20 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS
21 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
22 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
23 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
24 * BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
25 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE
26 * OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
27 * IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28 */
29
30#define LOG_TAG "audio_hw_generic_effect"
31//#define LOG_NDEBUG 0
32#define LOG_NDDEBUG 0
33
34#include <errno.h>
35#include <math.h>
36#include <cutils/log.h>
37#include <fcntl.h>
38#include <dirent.h>
39#include "audio_hw.h"
40#include "platform.h"
41#include "platform_api.h"
42#include <sys/stat.h>
43#include <stdlib.h>
44#include <dlfcn.h>
45#include <math.h>
46#include <cutils/properties.h>
47#include "audio_extn.h"
48#include "audio_hw.h"
49
Revathi Uddaraju1eac8b02017-05-18 17:13:33 +053050#ifdef DYNAMIC_LOG_ENABLED
51#include <log_xml_parser.h>
52#define LOG_MASK HAL_MOD_FILE_GEF
53#include <log_utils.h>
54#endif
55
Weiyin Jiangaa80acd2016-09-21 16:42:11 +080056#ifdef AUDIO_GENERIC_EFFECT_FRAMEWORK_ENABLED
57
Weiyin Jiang82e40942017-01-10 16:07:34 +080058#if LINUX_ENABLED
Manish Dewangan6a36d002017-10-09 14:11:00 +053059#define GEF_LIBRARY "libqtigef.so"
Weiyin Jiang82e40942017-01-10 16:07:34 +080060#else
David Ng06ccd872017-03-15 11:39:33 -070061#define GEF_LIBRARY "/vendor/lib/libqtigef.so"
Weiyin Jiang82e40942017-01-10 16:07:34 +080062#endif
Weiyin Jiangaa80acd2016-09-21 16:42:11 +080063
64typedef void* (*gef_init_t)(void*);
Dhananjay Kumara427f142017-07-10 13:49:13 +053065typedef void (*gef_deinit_t)(void*);
Weiyin Jiangaa80acd2016-09-21 16:42:11 +080066typedef void (*gef_device_config_cb_t)(void*, audio_devices_t,
Weiyin Jiang6f4c8062016-11-23 15:30:29 +080067 audio_channel_mask_t, int, int);
Weiyin Jiangaa80acd2016-09-21 16:42:11 +080068
69typedef struct {
70 void* handle;
71 void* gef_ptr;
72 gef_init_t init;
Dhananjay Kumara427f142017-07-10 13:49:13 +053073 gef_deinit_t deinit;
Weiyin Jiangaa80acd2016-09-21 16:42:11 +080074 gef_device_config_cb_t device_config_cb;
75} gef_data;
76
77static gef_data gef_hal_handle;
78
79typedef enum {
80 ASM = 0,
81 ADM
82} gef_calibration_type;
83
84typedef enum {
85 AUDIO_DEVICE_CAL_TYPE = 0,
86 AUDIO_STREAM_CAL_TYPE,
87} acdb_device_type;
88
89
90static acdb_device_type make_acdb_device_type_from_gef_cal_type
91 (gef_calibration_type gef_cal_type)
92{
93 int acdb_device_type = 0;
94
95 switch (gef_cal_type) {
96 case ASM:
97 acdb_device_type = AUDIO_STREAM_CAL_TYPE;
98 break;
99 case ADM:
100 acdb_device_type = AUDIO_DEVICE_CAL_TYPE;
101 break;
102 default:
103 acdb_device_type = -1;
104 break;
105 }
106
107 return ((int)acdb_device_type);
108}
109
110void audio_extn_gef_init(struct audio_device *adev)
111{
112 int ret = 0;
113 const char* error = NULL;
114
115 ALOGV("%s: Enter with error", __func__);
116
117 memset(&gef_hal_handle, 0, sizeof(gef_data));
118
119 ret = access(GEF_LIBRARY, R_OK);
120 if (ret == 0) {
121 //: check error for dlopen
122 gef_hal_handle.handle = dlopen(GEF_LIBRARY, RTLD_LAZY);
123 if (gef_hal_handle.handle == NULL) {
124 ALOGE("%s: DLOPEN failed for %s with error %s",
125 __func__, GEF_LIBRARY, dlerror());
126 goto ERROR_RETURN;
127 } else {
128 ALOGV("%s: DLOPEN successful for %s", __func__, GEF_LIBRARY);
129
130 //call dlerror to clear the error
131 dlerror();
132 gef_hal_handle.init =
133 (gef_init_t)dlsym(gef_hal_handle.handle, "gef_init");
134 error = dlerror();
135
136 if(error != NULL) {
137 ALOGE("%s: dlsym of %s failed with error %s",
138 __func__, "gef_init", error);
139 goto ERROR_RETURN;
140 }
141
142 //call dlerror to clear the error
Dhananjay Kumara427f142017-07-10 13:49:13 +0530143 dlerror();
144 gef_hal_handle.deinit =
145 (gef_deinit_t)dlsym(gef_hal_handle.handle, "gef_deinit");
146 error = dlerror();
147
148 if(error != NULL) {
149 ALOGE("%s: dlsym of %s failed with error %s",
150 __func__, "gef_deinit", error);
151 goto ERROR_RETURN;
152 }
153
154 //call dlerror to clear the error
Weiyin Jiangaa80acd2016-09-21 16:42:11 +0800155 error = dlerror();
156 gef_hal_handle.device_config_cb =
157 (gef_device_config_cb_t)dlsym(gef_hal_handle.handle,
158 "gef_device_config_cb");
159 error = dlerror();
160
161 if(error != NULL) {
162 ALOGE("%s: dlsym of %s failed with error %s",
163 __func__, "gef_device_config_cb", error);
164 goto ERROR_RETURN;
165 }
166
167 gef_hal_handle.gef_ptr = gef_hal_handle.init((void*)adev);
168 }
169 } else {
170 ALOGE("%s: %s access failed", __func__, GEF_LIBRARY);
171 }
172
173ERROR_RETURN:
174 ALOGV("%s: Exit with error %d", __func__, ret);
175 return;
176}
177
178
179//this will be called from GEF to exchange calibration using acdb
180int audio_extn_gef_send_audio_cal(void* dev, int acdb_dev_id,
181 int gef_cal_type, int app_type, int topology_id, int sample_rate,
182 uint32_t module_id, uint32_t param_id, void* data, int length, bool persist)
183{
184 int ret = 0;
185 struct audio_device *adev = (struct audio_device*)dev;
186 int acdb_device_type =
187 make_acdb_device_type_from_gef_cal_type(gef_cal_type);
188
189 ALOGV("%s: Enter", __func__);
190
191 //lock adev
192 pthread_mutex_lock(&adev->lock);
193
194 //send cal
195 ret = platform_send_audio_cal(adev->platform, acdb_dev_id,
196 acdb_device_type, app_type, topology_id, sample_rate,
197 module_id, param_id, data, length, persist);
198
199 pthread_mutex_unlock(&adev->lock);
200
201 ALOGV("%s: Exit with error %d", __func__, ret);
202
203 return ret;
204}
205
206//this will be called from GEF to exchange calibration using acdb
207int audio_extn_gef_get_audio_cal(void* dev, int acdb_dev_id,
208 int gef_cal_type, int app_type, int topology_id, int sample_rate,
209 uint32_t module_id, uint32_t param_id, void* data, int* length, bool persist)
210{
211 int ret = 0;
212 struct audio_device *adev = (struct audio_device*)dev;
213 int acdb_device_type =
214 make_acdb_device_type_from_gef_cal_type(gef_cal_type);
215
216 ALOGV("%s: Enter", __func__);
217
218 //lock adev
219 pthread_mutex_lock(&adev->lock);
220
221 ret = platform_get_audio_cal(adev->platform, acdb_dev_id,
222 acdb_device_type, app_type, topology_id, sample_rate,
223 module_id, param_id, data, length, persist);
224
225 pthread_mutex_unlock(&adev->lock);
226
227 ALOGV("%s: Exit with error %d", __func__, ret);
228
229 return ret;
230}
231
232//this will be called from GEF to store into acdb
233int audio_extn_gef_store_audio_cal(void* dev, int acdb_dev_id,
234 int gef_cal_type, int app_type, int topology_id, int sample_rate,
235 uint32_t module_id, uint32_t param_id, void* data, int length)
236{
237 int ret = 0;
238 struct audio_device *adev = (struct audio_device*)dev;
239 int acdb_device_type =
240 make_acdb_device_type_from_gef_cal_type(gef_cal_type);
241
242 ALOGV("%s: Enter", __func__);
243
244 //lock adev
245 pthread_mutex_lock(&adev->lock);
246
247 ret = platform_store_audio_cal(adev->platform, acdb_dev_id,
248 acdb_device_type, app_type, topology_id, sample_rate,
249 module_id, param_id, data, length);
250
251 pthread_mutex_unlock(&adev->lock);
252
253 ALOGV("%s: Exit with error %d", __func__, ret);
254
255 return ret;
256}
257
258//this will be called from GEF to retrieve calibration using acdb
259int audio_extn_gef_retrieve_audio_cal(void* dev, int acdb_dev_id,
260 int gef_cal_type, int app_type, int topology_id, int sample_rate,
261 uint32_t module_id, uint32_t param_id, void* data, int* length)
262{
263 int ret = 0;
264 struct audio_device *adev = (struct audio_device*)dev;
265 int acdb_device_type =
266 make_acdb_device_type_from_gef_cal_type(gef_cal_type);
267
268 ALOGV("%s: Enter", __func__);
269
270 //lock adev
271 pthread_mutex_lock(&adev->lock);
272
273 ret = platform_retrieve_audio_cal(adev->platform, acdb_dev_id,
274 acdb_device_type, app_type, topology_id, sample_rate,
275 module_id, param_id, data, length);
276
277 pthread_mutex_unlock(&adev->lock);
278
279 ALOGV("%s: Exit with error %d", __func__, ret);
280
281 return ret;
282}
283
284//this will be called from HAL to notify GEF of new device configuration
285void audio_extn_gef_notify_device_config(audio_devices_t audio_device,
Weiyin Jiang6f4c8062016-11-23 15:30:29 +0800286 audio_channel_mask_t channel_mask, int sample_rate, int acdb_id)
Weiyin Jiangaa80acd2016-09-21 16:42:11 +0800287{
288 ALOGV("%s: Enter", __func__);
289
290 //call into GEF to share channel mask and device info
291 if (gef_hal_handle.handle && gef_hal_handle.device_config_cb) {
292 gef_hal_handle.device_config_cb(gef_hal_handle.gef_ptr, audio_device, channel_mask,
Weiyin Jiang6f4c8062016-11-23 15:30:29 +0800293 sample_rate, acdb_id);
Weiyin Jiangaa80acd2016-09-21 16:42:11 +0800294 }
295
296 ALOGV("%s: Exit", __func__);
297
298 return;
299}
300
301void audio_extn_gef_deinit()
302{
303 ALOGV("%s: Enter", __func__);
304
305 if (gef_hal_handle.handle) {
Dhananjay Kumara427f142017-07-10 13:49:13 +0530306 if (gef_hal_handle.handle && gef_hal_handle.deinit)
307 gef_hal_handle.deinit(gef_hal_handle.gef_ptr);
Weiyin Jiangaa80acd2016-09-21 16:42:11 +0800308 dlclose(gef_hal_handle.handle);
309 }
310
311 memset(&gef_hal_handle, 0, sizeof(gef_data));
312
313 ALOGV("%s: Exit", __func__);
314}
315
316#endif