blob: bde8ef85439e6d35188a7f013ddcfc04affa645d [file] [log] [blame]
Subhash Chandra Bose Naripeddy3eedc002013-11-12 20:45:15 -08001/*
2 * Copyright (c) 2013, The Linux Foundation. All rights reserved.
3 * Not a Contribution.
4 *
5 * Copyright (C) 2013 The Android Open Source Project
6 *
7 * Licensed under the Apache License, Version 2.0 (the "License");
8 * you may not use this file except in compliance with the License.
9 * You may obtain a copy of the License at
10 *
11 * http://www.apache.org/licenses/LICENSE-2.0
12 *
13 * Unless required by applicable law or agreed to in writing, software
14 * distributed under the License is distributed on an "AS IS" BASIS,
15 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16 * See the License for the specific language governing permissions and
17 * limitations under the License.
18 */
19
20#define LOG_TAG "offload_effect_equalizer"
21#define LOG_NDEBUG 0
22
23#include <cutils/list.h>
24#include <cutils/log.h>
25#include <tinyalsa/asoundlib.h>
Subhash Chandra Bose Naripeddy090a2aa2014-01-30 14:03:12 -080026#include <sound/audio_effects.h>
Subhash Chandra Bose Naripeddy3eedc002013-11-12 20:45:15 -080027#include <audio_effects/effect_equalizer.h>
28
29#include "effect_api.h"
30#include "equalizer.h"
31
32/* Offload equalizer UUID: a0dac280-401c-11e3-9379-0002a5d5c51b */
33const effect_descriptor_t equalizer_descriptor = {
34 {0x0bed4300, 0xddd6, 0x11db, 0x8f34, {0x00, 0x02, 0xa5, 0xd5, 0xc5, 0x1b}}, // type
35 {0xa0dac280, 0x401c, 0x11e3, 0x9379, {0x00, 0x02, 0xa5, 0xd5, 0xc5, 0x1b}}, // uuid
36 EFFECT_CONTROL_API_VERSION,
37 (EFFECT_FLAG_TYPE_INSERT | EFFECT_FLAG_HW_ACC_TUNNEL),
38 0, /* TODO */
39 1,
40 "MSM offload equalizer",
41 "The Android Open Source Project",
42};
43
44static const char *equalizer_preset_names[] = {
45 "Normal",
46 "Classical",
47 "Dance",
48 "Flat",
49 "Folk",
50 "Heavy Metal",
51 "Hip Hop",
52 "Jazz",
53 "Pop",
54 "Rock"
55 };
56
57static const uint32_t equalizer_band_freq_range[NUM_EQ_BANDS][2] = {
58 {30000, 120000},
59 {120001, 460000},
60 {460001, 1800000},
61 {1800001, 7000000},
62 {7000001, 20000000}};
63
64static const uint16_t equalizer_band_presets_level[] = {
65 3, 0, 0, 0, 3, /* Normal Preset */
66 5, 3, -2, 4, 4, /* Classical Preset */
67 6, 0, 2, 4, 1, /* Dance Preset */
68 0, 0, 0, 0, 0, /* Flat Preset */
69 3, 0, 0, 2, -1, /* Folk Preset */
70 4, 1, 9, 3, 0, /* Heavy Metal Preset */
71 5, 3, 0, 1, 3, /* Hip Hop Preset */
72 4, 2, -2, 2, 5, /* Jazz Preset */
73 -1, 2, 5, 1, -2, /* Pop Preset */
74 5, 3, -1, 3, 5}; /* Rock Preset */
75
76const uint16_t equalizer_band_presets_freq[NUM_EQ_BANDS] = {
77 60, /* Frequencies in Hz */
78 230,
79 910,
80 3600,
81 14000
82};
83
84/*
85 * Equalizer operations
86 */
87
88int equalizer_get_band_level(equalizer_context_t *context, int32_t band)
89{
90 ALOGV("%s: band: %d level: %d", __func__, band,
91 context->band_levels[band] * 100);
92 return context->band_levels[band] * 100;
93}
94
95int equalizer_set_band_level(equalizer_context_t *context, int32_t band,
96 int32_t level)
97{
98 ALOGV("%s: band: %d, level: %d", __func__, band, level);
99 if (level > 0) {
100 level = (int)((level+50)/100);
101 } else {
102 level = (int)((level-50)/100);
103 }
104 context->band_levels[band] = level;
105 context->preset = PRESET_CUSTOM;
106
107 offload_eq_set_preset(&(context->offload_eq), PRESET_CUSTOM);
108 offload_eq_set_bands_level(&(context->offload_eq),
109 NUM_EQ_BANDS,
110 equalizer_band_presets_freq,
111 context->band_levels);
112 if (context->ctl)
113 offload_eq_send_params(context->ctl, context->offload_eq,
114 OFFLOAD_SEND_EQ_ENABLE_FLAG |
115 OFFLOAD_SEND_EQ_BANDS_LEVEL);
116 return 0;
117}
118
119int equalizer_get_center_frequency(equalizer_context_t *context, int32_t band)
120{
121 ALOGV("%s: band: %d", __func__, band);
122 return (equalizer_band_freq_range[band][0] +
123 equalizer_band_freq_range[band][1]) / 2;
124}
125
126int equalizer_get_band_freq_range(equalizer_context_t *context, int32_t band,
127 uint32_t *low, uint32_t *high)
128{
129 ALOGV("%s: band: %d", __func__, band);
130 *low = equalizer_band_freq_range[band][0];
131 *high = equalizer_band_freq_range[band][1];
132 return 0;
133}
134
135int equalizer_get_band(equalizer_context_t *context, uint32_t freq)
136{
137 int i;
138
139 ALOGV("%s: freq: %d", __func__, freq);
140 for(i = 0; i < NUM_EQ_BANDS; i++) {
141 if (freq <= equalizer_band_freq_range[i][1]) {
142 return i;
143 }
144 }
145 return NUM_EQ_BANDS - 1;
146}
147
148int equalizer_get_preset(equalizer_context_t *context)
149{
150 ALOGV("%s: preset: %d", __func__, context->preset);
151 return context->preset;
152}
153
154int equalizer_set_preset(equalizer_context_t *context, int preset)
155{
156 int i;
157
158 ALOGV("%s: preset: %d", __func__, preset);
159 context->preset = preset;
160 for (i=0; i<NUM_EQ_BANDS; i++)
161 context->band_levels[i] =
162 equalizer_band_presets_level[i + preset * NUM_EQ_BANDS];
163
164 offload_eq_set_preset(&(context->offload_eq), preset);
165 offload_eq_set_bands_level(&(context->offload_eq),
166 NUM_EQ_BANDS,
167 equalizer_band_presets_freq,
168 context->band_levels);
169 if(context->ctl)
170 offload_eq_send_params(context->ctl, context->offload_eq,
171 OFFLOAD_SEND_EQ_ENABLE_FLAG |
172 OFFLOAD_SEND_EQ_PRESET);
173 return 0;
174}
175
176const char * equalizer_get_preset_name(equalizer_context_t *context,
177 int32_t preset)
178{
179 ALOGV("%s: preset: %s", __func__, equalizer_preset_names[preset]);
180 if (preset == PRESET_CUSTOM) {
181 return "Custom";
182 } else {
183 return equalizer_preset_names[preset];
184 }
185}
186
187int equalizer_get_num_presets(equalizer_context_t *context)
188{
189 ALOGV("%s: presets_num: %d", __func__,
190 sizeof(equalizer_preset_names)/sizeof(char *));
191 return sizeof(equalizer_preset_names)/sizeof(char *);
192}
193
194int equalizer_get_parameter(effect_context_t *context, effect_param_t *p,
195 uint32_t *size)
196{
197 equalizer_context_t *eq_ctxt = (equalizer_context_t *)context;
198 int voffset = ((p->psize - 1) / sizeof(int32_t) + 1) * sizeof(int32_t);
199 int32_t *param_tmp = (int32_t *)p->data;
200 int32_t param = *param_tmp++;
201 int32_t param2;
202 char *name;
203 void *value = p->data + voffset;
204 int i;
205
206 ALOGV("%s", __func__);
207
208 p->status = 0;
209
210 switch (param) {
211 case EQ_PARAM_NUM_BANDS:
212 case EQ_PARAM_CUR_PRESET:
213 case EQ_PARAM_GET_NUM_OF_PRESETS:
214 case EQ_PARAM_BAND_LEVEL:
215 case EQ_PARAM_GET_BAND:
216 if (p->vsize < sizeof(int16_t))
217 p->status = -EINVAL;
218 p->vsize = sizeof(int16_t);
219 break;
220
221 case EQ_PARAM_LEVEL_RANGE:
222 if (p->vsize < 2 * sizeof(int16_t))
223 p->status = -EINVAL;
224 p->vsize = 2 * sizeof(int16_t);
225 break;
226 case EQ_PARAM_BAND_FREQ_RANGE:
227 if (p->vsize < 2 * sizeof(int32_t))
228 p->status = -EINVAL;
229 p->vsize = 2 * sizeof(int32_t);
230 break;
231
232 case EQ_PARAM_CENTER_FREQ:
233 if (p->vsize < sizeof(int32_t))
234 p->status = -EINVAL;
235 p->vsize = sizeof(int32_t);
236 break;
237
238 case EQ_PARAM_GET_PRESET_NAME:
239 break;
240
241 case EQ_PARAM_PROPERTIES:
242 if (p->vsize < (2 + NUM_EQ_BANDS) * sizeof(uint16_t))
243 p->status = -EINVAL;
244 p->vsize = (2 + NUM_EQ_BANDS) * sizeof(uint16_t);
245 break;
246
247 default:
248 p->status = -EINVAL;
249 }
250
251 *size = sizeof(effect_param_t) + voffset + p->vsize;
252
253 if (p->status != 0)
254 return 0;
255
256 switch (param) {
257 case EQ_PARAM_NUM_BANDS:
258 ALOGV("%s: EQ_PARAM_NUM_BANDS", __func__);
259 *(uint16_t *)value = (uint16_t)NUM_EQ_BANDS;
260 break;
261
262 case EQ_PARAM_LEVEL_RANGE:
263 ALOGV("%s: EQ_PARAM_LEVEL_RANGE", __func__);
264 *(int16_t *)value = -1500;
265 *((int16_t *)value + 1) = 1500;
266 break;
267
268 case EQ_PARAM_BAND_LEVEL:
269 ALOGV("%s: EQ_PARAM_BAND_LEVEL", __func__);
270 param2 = *param_tmp;
271 if (param2 >= NUM_EQ_BANDS) {
272 p->status = -EINVAL;
273 break;
274 }
275 *(int16_t *)value = (int16_t)equalizer_get_band_level(eq_ctxt, param2);
276 break;
277
278 case EQ_PARAM_CENTER_FREQ:
279 ALOGV("%s: EQ_PARAM_CENTER_FREQ", __func__);
280 param2 = *param_tmp;
281 if (param2 >= NUM_EQ_BANDS) {
282 p->status = -EINVAL;
283 break;
284 }
285 *(int32_t *)value = equalizer_get_center_frequency(eq_ctxt, param2);
286 break;
287
288 case EQ_PARAM_BAND_FREQ_RANGE:
289 ALOGV("%s: EQ_PARAM_BAND_FREQ_RANGE", __func__);
290 param2 = *param_tmp;
291 if (param2 >= NUM_EQ_BANDS) {
292 p->status = -EINVAL;
293 break;
294 }
295 equalizer_get_band_freq_range(eq_ctxt, param2, (uint32_t *)value,
296 ((uint32_t *)value + 1));
297 break;
298
299 case EQ_PARAM_GET_BAND:
300 ALOGV("%s: EQ_PARAM_GET_BAND", __func__);
301 param2 = *param_tmp;
302 *(uint16_t *)value = (uint16_t)equalizer_get_band(eq_ctxt, param2);
303 break;
304
305 case EQ_PARAM_CUR_PRESET:
306 ALOGV("%s: EQ_PARAM_CUR_PRESET", __func__);
307 *(uint16_t *)value = (uint16_t)equalizer_get_preset(eq_ctxt);
308 break;
309
310 case EQ_PARAM_GET_NUM_OF_PRESETS:
311 ALOGV("%s: EQ_PARAM_GET_NUM_OF_PRESETS", __func__);
312 *(uint16_t *)value = (uint16_t)equalizer_get_num_presets(eq_ctxt);
313 break;
314
315 case EQ_PARAM_GET_PRESET_NAME:
316 ALOGV("%s: EQ_PARAM_GET_PRESET_NAME", __func__);
317 param2 = *param_tmp;
318 ALOGV("param2: %d", param2);
319 if (param2 >= equalizer_get_num_presets(eq_ctxt)) {
320 p->status = -EINVAL;
321 break;
322 }
323 name = (char *)value;
324 strlcpy(name, equalizer_get_preset_name(eq_ctxt, param2), p->vsize - 1);
325 name[p->vsize - 1] = 0;
326 p->vsize = strlen(name) + 1;
327 break;
328
329 case EQ_PARAM_PROPERTIES: {
330 ALOGV("%s: EQ_PARAM_PROPERTIES", __func__);
331 int16_t *prop = (int16_t *)value;
332 prop[0] = (int16_t)equalizer_get_preset(eq_ctxt);
333 prop[1] = (int16_t)NUM_EQ_BANDS;
334 for (i = 0; i < NUM_EQ_BANDS; i++) {
335 prop[2 + i] = (int16_t)equalizer_get_band_level(eq_ctxt, i);
336 }
337 } break;
338
339 default:
340 p->status = -EINVAL;
341 break;
342 }
343
344 return 0;
345}
346
347int equalizer_set_parameter(effect_context_t *context, effect_param_t *p,
348 uint32_t size)
349{
350 equalizer_context_t *eq_ctxt = (equalizer_context_t *)context;
351 int voffset = ((p->psize - 1) / sizeof(int32_t) + 1) * sizeof(int32_t);
352 void *value = p->data + voffset;
353 int32_t *param_tmp = (int32_t *)p->data;
354 int32_t param = *param_tmp++;
355 int32_t preset;
356 int32_t band;
357 int32_t level;
358 int i;
359
360 ALOGV("%s", __func__);
361
362 p->status = 0;
363
364 switch (param) {
365 case EQ_PARAM_CUR_PRESET:
366 ALOGV("EQ_PARAM_CUR_PRESET");
367 preset = (int32_t)(*(uint16_t *)value);
368
369 if ((preset >= equalizer_get_num_presets(eq_ctxt)) || (preset < 0)) {
370 p->status = -EINVAL;
371 break;
372 }
373 equalizer_set_preset(eq_ctxt, preset);
374 break;
375 case EQ_PARAM_BAND_LEVEL:
376 ALOGV("EQ_PARAM_BAND_LEVEL");
377 band = *param_tmp;
378 level = (int32_t)(*(int16_t *)value);
379 if (band >= NUM_EQ_BANDS) {
380 p->status = -EINVAL;
381 break;
382 }
383 equalizer_set_band_level(eq_ctxt, band, level);
384 break;
385 case EQ_PARAM_PROPERTIES: {
386 ALOGV("EQ_PARAM_PROPERTIES");
387 int16_t *prop = (int16_t *)value;
388 if ((int)prop[0] >= equalizer_get_num_presets(eq_ctxt)) {
389 p->status = -EINVAL;
390 break;
391 }
392 if (prop[0] >= 0) {
393 equalizer_set_preset(eq_ctxt, (int)prop[0]);
394 } else {
395 if ((int)prop[1] != NUM_EQ_BANDS) {
396 p->status = -EINVAL;
397 break;
398 }
399 for (i = 0; i < NUM_EQ_BANDS; i++) {
400 equalizer_set_band_level(eq_ctxt, i, (int)prop[2 + i]);
401 }
402 }
403 } break;
404 default:
405 p->status = -EINVAL;
406 break;
407 }
408
409 return 0;
410}
411
412int equalizer_set_device(effect_context_t *context, uint32_t device)
413{
414 ALOGV("%s: device: %d", __func__, device);
415 equalizer_context_t *eq_ctxt = (equalizer_context_t *)context;
416 eq_ctxt->device = device;
417 offload_eq_set_device(&(eq_ctxt->offload_eq), device);
418 return 0;
419}
420
421int equalizer_reset(effect_context_t *context)
422{
423 equalizer_context_t *eq_ctxt = (equalizer_context_t *)context;
424
425 return 0;
426}
427
428int equalizer_init(effect_context_t *context)
429{
430 ALOGV("%s", __func__);
431 equalizer_context_t *eq_ctxt = (equalizer_context_t *)context;
432
433 context->config.inputCfg.accessMode = EFFECT_BUFFER_ACCESS_READ;
434 context->config.inputCfg.channels = AUDIO_CHANNEL_OUT_STEREO;
435 context->config.inputCfg.format = AUDIO_FORMAT_PCM_16_BIT;
436 context->config.inputCfg.samplingRate = 44100;
437 context->config.inputCfg.bufferProvider.getBuffer = NULL;
438 context->config.inputCfg.bufferProvider.releaseBuffer = NULL;
439 context->config.inputCfg.bufferProvider.cookie = NULL;
440 context->config.inputCfg.mask = EFFECT_CONFIG_ALL;
441 context->config.outputCfg.accessMode = EFFECT_BUFFER_ACCESS_ACCUMULATE;
442 context->config.outputCfg.channels = AUDIO_CHANNEL_OUT_STEREO;
443 context->config.outputCfg.format = AUDIO_FORMAT_PCM_16_BIT;
444 context->config.outputCfg.samplingRate = 44100;
445 context->config.outputCfg.bufferProvider.getBuffer = NULL;
446 context->config.outputCfg.bufferProvider.releaseBuffer = NULL;
447 context->config.outputCfg.bufferProvider.cookie = NULL;
448 context->config.outputCfg.mask = EFFECT_CONFIG_ALL;
449
450 set_config(context, &context->config);
451
452 memset(&(eq_ctxt->offload_eq), 0, sizeof(struct eq_params));
453 offload_eq_set_preset(&(eq_ctxt->offload_eq), INVALID_PRESET);
454
455 return 0;
456}
457
458int equalizer_enable(effect_context_t *context)
459{
460 equalizer_context_t *eq_ctxt = (equalizer_context_t *)context;
461
462 ALOGV("%s", __func__);
463
464 if (!offload_eq_get_enable_flag(&(eq_ctxt->offload_eq))) {
465 offload_eq_set_enable_flag(&(eq_ctxt->offload_eq), true);
466 if (eq_ctxt->ctl)
467 offload_eq_send_params(eq_ctxt->ctl, eq_ctxt->offload_eq,
468 OFFLOAD_SEND_EQ_ENABLE_FLAG |
469 OFFLOAD_SEND_EQ_BANDS_LEVEL);
470 }
471 return 0;
472}
473
474int equalizer_disable(effect_context_t *context)
475{
476 equalizer_context_t *eq_ctxt = (equalizer_context_t *)context;
477
478 ALOGV("%s", __func__);
479 if (offload_eq_get_enable_flag(&(eq_ctxt->offload_eq))) {
480 offload_eq_set_enable_flag(&(eq_ctxt->offload_eq), false);
481 if (eq_ctxt->ctl)
482 offload_eq_send_params(eq_ctxt->ctl, eq_ctxt->offload_eq,
483 OFFLOAD_SEND_EQ_ENABLE_FLAG);
484 }
485 return 0;
486}
487
488int equalizer_start(effect_context_t *context, output_context_t *output)
489{
490 equalizer_context_t *eq_ctxt = (equalizer_context_t *)context;
491
492 ALOGV("%s: %p", __func__, output->ctl);
493 eq_ctxt->ctl = output->ctl;
Amit Shekhard02f2cd2014-01-16 16:51:43 -0800494 if (offload_eq_get_enable_flag(&(eq_ctxt->offload_eq)))
495 if (eq_ctxt->ctl)
496 offload_eq_send_params(eq_ctxt->ctl, eq_ctxt->offload_eq,
497 OFFLOAD_SEND_EQ_ENABLE_FLAG |
498 OFFLOAD_SEND_EQ_BANDS_LEVEL);
Subhash Chandra Bose Naripeddy3eedc002013-11-12 20:45:15 -0800499 return 0;
500}
501
502int equalizer_stop(effect_context_t *context, output_context_t *output)
503{
504 equalizer_context_t *eq_ctxt = (equalizer_context_t *)context;
505
506 ALOGV("%s", __func__);
507 eq_ctxt->ctl = NULL;
508 return 0;
509}