blob: 352c4d8e6a993c3ccc3a2d412752d2c9371edd9e [file] [log] [blame]
Eric Laurentb23d5282013-05-14 15:27:20 -07001/*
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -07002 * Copyright (c) 2013, The Linux Foundation. All rights reserved.
3 * Not a Contribution.
4 *
Eric Laurentb23d5282013-05-14 15:27:20 -07005 * 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 "msm8974_platform"
21/*#define LOG_NDEBUG 0*/
22#define LOG_NDDEBUG 0
23
24#include <stdlib.h>
25#include <dlfcn.h>
26#include <cutils/log.h>
27#include <cutils/properties.h>
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -070028#include <cutils/str_parms.h>
Eric Laurentb23d5282013-05-14 15:27:20 -070029#include <audio_hw.h>
30#include <platform_api.h>
31#include "platform.h"
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -070032#include "audio_extn.h"
Narsinga Rao Chella05573b72013-11-15 15:21:40 -080033#include "voice_extn.h"
Eric Laurentb23d5282013-05-14 15:27:20 -070034
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -070035#define MIXER_XML_PATH "/system/etc/mixer_paths.xml"
Damir Didjustof1d46c72013-11-06 17:59:04 -080036#define MIXER_XML_PATH_AUXPCM "/system/etc/mixer_paths_auxpcm.xml"
Eric Laurentb23d5282013-05-14 15:27:20 -070037#define LIB_ACDB_LOADER "libacdbloader.so"
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -070038#define AUDIO_DATA_BLOCK_MIXER_CTL "HDMI EDID"
Eric Laurentb23d5282013-05-14 15:27:20 -070039
Eric Laurentb23d5282013-05-14 15:27:20 -070040/*
Eric Laurentb23d5282013-05-14 15:27:20 -070041 * This file will have a maximum of 38 bytes:
42 *
43 * 4 bytes: number of audio blocks
44 * 4 bytes: total length of Short Audio Descriptor (SAD) blocks
45 * Maximum 10 * 3 bytes: SAD blocks
46 */
47#define MAX_SAD_BLOCKS 10
48#define SAD_BLOCK_SIZE 3
49
50/* EDID format ID for LPCM audio */
51#define EDID_FORMAT_LPCM 1
52
sangwoo1b9f4b32013-06-21 18:22:55 -070053/* Retry for delay in FW loading*/
54#define RETRY_NUMBER 10
55#define RETRY_US 500000
56
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -070057#define SAMPLE_RATE_8KHZ 8000
58#define SAMPLE_RATE_16KHZ 16000
59
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -070060#define AUDIO_PARAMETER_KEY_FLUENCE_TYPE "fluence"
61#define AUDIO_PARAMETER_KEY_BTSCO "bt_samplerate"
62#define AUDIO_PARAMETER_KEY_SLOWTALK "st_enable"
sangwoo53b2cf02013-07-25 19:18:44 -070063
Eric Laurentb23d5282013-05-14 15:27:20 -070064struct audio_block_header
65{
66 int reserved;
67 int length;
68};
69
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -080070/* Audio calibration related functions */
Eric Laurentb23d5282013-05-14 15:27:20 -070071typedef void (*acdb_deallocate_t)();
72typedef int (*acdb_init_t)();
73typedef void (*acdb_send_audio_cal_t)(int, int);
74typedef void (*acdb_send_voice_cal_t)(int, int);
75
Eric Laurentb23d5282013-05-14 15:27:20 -070076struct platform_data {
77 struct audio_device *adev;
78 bool fluence_in_spkr_mode;
79 bool fluence_in_voice_call;
80 bool fluence_in_voice_rec;
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -080081 bool fluence_in_audio_rec;
Mingming Yin8e5a4f62013-10-07 15:23:41 -070082 int fluence_type;
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -070083 int btsco_sample_rate;
84 bool slowtalk;
Ravi Kumar Alamanda48c921d2013-10-29 06:07:44 -070085 /* Audio calibration related functions */
Eric Laurentb23d5282013-05-14 15:27:20 -070086 void *acdb_handle;
87 acdb_init_t acdb_init;
88 acdb_deallocate_t acdb_deallocate;
89 acdb_send_audio_cal_t acdb_send_audio_cal;
90 acdb_send_voice_cal_t acdb_send_voice_cal;
Ravi Kumar Alamanda48c921d2013-10-29 06:07:44 -070091
92 void *hw_info;
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -080093 struct csd_data *csd;
Eric Laurentb23d5282013-05-14 15:27:20 -070094};
95
96static const int pcm_device_table[AUDIO_USECASE_MAX][2] = {
Mingming Yin8e5a4f62013-10-07 15:23:41 -070097 [USECASE_AUDIO_PLAYBACK_DEEP_BUFFER] = {DEEP_BUFFER_PCM_DEVICE,
98 DEEP_BUFFER_PCM_DEVICE},
99 [USECASE_AUDIO_PLAYBACK_LOW_LATENCY] = {LOWLATENCY_PCM_DEVICE,
Preetam Singh Ranawatde84f1a2013-11-01 14:58:16 -0700100 LOWLATENCY_PCM_DEVICE},
101 [USECASE_AUDIO_PLAYBACK_MULTI_CH] = {MULTIMEDIA2_PCM_DEVICE,
102 MULTIMEDIA2_PCM_DEVICE},
Krishnankutty Kolathappillya43f96e2013-11-01 12:17:53 -0700103 [USECASE_AUDIO_PLAYBACK_OFFLOAD] =
104 {PLAYBACK_OFFLOAD_DEVICE, PLAYBACK_OFFLOAD_DEVICE},
Shiv Maliyappanahallida107642013-10-17 11:16:13 -0700105 [USECASE_AUDIO_RECORD] = {AUDIO_RECORD_PCM_DEVICE, AUDIO_RECORD_PCM_DEVICE},
Mingming Yine62d7842013-10-25 16:26:03 -0700106 [USECASE_AUDIO_RECORD_COMPRESS] = {COMPRESS_CAPTURE_DEVICE, COMPRESS_CAPTURE_DEVICE},
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700107 [USECASE_AUDIO_RECORD_LOW_LATENCY] = {LOWLATENCY_PCM_DEVICE,
108 LOWLATENCY_PCM_DEVICE},
Preetam Singh Ranawatde84f1a2013-11-01 14:58:16 -0700109 [USECASE_AUDIO_RECORD_FM_VIRTUAL] = {MULTIMEDIA2_PCM_DEVICE,
110 MULTIMEDIA2_PCM_DEVICE},
Apoorv Raghuvanshi6e262842013-10-06 14:39:35 -0700111 [USECASE_AUDIO_PLAYBACK_FM] = {FM_PLAYBACK_PCM_DEVICE, FM_CAPTURE_PCM_DEVICE},
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700112 [USECASE_VOICE_CALL] = {VOICE_CALL_PCM_DEVICE, VOICE_CALL_PCM_DEVICE},
113 [USECASE_VOICE2_CALL] = {VOICE2_CALL_PCM_DEVICE, VOICE2_CALL_PCM_DEVICE},
114 [USECASE_VOLTE_CALL] = {VOLTE_CALL_PCM_DEVICE, VOLTE_CALL_PCM_DEVICE},
115 [USECASE_QCHAT_CALL] = {QCHAT_CALL_PCM_DEVICE, QCHAT_CALL_PCM_DEVICE},
Narsinga Rao Chella05573b72013-11-15 15:21:40 -0800116 [USECASE_COMPRESS_VOIP_CALL] = {COMPRESS_VOIP_CALL_PCM_DEVICE, COMPRESS_VOIP_CALL_PCM_DEVICE},
Shiv Maliyappanahallida107642013-10-17 11:16:13 -0700117 [USECASE_INCALL_REC_UPLINK] = {AUDIO_RECORD_PCM_DEVICE,
118 AUDIO_RECORD_PCM_DEVICE},
119 [USECASE_INCALL_REC_DOWNLINK] = {AUDIO_RECORD_PCM_DEVICE,
120 AUDIO_RECORD_PCM_DEVICE},
121 [USECASE_INCALL_REC_UPLINK_AND_DOWNLINK] = {AUDIO_RECORD_PCM_DEVICE,
122 AUDIO_RECORD_PCM_DEVICE},
Shiv Maliyappanahallif3b9a422013-10-22 16:38:08 -0700123 [USECASE_INCALL_MUSIC_UPLINK] = {INCALL_MUSIC_UPLINK_PCM_DEVICE,
124 INCALL_MUSIC_UPLINK_PCM_DEVICE},
125 [USECASE_INCALL_MUSIC_UPLINK2] = {INCALL_MUSIC_UPLINK2_PCM_DEVICE,
126 INCALL_MUSIC_UPLINK2_PCM_DEVICE},
Gopikrishnaiah Anandanf538cef2013-10-28 14:06:03 -0700127 [USECASE_AUDIO_SPKR_CALIB_RX] = {SPKR_PROT_CALIB_RX_PCM_DEVICE, -1},
128 [USECASE_AUDIO_SPKR_CALIB_TX] = {-1, SPKR_PROT_CALIB_TX_PCM_DEVICE},
Eric Laurentb23d5282013-05-14 15:27:20 -0700129};
130
131/* Array to store sound devices */
132static const char * const device_table[SND_DEVICE_MAX] = {
133 [SND_DEVICE_NONE] = "none",
134 /* Playback sound devices */
135 [SND_DEVICE_OUT_HANDSET] = "handset",
136 [SND_DEVICE_OUT_SPEAKER] = "speaker",
137 [SND_DEVICE_OUT_SPEAKER_REVERSE] = "speaker-reverse",
138 [SND_DEVICE_OUT_HEADPHONES] = "headphones",
139 [SND_DEVICE_OUT_SPEAKER_AND_HEADPHONES] = "speaker-and-headphones",
140 [SND_DEVICE_OUT_VOICE_HANDSET] = "voice-handset",
141 [SND_DEVICE_OUT_VOICE_SPEAKER] = "voice-speaker",
142 [SND_DEVICE_OUT_VOICE_HEADPHONES] = "voice-headphones",
143 [SND_DEVICE_OUT_HDMI] = "hdmi",
144 [SND_DEVICE_OUT_SPEAKER_AND_HDMI] = "speaker-and-hdmi",
145 [SND_DEVICE_OUT_BT_SCO] = "bt-sco-headset",
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700146 [SND_DEVICE_OUT_BT_SCO_WB] = "bt-sco-headset-wb",
Eric Laurentb23d5282013-05-14 15:27:20 -0700147 [SND_DEVICE_OUT_VOICE_TTY_FULL_HEADPHONES] = "voice-tty-full-headphones",
148 [SND_DEVICE_OUT_VOICE_TTY_VCO_HEADPHONES] = "voice-tty-vco-headphones",
149 [SND_DEVICE_OUT_VOICE_TTY_HCO_HANDSET] = "voice-tty-hco-handset",
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700150 [SND_DEVICE_OUT_AFE_PROXY] = "afe-proxy",
151 [SND_DEVICE_OUT_USB_HEADSET] = "usb-headphones",
152 [SND_DEVICE_OUT_SPEAKER_AND_USB_HEADSET] = "speaker-and-usb-headphones",
Apoorv Raghuvanshi6e262842013-10-06 14:39:35 -0700153 [SND_DEVICE_OUT_TRANSMISSION_FM] = "transmission-fm",
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700154 [SND_DEVICE_OUT_ANC_HEADSET] = "anc-headphones",
155 [SND_DEVICE_OUT_ANC_FB_HEADSET] = "anc-fb-headphones",
156 [SND_DEVICE_OUT_VOICE_ANC_HEADSET] = "voice-anc-headphones",
157 [SND_DEVICE_OUT_VOICE_ANC_FB_HEADSET] = "voice-anc-fb-headphones",
158 [SND_DEVICE_OUT_SPEAKER_AND_ANC_HEADSET] = "speaker-and-anc-headphones",
159 [SND_DEVICE_OUT_ANC_HANDSET] = "anc-handset",
Gopikrishnaiah Anandanf538cef2013-10-28 14:06:03 -0700160 [SND_DEVICE_OUT_SPEAKER_PROTECTED] = "speaker-protected",
Eric Laurentb23d5282013-05-14 15:27:20 -0700161
162 /* Capture sound devices */
163 [SND_DEVICE_IN_HANDSET_MIC] = "handset-mic",
Eric Laurentb23d5282013-05-14 15:27:20 -0700164 [SND_DEVICE_IN_HANDSET_MIC_AEC] = "handset-mic",
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800165 [SND_DEVICE_IN_HANDSET_MIC_NS] = "handset-mic",
166 [SND_DEVICE_IN_HANDSET_MIC_AEC_NS] = "handset-mic",
167 [SND_DEVICE_IN_HANDSET_DMIC] = "dmic-endfire",
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800168 [SND_DEVICE_IN_HANDSET_DMIC_AEC] = "dmic-endfire",
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800169 [SND_DEVICE_IN_HANDSET_DMIC_NS] = "dmic-endfire",
170 [SND_DEVICE_IN_HANDSET_DMIC_AEC_NS] = "dmic-endfire",
171 [SND_DEVICE_IN_SPEAKER_MIC] = "speaker-mic",
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800172 [SND_DEVICE_IN_SPEAKER_MIC_AEC] = "speaker-mic",
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800173 [SND_DEVICE_IN_SPEAKER_MIC_NS] = "speaker-mic",
174 [SND_DEVICE_IN_SPEAKER_MIC_AEC_NS] = "speaker-mic",
175 [SND_DEVICE_IN_SPEAKER_DMIC] = "speaker-dmic-endfire",
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800176 [SND_DEVICE_IN_SPEAKER_DMIC_AEC] = "speaker-dmic-endfire",
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800177 [SND_DEVICE_IN_SPEAKER_DMIC_NS] = "speaker-dmic-endfire",
178 [SND_DEVICE_IN_SPEAKER_DMIC_AEC_NS] = "speaker-dmic-endfire",
179 [SND_DEVICE_IN_HEADSET_MIC] = "headset-mic",
180 [SND_DEVICE_IN_HEADSET_MIC_FLUENCE] = "headset-mic",
Eric Laurentb23d5282013-05-14 15:27:20 -0700181 [SND_DEVICE_IN_VOICE_SPEAKER_MIC] = "voice-speaker-mic",
182 [SND_DEVICE_IN_VOICE_HEADSET_MIC] = "voice-headset-mic",
183 [SND_DEVICE_IN_HDMI_MIC] = "hdmi-mic",
184 [SND_DEVICE_IN_BT_SCO_MIC] = "bt-sco-mic",
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700185 [SND_DEVICE_IN_BT_SCO_MIC_WB] = "bt-sco-mic-wb",
Eric Laurentb23d5282013-05-14 15:27:20 -0700186 [SND_DEVICE_IN_CAMCORDER_MIC] = "camcorder-mic",
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700187 [SND_DEVICE_IN_VOICE_DMIC] = "voice-dmic-ef",
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700188 [SND_DEVICE_IN_VOICE_SPEAKER_DMIC] = "voice-speaker-dmic-ef",
Eric Laurentb23d5282013-05-14 15:27:20 -0700189 [SND_DEVICE_IN_VOICE_TTY_FULL_HEADSET_MIC] = "voice-tty-full-headset-mic",
190 [SND_DEVICE_IN_VOICE_TTY_VCO_HANDSET_MIC] = "voice-tty-vco-handset-mic",
191 [SND_DEVICE_IN_VOICE_TTY_HCO_HEADSET_MIC] = "voice-tty-hco-headset-mic",
192 [SND_DEVICE_IN_VOICE_REC_MIC] = "voice-rec-mic",
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800193 [SND_DEVICE_IN_VOICE_REC_MIC_NS] = "voice-rec-mic",
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800194 [SND_DEVICE_IN_VOICE_REC_DMIC_STEREO] = "voice-rec-dmic-ef",
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700195 [SND_DEVICE_IN_VOICE_REC_DMIC_FLUENCE] = "voice-rec-dmic-ef-fluence",
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700196 [SND_DEVICE_IN_USB_HEADSET_MIC] = "usb-headset-mic",
Apoorv Raghuvanshi6e262842013-10-06 14:39:35 -0700197 [SND_DEVICE_IN_CAPTURE_FM] = "capture-fm",
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700198 [SND_DEVICE_IN_AANC_HANDSET_MIC] = "aanc-handset-mic",
Apoorv Raghuvanshi6178a3f2013-10-19 12:38:54 -0700199 [SND_DEVICE_IN_QUAD_MIC] = "quad-mic",
Apoorv Raghuvanshi6bd8dbf2013-10-19 18:37:52 -0700200 [SND_DEVICE_IN_HANDSET_STEREO_DMIC] = "handset-stereo-dmic-ef",
201 [SND_DEVICE_IN_SPEAKER_STEREO_DMIC] = "speaker-stereo-dmic-ef",
Gopikrishnaiah Anandanf538cef2013-10-28 14:06:03 -0700202 [SND_DEVICE_IN_CAPTURE_VI_FEEDBACK] = "vi-feedback",
Eric Laurentb23d5282013-05-14 15:27:20 -0700203};
204
205/* ACDB IDs (audio DSP path configuration IDs) for each sound device */
206static const int acdb_device_table[SND_DEVICE_MAX] = {
207 [SND_DEVICE_NONE] = -1,
208 [SND_DEVICE_OUT_HANDSET] = 7,
Vidyakumar Athotac29d4ab2013-11-14 16:58:02 -0800209 [SND_DEVICE_OUT_SPEAKER] = 14,
210 [SND_DEVICE_OUT_SPEAKER_REVERSE] = 14,
Eric Laurentb23d5282013-05-14 15:27:20 -0700211 [SND_DEVICE_OUT_HEADPHONES] = 10,
212 [SND_DEVICE_OUT_SPEAKER_AND_HEADPHONES] = 10,
213 [SND_DEVICE_OUT_VOICE_HANDSET] = 7,
Vidyakumar Athotac29d4ab2013-11-14 16:58:02 -0800214 [SND_DEVICE_OUT_VOICE_SPEAKER] = 14,
Eric Laurentb23d5282013-05-14 15:27:20 -0700215 [SND_DEVICE_OUT_VOICE_HEADPHONES] = 10,
216 [SND_DEVICE_OUT_HDMI] = 18,
Vidyakumar Athotac29d4ab2013-11-14 16:58:02 -0800217 [SND_DEVICE_OUT_SPEAKER_AND_HDMI] = 14,
Eric Laurentb23d5282013-05-14 15:27:20 -0700218 [SND_DEVICE_OUT_BT_SCO] = 22,
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700219 [SND_DEVICE_OUT_BT_SCO_WB] = 39,
Eric Laurentb23d5282013-05-14 15:27:20 -0700220 [SND_DEVICE_OUT_VOICE_TTY_FULL_HEADPHONES] = 17,
221 [SND_DEVICE_OUT_VOICE_TTY_VCO_HEADPHONES] = 17,
222 [SND_DEVICE_OUT_VOICE_TTY_HCO_HANDSET] = 37,
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700223 [SND_DEVICE_OUT_AFE_PROXY] = 0,
224 [SND_DEVICE_OUT_USB_HEADSET] = 0,
225 [SND_DEVICE_OUT_SPEAKER_AND_USB_HEADSET] = 14,
Apoorv Raghuvanshi6e262842013-10-06 14:39:35 -0700226 [SND_DEVICE_OUT_TRANSMISSION_FM] = 0,
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700227 [SND_DEVICE_OUT_ANC_HEADSET] = 26,
Ravi Kumar Alamanda48c921d2013-10-29 06:07:44 -0700228 [SND_DEVICE_OUT_ANC_FB_HEADSET] = 27,
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700229 [SND_DEVICE_OUT_VOICE_ANC_HEADSET] = 26,
Ravi Kumar Alamanda48c921d2013-10-29 06:07:44 -0700230 [SND_DEVICE_OUT_VOICE_ANC_FB_HEADSET] = 27,
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700231 [SND_DEVICE_OUT_SPEAKER_AND_ANC_HEADSET] = 26,
232 [SND_DEVICE_OUT_ANC_HANDSET] = 103,
Gopikrishnaiah Anandanf538cef2013-10-28 14:06:03 -0700233 [SND_DEVICE_OUT_SPEAKER_PROTECTED] = 101,
Eric Laurentb23d5282013-05-14 15:27:20 -0700234
235 [SND_DEVICE_IN_HANDSET_MIC] = 4,
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800236 [SND_DEVICE_IN_HANDSET_MIC_AEC] = 106,
237 [SND_DEVICE_IN_HANDSET_MIC_NS] = 107,
238 [SND_DEVICE_IN_HANDSET_MIC_AEC_NS] = 108,
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800239 [SND_DEVICE_IN_HANDSET_DMIC] = 41,
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800240 [SND_DEVICE_IN_HANDSET_DMIC_AEC] = 109,
241 [SND_DEVICE_IN_HANDSET_DMIC_NS] = 110,
242 [SND_DEVICE_IN_HANDSET_DMIC_AEC_NS] = 111,
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800243 [SND_DEVICE_IN_SPEAKER_MIC] = 11,
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800244 [SND_DEVICE_IN_SPEAKER_MIC_AEC] = 112,
245 [SND_DEVICE_IN_SPEAKER_MIC_NS] = 113,
246 [SND_DEVICE_IN_SPEAKER_MIC_AEC_NS] = 114,
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800247 [SND_DEVICE_IN_SPEAKER_DMIC] = 43,
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800248 [SND_DEVICE_IN_SPEAKER_DMIC_AEC] = 115,
249 [SND_DEVICE_IN_SPEAKER_DMIC_NS] = 116,
250 [SND_DEVICE_IN_SPEAKER_DMIC_AEC_NS] = 117,
Eric Laurentb23d5282013-05-14 15:27:20 -0700251 [SND_DEVICE_IN_HEADSET_MIC] = 8,
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800252 [SND_DEVICE_IN_HEADSET_MIC_FLUENCE] = 47,
Eric Laurentb23d5282013-05-14 15:27:20 -0700253 [SND_DEVICE_IN_VOICE_SPEAKER_MIC] = 11,
254 [SND_DEVICE_IN_VOICE_HEADSET_MIC] = 8,
255 [SND_DEVICE_IN_HDMI_MIC] = 4,
256 [SND_DEVICE_IN_BT_SCO_MIC] = 21,
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700257 [SND_DEVICE_IN_BT_SCO_MIC_WB] = 38,
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800258 [SND_DEVICE_IN_CAMCORDER_MIC] = 4,
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700259 [SND_DEVICE_IN_VOICE_DMIC] = 41,
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700260 [SND_DEVICE_IN_VOICE_SPEAKER_DMIC] = 43,
Eric Laurentb23d5282013-05-14 15:27:20 -0700261 [SND_DEVICE_IN_VOICE_TTY_FULL_HEADSET_MIC] = 16,
262 [SND_DEVICE_IN_VOICE_TTY_VCO_HANDSET_MIC] = 36,
263 [SND_DEVICE_IN_VOICE_TTY_HCO_HEADSET_MIC] = 16,
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800264 [SND_DEVICE_IN_VOICE_REC_MIC] = 4,
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800265 [SND_DEVICE_IN_VOICE_REC_MIC_NS] = 107,
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800266 [SND_DEVICE_IN_VOICE_REC_DMIC_STEREO] = 34,
267 [SND_DEVICE_IN_VOICE_REC_DMIC_FLUENCE] = 41,
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700268 [SND_DEVICE_IN_USB_HEADSET_MIC] = 44,
Apoorv Raghuvanshi6e262842013-10-06 14:39:35 -0700269 [SND_DEVICE_IN_CAPTURE_FM] = 0,
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700270 [SND_DEVICE_IN_AANC_HANDSET_MIC] = 104,
Apoorv Raghuvanshi6178a3f2013-10-19 12:38:54 -0700271 [SND_DEVICE_IN_QUAD_MIC] = 46,
Apoorv Raghuvanshi6bd8dbf2013-10-19 18:37:52 -0700272 [SND_DEVICE_IN_HANDSET_STEREO_DMIC] = 34,
273 [SND_DEVICE_IN_SPEAKER_STEREO_DMIC] = 35,
Gopikrishnaiah Anandanf538cef2013-10-28 14:06:03 -0700274 [SND_DEVICE_IN_CAPTURE_VI_FEEDBACK] = 102,
Eric Laurentb23d5282013-05-14 15:27:20 -0700275};
276
Haynes Mathew George7ff216f2013-09-11 19:51:41 -0700277#define DEEP_BUFFER_PLATFORM_DELAY (29*1000LL)
278#define LOW_LATENCY_PLATFORM_DELAY (13*1000LL)
279
Eric Laurentb23d5282013-05-14 15:27:20 -0700280static int set_echo_reference(struct mixer *mixer, const char* ec_ref)
281{
282 struct mixer_ctl *ctl;
283 const char *mixer_ctl_name = "EC_REF_RX";
284
285 ctl = mixer_get_ctl_by_name(mixer, mixer_ctl_name);
286 if (!ctl) {
287 ALOGE("%s: Could not get ctl for mixer cmd - %s",
288 __func__, mixer_ctl_name);
289 return -EINVAL;
290 }
291 ALOGV("Setting EC Reference: %s", ec_ref);
292 mixer_ctl_set_enum_by_string(ctl, ec_ref);
293 return 0;
294}
295
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800296static struct csd_data *open_csd_client()
297{
298 struct csd_data *csd = calloc(1, sizeof(struct csd_data));
299
300 csd->csd_client = dlopen(LIB_CSD_CLIENT, RTLD_NOW);
301 if (csd->csd_client == NULL) {
302 ALOGE("%s: DLOPEN failed for %s", __func__, LIB_CSD_CLIENT);
303 goto error;
304 } else {
305 ALOGV("%s: DLOPEN successful for %s", __func__, LIB_CSD_CLIENT);
306
307 csd->deinit = (deinit_t)dlsym(csd->csd_client,
308 "csd_client_deinit");
309 if (csd->deinit == NULL) {
310 ALOGE("%s: dlsym error %s for csd_client_deinit", __func__,
311 dlerror());
312 goto error;
313 }
314 csd->disable_device = (disable_device_t)dlsym(csd->csd_client,
315 "csd_client_disable_device");
316 if (csd->disable_device == NULL) {
317 ALOGE("%s: dlsym error %s for csd_client_disable_device",
318 __func__, dlerror());
319 goto error;
320 }
321 csd->enable_device = (enable_device_t)dlsym(csd->csd_client,
322 "csd_client_enable_device");
323 if (csd->enable_device == NULL) {
324 ALOGE("%s: dlsym error %s for csd_client_enable_device",
325 __func__, dlerror());
326 goto error;
327 }
328 csd->start_voice = (start_voice_t)dlsym(csd->csd_client,
329 "csd_client_start_voice");
330 if (csd->start_voice == NULL) {
331 ALOGE("%s: dlsym error %s for csd_client_start_voice",
332 __func__, dlerror());
333 goto error;
334 }
335 csd->stop_voice = (stop_voice_t)dlsym(csd->csd_client,
336 "csd_client_stop_voice");
337 if (csd->stop_voice == NULL) {
338 ALOGE("%s: dlsym error %s for csd_client_stop_voice",
339 __func__, dlerror());
340 goto error;
341 }
342 csd->volume = (volume_t)dlsym(csd->csd_client,
343 "csd_client_volume");
344 if (csd->volume == NULL) {
345 ALOGE("%s: dlsym error %s for csd_client_volume",
346 __func__, dlerror());
347 goto error;
348 }
349 csd->mic_mute = (mic_mute_t)dlsym(csd->csd_client,
350 "csd_client_mic_mute");
351 if (csd->mic_mute == NULL) {
352 ALOGE("%s: dlsym error %s for csd_client_mic_mute",
353 __func__, dlerror());
354 goto error;
355 }
356 csd->slow_talk = (slow_talk_t)dlsym(csd->csd_client,
357 "csd_client_slow_talk");
358 if (csd->slow_talk == NULL) {
359 ALOGE("%s: dlsym error %s for csd_client_slow_talk",
360 __func__, dlerror());
361 goto error;
362 }
363 csd->start_playback = (start_playback_t)dlsym(csd->csd_client,
364 "csd_client_start_playback");
365 if (csd->start_playback == NULL) {
366 ALOGE("%s: dlsym error %s for csd_client_start_playback",
367 __func__, dlerror());
368 goto error;
369 }
370 csd->stop_playback = (stop_playback_t)dlsym(csd->csd_client,
371 "csd_client_stop_playback");
372 if (csd->stop_playback == NULL) {
373 ALOGE("%s: dlsym error %s for csd_client_stop_playback",
374 __func__, dlerror());
375 goto error;
376 }
377 csd->start_record = (start_record_t)dlsym(csd->csd_client,
378 "csd_client_start_record");
379 if (csd->start_record == NULL) {
380 ALOGE("%s: dlsym error %s for csd_client_start_record",
381 __func__, dlerror());
382 goto error;
383 }
384 csd->stop_record = (stop_record_t)dlsym(csd->csd_client,
385 "csd_client_stop_record");
386 if (csd->stop_record == NULL) {
387 ALOGE("%s: dlsym error %s for csd_client_stop_record",
388 __func__, dlerror());
389 goto error;
390 }
391 csd->init = (init_t)dlsym(csd->csd_client, "csd_client_init");
392
393 if (csd->init == NULL) {
394 ALOGE("%s: dlsym error %s for csd_client_init",
395 __func__, dlerror());
396 goto error;
397 } else {
398 csd->init();
399 }
400 }
401 return csd;
402
403error:
404 free(csd);
405 csd = NULL;
406 return csd;
407}
408
409void close_csd_client(struct csd_data *csd)
410{
411 if (csd != NULL) {
412 csd->deinit();
413 dlclose(csd->csd_client);
414 free(csd);
415 csd = NULL;
416 }
417}
418
Eric Laurentb23d5282013-05-14 15:27:20 -0700419void *platform_init(struct audio_device *adev)
420{
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800421 char platform[PROPERTY_VALUE_MAX];
422 char baseband[PROPERTY_VALUE_MAX];
Eric Laurentb23d5282013-05-14 15:27:20 -0700423 char value[PROPERTY_VALUE_MAX];
424 struct platform_data *my_data;
Ravi Kumar Alamanda48c921d2013-10-29 06:07:44 -0700425 int retry_num = 0;
426 const char *snd_card_name;
sangwoo1b9f4b32013-06-21 18:22:55 -0700427
428 adev->mixer = mixer_open(MIXER_CARD);
429
430 while (!adev->mixer && retry_num < RETRY_NUMBER) {
431 usleep(RETRY_US);
432 adev->mixer = mixer_open(MIXER_CARD);
433 retry_num++;
434 }
435
436 if (!adev->mixer) {
437 ALOGE("Unable to open the mixer, aborting.");
438 return NULL;
439 }
440
Damir Didjustof1d46c72013-11-06 17:59:04 -0800441 if (audio_extn_read_xml(adev, MIXER_CARD, MIXER_XML_PATH,
442 MIXER_XML_PATH_AUXPCM) == -ENOSYS)
443 adev->audio_route = audio_route_init(MIXER_CARD, MIXER_XML_PATH);
444
sangwoo1b9f4b32013-06-21 18:22:55 -0700445 if (!adev->audio_route) {
446 ALOGE("%s: Failed to init audio route controls, aborting.", __func__);
447 return NULL;
448 }
Eric Laurentb23d5282013-05-14 15:27:20 -0700449
450 my_data = calloc(1, sizeof(struct platform_data));
451
Ravi Kumar Alamanda48c921d2013-10-29 06:07:44 -0700452 snd_card_name = mixer_get_name(adev->mixer);
453 my_data->hw_info = hw_info_init(snd_card_name);
454 if (!my_data->hw_info) {
455 ALOGE("%s: Failed to init hardware info", __func__);
456 }
457
Eric Laurentb23d5282013-05-14 15:27:20 -0700458 my_data->adev = adev;
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700459 my_data->btsco_sample_rate = SAMPLE_RATE_8KHZ;
Eric Laurentb23d5282013-05-14 15:27:20 -0700460 my_data->fluence_in_spkr_mode = false;
461 my_data->fluence_in_voice_call = false;
462 my_data->fluence_in_voice_rec = false;
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800463 my_data->fluence_in_audio_rec = false;
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700464 my_data->fluence_type = FLUENCE_NONE;
Eric Laurentb23d5282013-05-14 15:27:20 -0700465
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700466 property_get("ro.qc.sdk.audio.fluencetype", value, "");
467 if (!strncmp("fluencepro", value, sizeof("fluencepro"))) {
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800468 my_data->fluence_type = FLUENCE_QUAD_MIC | FLUENCE_DUAL_MIC;
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700469 } else if (!strncmp("fluence", value, sizeof("fluence"))) {
470 my_data->fluence_type = FLUENCE_DUAL_MIC;
471 } else {
472 my_data->fluence_type = FLUENCE_NONE;
Eric Laurentb23d5282013-05-14 15:27:20 -0700473 }
474
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700475 if (my_data->fluence_type != FLUENCE_NONE) {
Eric Laurentb23d5282013-05-14 15:27:20 -0700476 property_get("persist.audio.fluence.voicecall",value,"");
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700477 if (!strncmp("true", value, sizeof("true"))) {
Eric Laurentb23d5282013-05-14 15:27:20 -0700478 my_data->fluence_in_voice_call = true;
479 }
480
481 property_get("persist.audio.fluence.voicerec",value,"");
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700482 if (!strncmp("true", value, sizeof("true"))) {
Eric Laurentb23d5282013-05-14 15:27:20 -0700483 my_data->fluence_in_voice_rec = true;
484 }
485
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800486 property_get("persist.audio.fluence.audiorec",value,"");
487 if (!strncmp("true", value, sizeof("true"))) {
488 my_data->fluence_in_audio_rec = true;
489 }
490
Eric Laurentb23d5282013-05-14 15:27:20 -0700491 property_get("persist.audio.fluence.speaker",value,"");
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700492 if (!strncmp("true", value, sizeof("true"))) {
Eric Laurentb23d5282013-05-14 15:27:20 -0700493 my_data->fluence_in_spkr_mode = true;
494 }
495 }
496
497 my_data->acdb_handle = dlopen(LIB_ACDB_LOADER, RTLD_NOW);
498 if (my_data->acdb_handle == NULL) {
499 ALOGE("%s: DLOPEN failed for %s", __func__, LIB_ACDB_LOADER);
500 } else {
501 ALOGV("%s: DLOPEN successful for %s", __func__, LIB_ACDB_LOADER);
502 my_data->acdb_deallocate = (acdb_deallocate_t)dlsym(my_data->acdb_handle,
503 "acdb_loader_deallocate_ACDB");
504 my_data->acdb_send_audio_cal = (acdb_send_audio_cal_t)dlsym(my_data->acdb_handle,
505 "acdb_loader_send_audio_cal");
506 if (!my_data->acdb_send_audio_cal)
507 ALOGW("%s: Could not find the symbol acdb_send_audio_cal from %s",
508 __func__, LIB_ACDB_LOADER);
509 my_data->acdb_send_voice_cal = (acdb_send_voice_cal_t)dlsym(my_data->acdb_handle,
510 "acdb_loader_send_voice_cal");
511 my_data->acdb_init = (acdb_init_t)dlsym(my_data->acdb_handle,
512 "acdb_loader_init_ACDB");
513 if (my_data->acdb_init == NULL)
514 ALOGE("%s: dlsym error %s for acdb_loader_init_ACDB", __func__, dlerror());
515 else
516 my_data->acdb_init();
517 }
518
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800519 /* If platform is apq8084 and baseband is MDM, load CSD Client specific
520 * symbols. Voice call is handled by MDM and apps processor talks to
521 * MDM through CSD Client
522 */
523 property_get("ro.board.platform", platform, "");
524 property_get("ro.baseband", baseband, "");
525 if (!strncmp("apq8084", platform, sizeof("apq8084")) &&
526 !strncmp("mdm", baseband, sizeof("mdm"))) {
527 my_data->csd = open_csd_client();
528 }
529
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700530 /* init usb */
531 audio_extn_usb_init(adev);
532
Apoorv Raghuvanshi6178a3f2013-10-19 12:38:54 -0700533 /* Read one time ssr property */
534 audio_extn_ssr_update_enabled(adev);
Gopikrishnaiah Anandanf538cef2013-10-28 14:06:03 -0700535 audio_extn_spkr_prot_init(adev);
Eric Laurentb23d5282013-05-14 15:27:20 -0700536 return my_data;
537}
538
539void platform_deinit(void *platform)
540{
Ravi Kumar Alamanda48c921d2013-10-29 06:07:44 -0700541 struct platform_data *my_data = (struct platform_data *)platform;
542
543 hw_info_deinit(my_data->hw_info);
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800544 close_csd_client(my_data->csd);
545
Eric Laurentb23d5282013-05-14 15:27:20 -0700546 free(platform);
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700547 /* deinit usb */
548 audio_extn_usb_deinit();
Eric Laurentb23d5282013-05-14 15:27:20 -0700549}
550
551const char *platform_get_snd_device_name(snd_device_t snd_device)
552{
553 if (snd_device >= SND_DEVICE_MIN && snd_device < SND_DEVICE_MAX)
554 return device_table[snd_device];
555 else
556 return "";
557}
558
Ravi Kumar Alamanda48c921d2013-10-29 06:07:44 -0700559int platform_get_snd_device_name_extn(void *platform, snd_device_t snd_device,
560 char *device_name)
561{
562 struct platform_data *my_data = (struct platform_data *)platform;
563
564 if (snd_device >= SND_DEVICE_MIN && snd_device < SND_DEVICE_MAX) {
565 strlcpy(device_name, device_table[snd_device], DEVICE_NAME_MAX_SIZE);
566 hw_info_append_hw_type(my_data->hw_info, snd_device, device_name);
567 } else {
568 strlcpy(device_name, "", DEVICE_NAME_MAX_SIZE);
569 return -EINVAL;
570 }
571
572 return 0;
573}
574
Eric Laurentb23d5282013-05-14 15:27:20 -0700575void platform_add_backend_name(char *mixer_path, snd_device_t snd_device)
576{
577 if (snd_device == SND_DEVICE_IN_BT_SCO_MIC)
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700578 strlcat(mixer_path, " bt-sco", MIXER_PATH_MAX_LENGTH);
579 else if (snd_device == SND_DEVICE_IN_BT_SCO_MIC_WB)
580 strlcat(mixer_path, " bt-sco-wb", MIXER_PATH_MAX_LENGTH);
Eric Laurentb23d5282013-05-14 15:27:20 -0700581 else if(snd_device == SND_DEVICE_OUT_BT_SCO)
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700582 strlcat(mixer_path, " bt-sco", MIXER_PATH_MAX_LENGTH);
583 else if(snd_device == SND_DEVICE_OUT_BT_SCO_WB)
584 strlcat(mixer_path, " bt-sco-wb", MIXER_PATH_MAX_LENGTH);
Eric Laurentb23d5282013-05-14 15:27:20 -0700585 else if (snd_device == SND_DEVICE_OUT_HDMI)
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700586 strlcat(mixer_path, " hdmi", MIXER_PATH_MAX_LENGTH);
Eric Laurentb23d5282013-05-14 15:27:20 -0700587 else if (snd_device == SND_DEVICE_OUT_SPEAKER_AND_HDMI)
588 strcat(mixer_path, " speaker-and-hdmi");
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700589 else if (snd_device == SND_DEVICE_OUT_AFE_PROXY)
590 strlcat(mixer_path, " afe-proxy", MIXER_PATH_MAX_LENGTH);
591 else if (snd_device == SND_DEVICE_OUT_USB_HEADSET)
592 strlcat(mixer_path, " usb-headphones", MIXER_PATH_MAX_LENGTH);
593 else if (snd_device == SND_DEVICE_OUT_SPEAKER_AND_USB_HEADSET)
594 strlcat(mixer_path, " speaker-and-usb-headphones",
595 MIXER_PATH_MAX_LENGTH);
596 else if (snd_device == SND_DEVICE_IN_USB_HEADSET_MIC)
597 strlcat(mixer_path, " usb-headset-mic", MIXER_PATH_MAX_LENGTH);
Apoorv Raghuvanshi6e262842013-10-06 14:39:35 -0700598 else if (snd_device == SND_DEVICE_IN_CAPTURE_FM)
599 strlcat(mixer_path, " capture-fm", MIXER_PATH_MAX_LENGTH);
600 else if (snd_device == SND_DEVICE_OUT_TRANSMISSION_FM)
601 strlcat(mixer_path, " transmission-fm", MIXER_PATH_MAX_LENGTH);
Eric Laurentb23d5282013-05-14 15:27:20 -0700602}
603
604int platform_get_pcm_device_id(audio_usecase_t usecase, int device_type)
605{
606 int device_id;
607 if (device_type == PCM_PLAYBACK)
608 device_id = pcm_device_table[usecase][0];
609 else
610 device_id = pcm_device_table[usecase][1];
611 return device_id;
612}
613
614int platform_send_audio_calibration(void *platform, snd_device_t snd_device)
615{
616 struct platform_data *my_data = (struct platform_data *)platform;
617 int acdb_dev_id, acdb_dev_type;
618
619 acdb_dev_id = acdb_device_table[snd_device];
620 if (acdb_dev_id < 0) {
621 ALOGE("%s: Could not find acdb id for device(%d)",
622 __func__, snd_device);
623 return -EINVAL;
624 }
625 if (my_data->acdb_send_audio_cal) {
Eric Laurent994a6932013-07-17 11:51:42 -0700626 ("%s: sending audio calibration for snd_device(%d) acdb_id(%d)",
Eric Laurentb23d5282013-05-14 15:27:20 -0700627 __func__, snd_device, acdb_dev_id);
628 if (snd_device >= SND_DEVICE_OUT_BEGIN &&
629 snd_device < SND_DEVICE_OUT_END)
630 acdb_dev_type = ACDB_DEV_TYPE_OUT;
631 else
632 acdb_dev_type = ACDB_DEV_TYPE_IN;
633 my_data->acdb_send_audio_cal(acdb_dev_id, acdb_dev_type);
634 }
635 return 0;
636}
637
638int platform_switch_voice_call_device_pre(void *platform)
639{
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800640 struct platform_data *my_data = (struct platform_data *)platform;
641 int ret = 0;
642
643 if (my_data->csd != NULL) {
644 /* This must be called before disabling mixer controls on APQ side */
645 ret = my_data->csd->disable_device();
646 if (ret < 0) {
647 ALOGE("%s: csd_client_disable_device, failed, error %d",
648 __func__, ret);
649 }
650 }
651 return ret;
Eric Laurentb23d5282013-05-14 15:27:20 -0700652}
653
654int platform_switch_voice_call_device_post(void *platform,
655 snd_device_t out_snd_device,
656 snd_device_t in_snd_device)
657{
658 struct platform_data *my_data = (struct platform_data *)platform;
659 int acdb_rx_id, acdb_tx_id;
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800660 int ret = 0;
661
662 acdb_rx_id = acdb_device_table[out_snd_device];
663 acdb_tx_id = acdb_device_table[in_snd_device];
Eric Laurentb23d5282013-05-14 15:27:20 -0700664
665 if (my_data->acdb_send_voice_cal == NULL) {
666 ALOGE("%s: dlsym error for acdb_send_voice_call", __func__);
667 } else {
Eric Laurentb23d5282013-05-14 15:27:20 -0700668 if (acdb_rx_id > 0 && acdb_tx_id > 0)
669 my_data->acdb_send_voice_cal(acdb_rx_id, acdb_tx_id);
670 else
671 ALOGE("%s: Incorrect ACDB IDs (rx: %d tx: %d)", __func__,
672 acdb_rx_id, acdb_tx_id);
673 }
674
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800675 if (my_data->csd != NULL) {
676 if (acdb_rx_id > 0 || acdb_tx_id > 0) {
677 ret = my_data->csd->enable_device(acdb_rx_id, acdb_tx_id,
678 my_data->adev->acdb_settings);
679 if (ret < 0) {
680 ALOGE("%s: csd_enable_device, failed, error %d",
681 __func__, ret);
682 }
683 } else {
684 ALOGE("%s: Incorrect ACDB IDs (rx: %d tx: %d)", __func__,
685 acdb_rx_id, acdb_tx_id);
686 }
687 }
688 return ret;
Eric Laurentb23d5282013-05-14 15:27:20 -0700689}
690
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800691int platform_start_voice_call(void *platform, uint32_t vsid)
Eric Laurentb23d5282013-05-14 15:27:20 -0700692{
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800693 struct platform_data *my_data = (struct platform_data *)platform;
694 int ret = 0;
695
696 if (my_data->csd != NULL) {
697 ret = my_data->csd->start_voice(vsid);
698 if (ret < 0) {
699 ALOGE("%s: csd_start_voice error %d\n", __func__, ret);
700 }
701 }
702 return ret;
Eric Laurentb23d5282013-05-14 15:27:20 -0700703}
704
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800705int platform_stop_voice_call(void *platform, uint32_t vsid)
Eric Laurentb23d5282013-05-14 15:27:20 -0700706{
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800707 struct platform_data *my_data = (struct platform_data *)platform;
708 int ret = 0;
709
710 if (my_data->csd != NULL) {
711 ret = my_data->csd->stop_voice(vsid);
712 if (ret < 0) {
713 ALOGE("%s: csd_stop_voice error %d\n", __func__, ret);
714 }
715 }
716 return ret;
Eric Laurentb23d5282013-05-14 15:27:20 -0700717}
718
719int platform_set_voice_volume(void *platform, int volume)
720{
721 struct platform_data *my_data = (struct platform_data *)platform;
722 struct audio_device *adev = my_data->adev;
723 struct mixer_ctl *ctl;
sangwoo53b2cf02013-07-25 19:18:44 -0700724 const char *mixer_ctl_name = "Voice Rx Gain";
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800725 int vol_index = 0, ret = 0;
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700726 uint32_t set_values[ ] = {0,
727 ALL_SESSION_VSID,
728 DEFAULT_VOLUME_RAMP_DURATION_MS};
Eric Laurentb23d5282013-05-14 15:27:20 -0700729
730 // Voice volume levels are mapped to adsp volume levels as follows.
731 // 100 -> 5, 80 -> 4, 60 -> 3, 40 -> 2, 20 -> 1 0 -> 0
732 // But this values don't changed in kernel. So, below change is need.
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700733 vol_index = (int)percent_to_index(volume, MIN_VOL_INDEX, MAX_VOL_INDEX);
734 set_values[0] = vol_index;
Eric Laurentb23d5282013-05-14 15:27:20 -0700735
736 ctl = mixer_get_ctl_by_name(adev->mixer, mixer_ctl_name);
737 if (!ctl) {
738 ALOGE("%s: Could not get ctl for mixer cmd - %s",
739 __func__, mixer_ctl_name);
740 return -EINVAL;
741 }
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700742 ALOGV("Setting voice volume index: %d", set_values[0]);
743 mixer_ctl_set_array(ctl, set_values, ARRAY_SIZE(set_values));
Eric Laurentb23d5282013-05-14 15:27:20 -0700744
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800745 if (my_data->csd != NULL) {
746 ret = my_data->csd->volume(ALL_SESSION_VSID, volume);
747 if (ret < 0) {
748 ALOGE("%s: csd_volume error %d", __func__, ret);
749 }
750 }
751 return ret;
Eric Laurentb23d5282013-05-14 15:27:20 -0700752}
753
754int platform_set_mic_mute(void *platform, bool state)
755{
756 struct platform_data *my_data = (struct platform_data *)platform;
757 struct audio_device *adev = my_data->adev;
758 struct mixer_ctl *ctl;
759 const char *mixer_ctl_name = "Voice Tx Mute";
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800760 int ret = 0;
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700761 uint32_t set_values[ ] = {0,
762 ALL_SESSION_VSID,
763 DEFAULT_VOLUME_RAMP_DURATION_MS};
Eric Laurentb23d5282013-05-14 15:27:20 -0700764
Narsinga Rao Chella05573b72013-11-15 15:21:40 -0800765 set_values[0] = state;
766 ctl = mixer_get_ctl_by_name(adev->mixer, mixer_ctl_name);
767 if (!ctl) {
768 ALOGE("%s: Could not get ctl for mixer cmd - %s",
769 __func__, mixer_ctl_name);
770 return -EINVAL;
771 }
772 ALOGV("Setting voice mute state: %d", state);
773 mixer_ctl_set_array(ctl, set_values, ARRAY_SIZE(set_values));
Eric Laurentb23d5282013-05-14 15:27:20 -0700774
Narsinga Rao Chella05573b72013-11-15 15:21:40 -0800775 if (my_data->csd != NULL) {
776 ret = my_data->csd->mic_mute(ALL_SESSION_VSID, state);
777 if (ret < 0) {
778 ALOGE("%s: csd_mic_mute error %d", __func__, ret);
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800779 }
780 }
781 return ret;
Eric Laurentb23d5282013-05-14 15:27:20 -0700782}
783
784snd_device_t platform_get_output_snd_device(void *platform, audio_devices_t devices)
785{
786 struct platform_data *my_data = (struct platform_data *)platform;
787 struct audio_device *adev = my_data->adev;
788 audio_mode_t mode = adev->mode;
789 snd_device_t snd_device = SND_DEVICE_NONE;
790
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700791 audio_channel_mask_t channel_mask = (adev->active_input == NULL) ?
792 AUDIO_CHANNEL_IN_MONO : adev->active_input->channel_mask;
793 int channel_count = popcount(channel_mask);
794
Eric Laurentb23d5282013-05-14 15:27:20 -0700795 ALOGV("%s: enter: output devices(%#x)", __func__, devices);
796 if (devices == AUDIO_DEVICE_NONE ||
797 devices & AUDIO_DEVICE_BIT_IN) {
798 ALOGV("%s: Invalid output devices (%#x)", __func__, devices);
799 goto exit;
800 }
801
Narsinga Rao Chella05573b72013-11-15 15:21:40 -0800802 if ((mode == AUDIO_MODE_IN_CALL) ||
803 voice_extn_compress_voip_is_active(adev)) {
Eric Laurentb23d5282013-05-14 15:27:20 -0700804 if (devices & AUDIO_DEVICE_OUT_WIRED_HEADPHONE ||
805 devices & AUDIO_DEVICE_OUT_WIRED_HEADSET) {
Narsinga Rao Chella05573b72013-11-15 15:21:40 -0800806 if ((adev->voice.tty_mode != TTY_MODE_OFF) &&
807 !voice_extn_compress_voip_is_active(adev)) {
808 switch (adev->voice.tty_mode) {
809 case TTY_MODE_FULL:
810 snd_device = SND_DEVICE_OUT_VOICE_TTY_FULL_HEADPHONES;
811 break;
812 case TTY_MODE_VCO:
813 snd_device = SND_DEVICE_OUT_VOICE_TTY_VCO_HEADPHONES;
814 break;
815 case TTY_MODE_HCO:
816 snd_device = SND_DEVICE_OUT_VOICE_TTY_HCO_HANDSET;
817 break;
818 default:
819 ALOGE("%s: Invalid TTY mode (%#x)",
820 __func__, adev->voice.tty_mode);
821 }
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700822 } else if (audio_extn_get_anc_enabled()) {
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700823 if (audio_extn_should_use_fb_anc())
824 snd_device = SND_DEVICE_OUT_VOICE_ANC_FB_HEADSET;
825 else
826 snd_device = SND_DEVICE_OUT_VOICE_ANC_HEADSET;
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700827 } else {
Eric Laurentb23d5282013-05-14 15:27:20 -0700828 snd_device = SND_DEVICE_OUT_VOICE_HEADPHONES;
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700829 }
Eric Laurentb23d5282013-05-14 15:27:20 -0700830 } else if (devices & AUDIO_DEVICE_OUT_ALL_SCO) {
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700831 if (my_data->btsco_sample_rate == SAMPLE_RATE_16KHZ)
832 snd_device = SND_DEVICE_OUT_BT_SCO_WB;
833 else
834 snd_device = SND_DEVICE_OUT_BT_SCO;
Eric Laurentb23d5282013-05-14 15:27:20 -0700835 } else if (devices & AUDIO_DEVICE_OUT_SPEAKER) {
836 snd_device = SND_DEVICE_OUT_VOICE_SPEAKER;
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700837 } else if (devices & AUDIO_DEVICE_OUT_ANLG_DOCK_HEADSET ||
838 devices & AUDIO_DEVICE_OUT_DGTL_DOCK_HEADSET) {
839 snd_device = SND_DEVICE_OUT_USB_HEADSET;
Apoorv Raghuvanshi6e262842013-10-06 14:39:35 -0700840 } else if (devices & AUDIO_DEVICE_OUT_FM_TX) {
841 snd_device = SND_DEVICE_OUT_TRANSMISSION_FM;
Eric Laurentb23d5282013-05-14 15:27:20 -0700842 } else if (devices & AUDIO_DEVICE_OUT_EARPIECE) {
Ravi Kumar Alamandaceb40822013-11-06 11:01:47 -0800843 if (audio_extn_should_use_handset_anc(channel_count))
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700844 snd_device = SND_DEVICE_OUT_ANC_HANDSET;
Eric Laurentb23d5282013-05-14 15:27:20 -0700845 else
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700846 snd_device = SND_DEVICE_OUT_VOICE_HANDSET;
Eric Laurentb23d5282013-05-14 15:27:20 -0700847 }
848 if (snd_device != SND_DEVICE_NONE) {
849 goto exit;
850 }
851 }
852
853 if (popcount(devices) == 2) {
854 if (devices == (AUDIO_DEVICE_OUT_WIRED_HEADPHONE |
855 AUDIO_DEVICE_OUT_SPEAKER)) {
856 snd_device = SND_DEVICE_OUT_SPEAKER_AND_HEADPHONES;
857 } else if (devices == (AUDIO_DEVICE_OUT_WIRED_HEADSET |
858 AUDIO_DEVICE_OUT_SPEAKER)) {
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700859 if (audio_extn_get_anc_enabled())
860 snd_device = SND_DEVICE_OUT_SPEAKER_AND_ANC_HEADSET;
861 else
862 snd_device = SND_DEVICE_OUT_SPEAKER_AND_HEADPHONES;
Eric Laurentb23d5282013-05-14 15:27:20 -0700863 } else if (devices == (AUDIO_DEVICE_OUT_AUX_DIGITAL |
864 AUDIO_DEVICE_OUT_SPEAKER)) {
865 snd_device = SND_DEVICE_OUT_SPEAKER_AND_HDMI;
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700866 } else if (devices == (AUDIO_DEVICE_OUT_ANLG_DOCK_HEADSET |
867 AUDIO_DEVICE_OUT_SPEAKER)) {
868 snd_device = SND_DEVICE_OUT_SPEAKER_AND_USB_HEADSET;
Eric Laurentb23d5282013-05-14 15:27:20 -0700869 } else {
870 ALOGE("%s: Invalid combo device(%#x)", __func__, devices);
871 goto exit;
872 }
873 if (snd_device != SND_DEVICE_NONE) {
874 goto exit;
875 }
876 }
877
878 if (popcount(devices) != 1) {
879 ALOGE("%s: Invalid output devices(%#x)", __func__, devices);
880 goto exit;
881 }
882
883 if (devices & AUDIO_DEVICE_OUT_WIRED_HEADPHONE ||
884 devices & AUDIO_DEVICE_OUT_WIRED_HEADSET) {
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700885 if (devices & AUDIO_DEVICE_OUT_WIRED_HEADSET
886 && audio_extn_get_anc_enabled()) {
887 if (audio_extn_should_use_fb_anc())
888 snd_device = SND_DEVICE_OUT_ANC_FB_HEADSET;
889 else
890 snd_device = SND_DEVICE_OUT_ANC_HEADSET;
891 }
892 else
893 snd_device = SND_DEVICE_OUT_HEADPHONES;
Eric Laurentb23d5282013-05-14 15:27:20 -0700894 } else if (devices & AUDIO_DEVICE_OUT_SPEAKER) {
895 if (adev->speaker_lr_swap)
896 snd_device = SND_DEVICE_OUT_SPEAKER_REVERSE;
897 else
898 snd_device = SND_DEVICE_OUT_SPEAKER;
899 } else if (devices & AUDIO_DEVICE_OUT_ALL_SCO) {
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700900 if (my_data->btsco_sample_rate == SAMPLE_RATE_16KHZ)
901 snd_device = SND_DEVICE_OUT_BT_SCO_WB;
902 else
903 snd_device = SND_DEVICE_OUT_BT_SCO;
Eric Laurentb23d5282013-05-14 15:27:20 -0700904 } else if (devices & AUDIO_DEVICE_OUT_AUX_DIGITAL) {
905 snd_device = SND_DEVICE_OUT_HDMI ;
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700906 } else if (devices & AUDIO_DEVICE_OUT_ANLG_DOCK_HEADSET ||
907 devices & AUDIO_DEVICE_OUT_DGTL_DOCK_HEADSET) {
908 snd_device = SND_DEVICE_OUT_USB_HEADSET;
Apoorv Raghuvanshi6e262842013-10-06 14:39:35 -0700909 } else if (devices & AUDIO_DEVICE_OUT_FM_TX) {
910 snd_device = SND_DEVICE_OUT_TRANSMISSION_FM;
Eric Laurentb23d5282013-05-14 15:27:20 -0700911 } else if (devices & AUDIO_DEVICE_OUT_EARPIECE) {
912 snd_device = SND_DEVICE_OUT_HANDSET;
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700913 } else if (devices & AUDIO_DEVICE_OUT_PROXY) {
914 ALOGD("%s: setting sink capability for Proxy", __func__);
915 audio_extn_set_afe_proxy_channel_mixer(adev);
916 snd_device = SND_DEVICE_OUT_AFE_PROXY;
Eric Laurentb23d5282013-05-14 15:27:20 -0700917 } else {
918 ALOGE("%s: Unknown device(s) %#x", __func__, devices);
919 }
920exit:
921 ALOGV("%s: exit: snd_device(%s)", __func__, device_table[snd_device]);
922 return snd_device;
923}
924
925snd_device_t platform_get_input_snd_device(void *platform, audio_devices_t out_device)
926{
927 struct platform_data *my_data = (struct platform_data *)platform;
928 struct audio_device *adev = my_data->adev;
929 audio_source_t source = (adev->active_input == NULL) ?
930 AUDIO_SOURCE_DEFAULT : adev->active_input->source;
931
932 audio_mode_t mode = adev->mode;
933 audio_devices_t in_device = ((adev->active_input == NULL) ?
934 AUDIO_DEVICE_NONE : adev->active_input->device)
935 & ~AUDIO_DEVICE_BIT_IN;
936 audio_channel_mask_t channel_mask = (adev->active_input == NULL) ?
937 AUDIO_CHANNEL_IN_MONO : adev->active_input->channel_mask;
938 snd_device_t snd_device = SND_DEVICE_NONE;
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700939 int channel_count = popcount(channel_mask);
Eric Laurentb23d5282013-05-14 15:27:20 -0700940
941 ALOGV("%s: enter: out_device(%#x) in_device(%#x)",
942 __func__, out_device, in_device);
Narsinga Rao Chella05573b72013-11-15 15:21:40 -0800943 if ((mode == AUDIO_MODE_IN_CALL) ||
944 voice_extn_compress_voip_is_active(adev)) {
Eric Laurentb23d5282013-05-14 15:27:20 -0700945 if (out_device == AUDIO_DEVICE_NONE) {
946 ALOGE("%s: No output device set for voice call", __func__);
947 goto exit;
948 }
Narsinga Rao Chella05573b72013-11-15 15:21:40 -0800949 if ((adev->voice.tty_mode != TTY_MODE_OFF) &&
950 !voice_extn_compress_voip_is_active(adev)) {
Eric Laurentb23d5282013-05-14 15:27:20 -0700951 if (out_device & AUDIO_DEVICE_OUT_WIRED_HEADPHONE ||
952 out_device & AUDIO_DEVICE_OUT_WIRED_HEADSET) {
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700953 switch (adev->voice.tty_mode) {
Eric Laurentb23d5282013-05-14 15:27:20 -0700954 case TTY_MODE_FULL:
955 snd_device = SND_DEVICE_IN_VOICE_TTY_FULL_HEADSET_MIC;
956 break;
957 case TTY_MODE_VCO:
958 snd_device = SND_DEVICE_IN_VOICE_TTY_VCO_HANDSET_MIC;
959 break;
960 case TTY_MODE_HCO:
961 snd_device = SND_DEVICE_IN_VOICE_TTY_HCO_HEADSET_MIC;
962 break;
963 default:
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700964 ALOGE("%s: Invalid TTY mode (%#x)",
965 __func__, adev->voice.tty_mode);
Eric Laurentb23d5282013-05-14 15:27:20 -0700966 }
967 goto exit;
968 }
969 }
970 if (out_device & AUDIO_DEVICE_OUT_EARPIECE ||
971 out_device & AUDIO_DEVICE_OUT_WIRED_HEADPHONE) {
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700972 if (out_device & AUDIO_DEVICE_OUT_EARPIECE &&
973 audio_extn_should_use_handset_anc(channel_count)) {
974 snd_device = SND_DEVICE_IN_AANC_HANDSET_MIC;
975 } else if (my_data->fluence_type == FLUENCE_NONE ||
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700976 my_data->fluence_in_voice_call == false) {
Eric Laurentb23d5282013-05-14 15:27:20 -0700977 snd_device = SND_DEVICE_IN_HANDSET_MIC;
978 } else {
Ravi Kumar Alamandaceb40822013-11-06 11:01:47 -0800979 snd_device = SND_DEVICE_IN_VOICE_DMIC;
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700980 adev->acdb_settings |= DMIC_FLAG;
Eric Laurentb23d5282013-05-14 15:27:20 -0700981 }
982 } else if (out_device & AUDIO_DEVICE_OUT_WIRED_HEADSET) {
983 snd_device = SND_DEVICE_IN_VOICE_HEADSET_MIC;
984 } else if (out_device & AUDIO_DEVICE_OUT_ALL_SCO) {
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700985 if (my_data->btsco_sample_rate == SAMPLE_RATE_16KHZ)
986 snd_device = SND_DEVICE_IN_BT_SCO_MIC_WB;
987 else
988 snd_device = SND_DEVICE_IN_BT_SCO_MIC;
Eric Laurentb23d5282013-05-14 15:27:20 -0700989 } else if (out_device & AUDIO_DEVICE_OUT_SPEAKER) {
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700990 if (my_data->fluence_type != FLUENCE_NONE &&
991 my_data->fluence_in_voice_call &&
992 my_data->fluence_in_spkr_mode) {
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800993 if(my_data->fluence_type & FLUENCE_QUAD_MIC) {
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700994 adev->acdb_settings |= QMIC_FLAG;
995 snd_device = SND_DEVICE_IN_VOICE_SPEAKER_QMIC;
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800996 } else {
997 adev->acdb_settings |= DMIC_FLAG;
998 snd_device = SND_DEVICE_IN_VOICE_SPEAKER_DMIC;
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700999 }
Eric Laurentb23d5282013-05-14 15:27:20 -07001000 } else {
1001 snd_device = SND_DEVICE_IN_VOICE_SPEAKER_MIC;
1002 }
1003 }
1004 } else if (source == AUDIO_SOURCE_CAMCORDER) {
1005 if (in_device & AUDIO_DEVICE_IN_BUILTIN_MIC ||
1006 in_device & AUDIO_DEVICE_IN_BACK_MIC) {
1007 snd_device = SND_DEVICE_IN_CAMCORDER_MIC;
1008 }
1009 } else if (source == AUDIO_SOURCE_VOICE_RECOGNITION) {
1010 if (in_device & AUDIO_DEVICE_IN_BUILTIN_MIC) {
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -08001011 if (channel_count == 2) {
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -08001012 snd_device = SND_DEVICE_IN_VOICE_REC_DMIC_STEREO;
Mingming Yin8e5a4f62013-10-07 15:23:41 -07001013 adev->acdb_settings |= DMIC_FLAG;
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -08001014 } else if (adev->active_input->enable_ns)
1015 snd_device = SND_DEVICE_IN_VOICE_REC_MIC_NS;
1016 else if (my_data->fluence_type != FLUENCE_NONE &&
1017 my_data->fluence_in_voice_rec) {
1018 snd_device = SND_DEVICE_IN_VOICE_REC_DMIC_FLUENCE;
1019 adev->acdb_settings |= DMIC_FLAG;
1020 } else {
1021 snd_device = SND_DEVICE_IN_VOICE_REC_MIC;
1022 }
Eric Laurentb23d5282013-05-14 15:27:20 -07001023 }
1024 } else if (source == AUDIO_SOURCE_VOICE_COMMUNICATION) {
1025 if (out_device & AUDIO_DEVICE_OUT_SPEAKER)
1026 in_device = AUDIO_DEVICE_IN_BACK_MIC;
1027 if (adev->active_input) {
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -08001028 if (adev->active_input->enable_aec &&
1029 adev->active_input->enable_ns) {
Eric Laurentb23d5282013-05-14 15:27:20 -07001030 if (in_device & AUDIO_DEVICE_IN_BACK_MIC) {
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -08001031 if (my_data->fluence_type & FLUENCE_DUAL_MIC &&
1032 my_data->fluence_in_spkr_mode) {
1033 snd_device = SND_DEVICE_IN_SPEAKER_DMIC_AEC_NS;
1034 adev->acdb_settings |= DMIC_FLAG;
1035 } else
1036 snd_device = SND_DEVICE_IN_SPEAKER_MIC_AEC_NS;
Eric Laurentb23d5282013-05-14 15:27:20 -07001037 } else if (in_device & AUDIO_DEVICE_IN_BUILTIN_MIC) {
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -08001038 if (my_data->fluence_type & FLUENCE_DUAL_MIC) {
1039 snd_device = SND_DEVICE_IN_HANDSET_DMIC_AEC_NS;
1040 adev->acdb_settings |= DMIC_FLAG;
1041 } else
1042 snd_device = SND_DEVICE_IN_HANDSET_MIC_AEC_NS;
Eric Laurentb23d5282013-05-14 15:27:20 -07001043 } else if (in_device & AUDIO_DEVICE_IN_WIRED_HEADSET) {
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -08001044 snd_device = SND_DEVICE_IN_HEADSET_MIC_FLUENCE;
Eric Laurentb23d5282013-05-14 15:27:20 -07001045 }
1046 set_echo_reference(adev->mixer, "SLIM_RX");
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -08001047 } else if (adev->active_input->enable_aec) {
1048 if (in_device & AUDIO_DEVICE_IN_BACK_MIC) {
1049 if (my_data->fluence_type & FLUENCE_DUAL_MIC) {
1050 snd_device = SND_DEVICE_IN_SPEAKER_DMIC_AEC;
1051 adev->acdb_settings |= DMIC_FLAG;
1052 } else
1053 snd_device = SND_DEVICE_IN_SPEAKER_MIC_AEC;
1054 } else if (in_device & AUDIO_DEVICE_IN_BUILTIN_MIC) {
1055 if (my_data->fluence_type & FLUENCE_DUAL_MIC) {
1056 snd_device = SND_DEVICE_IN_HANDSET_DMIC_AEC;
1057 adev->acdb_settings |= DMIC_FLAG;
1058 } else
1059 snd_device = SND_DEVICE_IN_HANDSET_MIC_AEC;
1060 } else if (in_device & AUDIO_DEVICE_IN_WIRED_HEADSET) {
1061 snd_device = SND_DEVICE_IN_HEADSET_MIC_FLUENCE;
1062 }
1063 set_echo_reference(adev->mixer, "SLIM_RX");
1064 } else if (adev->active_input->enable_ns) {
1065 if (in_device & AUDIO_DEVICE_IN_BACK_MIC) {
1066 if (my_data->fluence_type & FLUENCE_DUAL_MIC) {
1067 snd_device = SND_DEVICE_IN_SPEAKER_DMIC_NS;
1068 adev->acdb_settings |= DMIC_FLAG;
1069 } else
1070 snd_device = SND_DEVICE_IN_SPEAKER_MIC_NS;
1071 } else if (in_device & AUDIO_DEVICE_IN_BUILTIN_MIC) {
1072 if (my_data->fluence_type & FLUENCE_DUAL_MIC) {
1073 snd_device = SND_DEVICE_IN_HANDSET_DMIC_NS;
1074 adev->acdb_settings |= DMIC_FLAG;
1075 } else
1076 snd_device = SND_DEVICE_IN_HANDSET_MIC_NS;
1077 } else if (in_device & AUDIO_DEVICE_IN_WIRED_HEADSET) {
1078 snd_device = SND_DEVICE_IN_HEADSET_MIC_FLUENCE;
1079 }
1080 set_echo_reference(adev->mixer, "NONE");
Eric Laurentb23d5282013-05-14 15:27:20 -07001081 } else
1082 set_echo_reference(adev->mixer, "NONE");
1083 }
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -08001084 } else if (source == AUDIO_SOURCE_MIC) {
Apoorv Raghuvanshic0536542013-11-14 16:25:59 -08001085 if (in_device & AUDIO_DEVICE_IN_BUILTIN_MIC &&
1086 channel_count == 1 ) {
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -08001087 if(my_data->fluence_type & FLUENCE_DUAL_MIC &&
1088 my_data->fluence_in_audio_rec)
1089 snd_device = SND_DEVICE_IN_HANDSET_DMIC;
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -08001090 }
Mingming Yinab429782013-11-07 11:16:55 -08001091 } else if (source == AUDIO_SOURCE_FM_RX ||
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -08001092 source == AUDIO_SOURCE_FM_RX_A2DP) {
Preetam Singh Ranawatde84f1a2013-11-01 14:58:16 -07001093 snd_device = SND_DEVICE_IN_CAPTURE_FM;
Eric Laurentb23d5282013-05-14 15:27:20 -07001094 } else if (source == AUDIO_SOURCE_DEFAULT) {
1095 goto exit;
1096 }
1097
1098
1099 if (snd_device != SND_DEVICE_NONE) {
1100 goto exit;
1101 }
1102
1103 if (in_device != AUDIO_DEVICE_NONE &&
1104 !(in_device & AUDIO_DEVICE_IN_VOICE_CALL) &&
1105 !(in_device & AUDIO_DEVICE_IN_COMMUNICATION)) {
1106 if (in_device & AUDIO_DEVICE_IN_BUILTIN_MIC) {
Apoorv Raghuvanshi6178a3f2013-10-19 12:38:54 -07001107 if (audio_extn_ssr_get_enabled() && channel_count == 6)
1108 snd_device = SND_DEVICE_IN_QUAD_MIC;
Ravi Kumar Alamandafae42112013-11-07 23:31:54 -08001109 else if (channel_count == 2)
Apoorv Raghuvanshi6bd8dbf2013-10-19 18:37:52 -07001110 snd_device = SND_DEVICE_IN_HANDSET_STEREO_DMIC;
Apoorv Raghuvanshi6178a3f2013-10-19 12:38:54 -07001111 else
1112 snd_device = SND_DEVICE_IN_HANDSET_MIC;
Eric Laurentb23d5282013-05-14 15:27:20 -07001113 } else if (in_device & AUDIO_DEVICE_IN_BACK_MIC) {
1114 snd_device = SND_DEVICE_IN_SPEAKER_MIC;
1115 } else if (in_device & AUDIO_DEVICE_IN_WIRED_HEADSET) {
1116 snd_device = SND_DEVICE_IN_HEADSET_MIC;
1117 } else if (in_device & AUDIO_DEVICE_IN_BLUETOOTH_SCO_HEADSET) {
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001118 if (my_data->btsco_sample_rate == SAMPLE_RATE_16KHZ)
1119 snd_device = SND_DEVICE_IN_BT_SCO_MIC_WB;
1120 else
1121 snd_device = SND_DEVICE_IN_BT_SCO_MIC;
Eric Laurentb23d5282013-05-14 15:27:20 -07001122 } else if (in_device & AUDIO_DEVICE_IN_AUX_DIGITAL) {
1123 snd_device = SND_DEVICE_IN_HDMI_MIC;
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -07001124 } else if (in_device & AUDIO_DEVICE_IN_ANLG_DOCK_HEADSET ||
1125 in_device & AUDIO_DEVICE_IN_DGTL_DOCK_HEADSET) {
1126 snd_device = SND_DEVICE_IN_USB_HEADSET_MIC;
Apoorv Raghuvanshi6e262842013-10-06 14:39:35 -07001127 } else if (in_device & AUDIO_DEVICE_IN_FM_RX) {
1128 snd_device = SND_DEVICE_IN_CAPTURE_FM;
Eric Laurentb23d5282013-05-14 15:27:20 -07001129 } else {
1130 ALOGE("%s: Unknown input device(s) %#x", __func__, in_device);
1131 ALOGW("%s: Using default handset-mic", __func__);
1132 snd_device = SND_DEVICE_IN_HANDSET_MIC;
1133 }
1134 } else {
1135 if (out_device & AUDIO_DEVICE_OUT_EARPIECE) {
1136 snd_device = SND_DEVICE_IN_HANDSET_MIC;
1137 } else if (out_device & AUDIO_DEVICE_OUT_WIRED_HEADSET) {
1138 snd_device = SND_DEVICE_IN_HEADSET_MIC;
1139 } else if (out_device & AUDIO_DEVICE_OUT_SPEAKER) {
Apoorv Raghuvanshi6bd8dbf2013-10-19 18:37:52 -07001140 if (channel_count > 1)
1141 snd_device = SND_DEVICE_IN_SPEAKER_STEREO_DMIC;
1142 else
1143 snd_device = SND_DEVICE_IN_SPEAKER_MIC;
Eric Laurentb23d5282013-05-14 15:27:20 -07001144 } else if (out_device & AUDIO_DEVICE_OUT_WIRED_HEADPHONE) {
1145 snd_device = SND_DEVICE_IN_HANDSET_MIC;
1146 } else if (out_device & AUDIO_DEVICE_OUT_BLUETOOTH_SCO_HEADSET) {
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001147 if (my_data->btsco_sample_rate == SAMPLE_RATE_16KHZ)
1148 snd_device = SND_DEVICE_IN_BT_SCO_MIC_WB;
1149 else
1150 snd_device = SND_DEVICE_IN_BT_SCO_MIC;
Eric Laurentb23d5282013-05-14 15:27:20 -07001151 } else if (out_device & AUDIO_DEVICE_OUT_AUX_DIGITAL) {
1152 snd_device = SND_DEVICE_IN_HDMI_MIC;
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -07001153 } else if (out_device & AUDIO_DEVICE_OUT_ANLG_DOCK_HEADSET ||
1154 out_device & AUDIO_DEVICE_OUT_DGTL_DOCK_HEADSET) {
1155 snd_device = SND_DEVICE_IN_USB_HEADSET_MIC;
Eric Laurentb23d5282013-05-14 15:27:20 -07001156 } else {
1157 ALOGE("%s: Unknown output device(s) %#x", __func__, out_device);
1158 ALOGW("%s: Using default handset-mic", __func__);
1159 snd_device = SND_DEVICE_IN_HANDSET_MIC;
1160 }
1161 }
1162exit:
1163 ALOGV("%s: exit: in_snd_device(%s)", __func__, device_table[snd_device]);
1164 return snd_device;
1165}
1166
1167int platform_set_hdmi_channels(void *platform, int channel_count)
1168{
1169 struct platform_data *my_data = (struct platform_data *)platform;
1170 struct audio_device *adev = my_data->adev;
1171 struct mixer_ctl *ctl;
1172 const char *channel_cnt_str = NULL;
1173 const char *mixer_ctl_name = "HDMI_RX Channels";
1174 switch (channel_count) {
1175 case 8:
1176 channel_cnt_str = "Eight"; break;
1177 case 7:
1178 channel_cnt_str = "Seven"; break;
1179 case 6:
1180 channel_cnt_str = "Six"; break;
1181 case 5:
1182 channel_cnt_str = "Five"; break;
1183 case 4:
1184 channel_cnt_str = "Four"; break;
1185 case 3:
1186 channel_cnt_str = "Three"; break;
1187 default:
1188 channel_cnt_str = "Two"; break;
1189 }
1190 ctl = mixer_get_ctl_by_name(adev->mixer, mixer_ctl_name);
1191 if (!ctl) {
1192 ALOGE("%s: Could not get ctl for mixer cmd - %s",
1193 __func__, mixer_ctl_name);
1194 return -EINVAL;
1195 }
1196 ALOGV("HDMI channel count: %s", channel_cnt_str);
1197 mixer_ctl_set_enum_by_string(ctl, channel_cnt_str);
1198 return 0;
1199}
1200
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001201int platform_edid_get_max_channels(void *platform)
Eric Laurentb23d5282013-05-14 15:27:20 -07001202{
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001203 struct platform_data *my_data = (struct platform_data *)platform;
1204 struct audio_device *adev = my_data->adev;
Eric Laurentb23d5282013-05-14 15:27:20 -07001205 char block[MAX_SAD_BLOCKS * SAD_BLOCK_SIZE];
1206 char *sad = block;
1207 int num_audio_blocks;
1208 int channel_count;
1209 int max_channels = 0;
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001210 int i, ret, count;
Eric Laurentb23d5282013-05-14 15:27:20 -07001211
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001212 struct mixer_ctl *ctl;
1213
1214 ctl = mixer_get_ctl_by_name(adev->mixer, AUDIO_DATA_BLOCK_MIXER_CTL);
1215 if (!ctl) {
1216 ALOGE("%s: Could not get ctl for mixer cmd - %s",
1217 __func__, AUDIO_DATA_BLOCK_MIXER_CTL);
Eric Laurentb23d5282013-05-14 15:27:20 -07001218 return 0;
1219 }
1220
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001221 mixer_ctl_update(ctl);
1222
1223 count = mixer_ctl_get_num_values(ctl);
Eric Laurentb23d5282013-05-14 15:27:20 -07001224
1225 /* Read SAD blocks, clamping the maximum size for safety */
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001226 if (count > (int)sizeof(block))
1227 count = (int)sizeof(block);
Eric Laurentb23d5282013-05-14 15:27:20 -07001228
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001229 ret = mixer_ctl_get_array(ctl, block, count);
1230 if (ret != 0) {
1231 ALOGE("%s: mixer_ctl_get_array() failed to get EDID info", __func__);
1232 return 0;
1233 }
Eric Laurentb23d5282013-05-14 15:27:20 -07001234
1235 /* Calculate the number of SAD blocks */
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001236 num_audio_blocks = count / SAD_BLOCK_SIZE;
Eric Laurentb23d5282013-05-14 15:27:20 -07001237
1238 for (i = 0; i < num_audio_blocks; i++) {
1239 /* Only consider LPCM blocks */
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001240 if ((sad[0] >> 3) != EDID_FORMAT_LPCM) {
1241 sad += 3;
Eric Laurentb23d5282013-05-14 15:27:20 -07001242 continue;
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001243 }
Eric Laurentb23d5282013-05-14 15:27:20 -07001244
1245 channel_count = (sad[0] & 0x7) + 1;
1246 if (channel_count > max_channels)
1247 max_channels = channel_count;
1248
1249 /* Advance to next block */
1250 sad += 3;
1251 }
1252
1253 return max_channels;
1254}
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001255
1256static int platform_set_slowtalk(struct platform_data *my_data, bool state)
1257{
1258 int ret = 0;
1259 struct audio_device *adev = my_data->adev;
1260 struct mixer_ctl *ctl;
1261 const char *mixer_ctl_name = "Slowtalk Enable";
1262 uint32_t set_values[ ] = {0,
1263 ALL_SESSION_VSID};
1264
1265 set_values[0] = state;
1266 ctl = mixer_get_ctl_by_name(adev->mixer, mixer_ctl_name);
1267 if (!ctl) {
1268 ALOGE("%s: Could not get ctl for mixer cmd - %s",
1269 __func__, mixer_ctl_name);
1270 ret = -EINVAL;
1271 } else {
1272 ALOGV("Setting slowtalk state: %d", state);
1273 ret = mixer_ctl_set_array(ctl, set_values, ARRAY_SIZE(set_values));
1274 my_data->slowtalk = state;
1275 }
1276
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -08001277 if (my_data->csd != NULL) {
1278 ret = my_data->csd->slow_talk(ALL_SESSION_VSID, state);
1279 if (ret < 0) {
1280 ALOGE("%s: csd_client_disable_device, failed, error %d",
1281 __func__, ret);
1282 }
1283 }
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001284 return ret;
1285}
1286
1287int platform_set_parameters(void *platform, struct str_parms *parms)
1288{
1289 struct platform_data *my_data = (struct platform_data *)platform;
1290 char *str;
1291 char value[32];
1292 int val;
1293 int ret = 0;
1294
1295 ALOGV("%s: enter: %s", __func__, str_parms_to_str(parms));
1296
1297 ret = str_parms_get_int(parms, AUDIO_PARAMETER_KEY_BTSCO, &val);
1298 if (ret >= 0) {
1299 str_parms_del(parms, AUDIO_PARAMETER_KEY_BTSCO);
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001300 my_data->btsco_sample_rate = val;
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001301 }
1302
1303 ret = str_parms_get_int(parms, AUDIO_PARAMETER_KEY_SLOWTALK, &val);
1304 if (ret >= 0) {
1305 str_parms_del(parms, AUDIO_PARAMETER_KEY_SLOWTALK);
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001306 ret = platform_set_slowtalk(my_data, val);
1307 if (ret)
1308 ALOGE("%s: Failed to set slow talk err: %d", __func__, ret);
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001309 }
1310
1311 ALOGV("%s: exit with code(%d)", __func__, ret);
1312 return ret;
1313}
1314
Shiv Maliyappanahallida107642013-10-17 11:16:13 -07001315int platform_set_incall_recoding_session_id(void *platform,
1316 uint32_t session_id)
1317{
1318 int ret = 0;
1319 struct platform_data *my_data = (struct platform_data *)platform;
1320 struct audio_device *adev = my_data->adev;
1321 struct mixer_ctl *ctl;
1322 const char *mixer_ctl_name = "Voc VSID";
1323 int num_ctl_values;
1324 int i;
1325
1326 ctl = mixer_get_ctl_by_name(adev->mixer, mixer_ctl_name);
1327 if (!ctl) {
1328 ALOGE("%s: Could not get ctl for mixer cmd - %s",
1329 __func__, mixer_ctl_name);
1330 ret = -EINVAL;
1331 } else {
1332 num_ctl_values = mixer_ctl_get_num_values(ctl);
1333 for (i = 0; i < num_ctl_values; i++) {
1334 if (mixer_ctl_set_value(ctl, i, session_id)) {
1335 ALOGV("Error: invalid session_id: %x", session_id);
1336 ret = -EINVAL;
1337 break;
1338 }
1339 }
1340 }
1341
1342 return ret;
1343}
1344
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001345void platform_get_parameters(void *platform,
1346 struct str_parms *query,
1347 struct str_parms *reply)
1348{
1349 struct platform_data *my_data = (struct platform_data *)platform;
1350 char *str = NULL;
1351 char value[256] = {0};
1352 int ret;
1353 int fluence_type;
1354
1355 ret = str_parms_get_str(query, AUDIO_PARAMETER_KEY_FLUENCE_TYPE,
1356 value, sizeof(value));
1357 if (ret >= 0) {
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -08001358 if (my_data->fluence_type & FLUENCE_QUAD_MIC) {
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001359 strlcpy(value, "fluencepro", sizeof(value));
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -08001360 } else if (my_data->fluence_type & FLUENCE_DUAL_MIC) {
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001361 strlcpy(value, "fluence", sizeof(value));
1362 } else {
1363 strlcpy(value, "none", sizeof(value));
1364 }
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001365
1366 str_parms_add_str(reply, AUDIO_PARAMETER_KEY_FLUENCE_TYPE, value);
1367 }
1368
1369 ALOGV("%s: exit: returns - %s", __func__, str_parms_to_str(reply));
1370}
1371
Haynes Mathew George7ff216f2013-09-11 19:51:41 -07001372/* Delay in Us */
1373int64_t platform_render_latency(audio_usecase_t usecase)
1374{
1375 switch (usecase) {
1376 case USECASE_AUDIO_PLAYBACK_DEEP_BUFFER:
1377 return DEEP_BUFFER_PLATFORM_DELAY;
1378 case USECASE_AUDIO_PLAYBACK_LOW_LATENCY:
1379 return LOW_LATENCY_PLATFORM_DELAY;
1380 default:
1381 return 0;
1382 }
1383}
Preetam Singh Ranawatde84f1a2013-11-01 14:58:16 -07001384
Mingming Yine62d7842013-10-25 16:26:03 -07001385int platform_update_usecase_from_source(int source, int usecase)
Preetam Singh Ranawatde84f1a2013-11-01 14:58:16 -07001386{
Mingming Yinab429782013-11-07 11:16:55 -08001387 ALOGV("%s: input source :%d", __func__, source);
1388 if(source == AUDIO_SOURCE_FM_RX_A2DP)
Mingming Yine62d7842013-10-25 16:26:03 -07001389 usecase = USECASE_AUDIO_RECORD_FM_VIRTUAL;
1390 return usecase;
Preetam Singh Ranawatde84f1a2013-11-01 14:58:16 -07001391}