blob: b8e8bb5f4c35c3ad3b3e527876d4754d1c5773e4 [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"
Eric Laurentb23d5282013-05-14 15:27:20 -070033
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -070034#define MIXER_XML_PATH "/system/etc/mixer_paths.xml"
Damir Didjustof1d46c72013-11-06 17:59:04 -080035#define MIXER_XML_PATH_AUXPCM "/system/etc/mixer_paths_auxpcm.xml"
Eric Laurentb23d5282013-05-14 15:27:20 -070036#define LIB_ACDB_LOADER "libacdbloader.so"
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -070037#define AUDIO_DATA_BLOCK_MIXER_CTL "HDMI EDID"
Eric Laurentb23d5282013-05-14 15:27:20 -070038
Eric Laurentb23d5282013-05-14 15:27:20 -070039/*
Eric Laurentb23d5282013-05-14 15:27:20 -070040 * This file will have a maximum of 38 bytes:
41 *
42 * 4 bytes: number of audio blocks
43 * 4 bytes: total length of Short Audio Descriptor (SAD) blocks
44 * Maximum 10 * 3 bytes: SAD blocks
45 */
46#define MAX_SAD_BLOCKS 10
47#define SAD_BLOCK_SIZE 3
48
49/* EDID format ID for LPCM audio */
50#define EDID_FORMAT_LPCM 1
51
sangwoo1b9f4b32013-06-21 18:22:55 -070052/* Retry for delay in FW loading*/
53#define RETRY_NUMBER 10
54#define RETRY_US 500000
55
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -070056#define SAMPLE_RATE_8KHZ 8000
57#define SAMPLE_RATE_16KHZ 16000
58
sangwoo53b2cf02013-07-25 19:18:44 -070059#define MAX_VOL_INDEX 5
60#define MIN_VOL_INDEX 0
61#define percent_to_index(val, min, max) \
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -070062 ((val) * ((max) - (min)) * 0.01 + (min) + .5)
63
64#define AUDIO_PARAMETER_KEY_FLUENCE_TYPE "fluence"
65#define AUDIO_PARAMETER_KEY_BTSCO "bt_samplerate"
66#define AUDIO_PARAMETER_KEY_SLOWTALK "st_enable"
sangwoo53b2cf02013-07-25 19:18:44 -070067
Eric Laurentb23d5282013-05-14 15:27:20 -070068struct audio_block_header
69{
70 int reserved;
71 int length;
72};
73
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -080074/* Audio calibration related functions */
Eric Laurentb23d5282013-05-14 15:27:20 -070075typedef void (*acdb_deallocate_t)();
76typedef int (*acdb_init_t)();
77typedef void (*acdb_send_audio_cal_t)(int, int);
78typedef void (*acdb_send_voice_cal_t)(int, int);
79
Eric Laurentb23d5282013-05-14 15:27:20 -070080struct platform_data {
81 struct audio_device *adev;
82 bool fluence_in_spkr_mode;
83 bool fluence_in_voice_call;
84 bool fluence_in_voice_rec;
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -080085 bool fluence_in_audio_rec;
Mingming Yin8e5a4f62013-10-07 15:23:41 -070086 int fluence_type;
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -070087 int btsco_sample_rate;
88 bool slowtalk;
Ravi Kumar Alamanda48c921d2013-10-29 06:07:44 -070089 /* Audio calibration related functions */
Eric Laurentb23d5282013-05-14 15:27:20 -070090 void *acdb_handle;
91 acdb_init_t acdb_init;
92 acdb_deallocate_t acdb_deallocate;
93 acdb_send_audio_cal_t acdb_send_audio_cal;
94 acdb_send_voice_cal_t acdb_send_voice_cal;
Ravi Kumar Alamanda48c921d2013-10-29 06:07:44 -070095
96 void *hw_info;
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -080097 struct csd_data *csd;
Eric Laurentb23d5282013-05-14 15:27:20 -070098};
99
100static const int pcm_device_table[AUDIO_USECASE_MAX][2] = {
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700101 [USECASE_AUDIO_PLAYBACK_DEEP_BUFFER] = {DEEP_BUFFER_PCM_DEVICE,
102 DEEP_BUFFER_PCM_DEVICE},
103 [USECASE_AUDIO_PLAYBACK_LOW_LATENCY] = {LOWLATENCY_PCM_DEVICE,
Preetam Singh Ranawatde84f1a2013-11-01 14:58:16 -0700104 LOWLATENCY_PCM_DEVICE},
105 [USECASE_AUDIO_PLAYBACK_MULTI_CH] = {MULTIMEDIA2_PCM_DEVICE,
106 MULTIMEDIA2_PCM_DEVICE},
Krishnankutty Kolathappillya43f96e2013-11-01 12:17:53 -0700107 [USECASE_AUDIO_PLAYBACK_OFFLOAD] =
108 {PLAYBACK_OFFLOAD_DEVICE, PLAYBACK_OFFLOAD_DEVICE},
Shiv Maliyappanahallida107642013-10-17 11:16:13 -0700109 [USECASE_AUDIO_RECORD] = {AUDIO_RECORD_PCM_DEVICE, AUDIO_RECORD_PCM_DEVICE},
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700110 [USECASE_AUDIO_RECORD_LOW_LATENCY] = {LOWLATENCY_PCM_DEVICE,
111 LOWLATENCY_PCM_DEVICE},
Preetam Singh Ranawatde84f1a2013-11-01 14:58:16 -0700112 [USECASE_AUDIO_RECORD_FM_VIRTUAL] = {MULTIMEDIA2_PCM_DEVICE,
113 MULTIMEDIA2_PCM_DEVICE},
Apoorv Raghuvanshi6e262842013-10-06 14:39:35 -0700114 [USECASE_AUDIO_PLAYBACK_FM] = {FM_PLAYBACK_PCM_DEVICE, FM_CAPTURE_PCM_DEVICE},
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700115 [USECASE_VOICE_CALL] = {VOICE_CALL_PCM_DEVICE, VOICE_CALL_PCM_DEVICE},
116 [USECASE_VOICE2_CALL] = {VOICE2_CALL_PCM_DEVICE, VOICE2_CALL_PCM_DEVICE},
117 [USECASE_VOLTE_CALL] = {VOLTE_CALL_PCM_DEVICE, VOLTE_CALL_PCM_DEVICE},
118 [USECASE_QCHAT_CALL] = {QCHAT_CALL_PCM_DEVICE, QCHAT_CALL_PCM_DEVICE},
Shiv Maliyappanahallida107642013-10-17 11:16:13 -0700119 [USECASE_INCALL_REC_UPLINK] = {AUDIO_RECORD_PCM_DEVICE,
120 AUDIO_RECORD_PCM_DEVICE},
121 [USECASE_INCALL_REC_DOWNLINK] = {AUDIO_RECORD_PCM_DEVICE,
122 AUDIO_RECORD_PCM_DEVICE},
123 [USECASE_INCALL_REC_UPLINK_AND_DOWNLINK] = {AUDIO_RECORD_PCM_DEVICE,
124 AUDIO_RECORD_PCM_DEVICE},
Shiv Maliyappanahallif3b9a422013-10-22 16:38:08 -0700125 [USECASE_INCALL_MUSIC_UPLINK] = {INCALL_MUSIC_UPLINK_PCM_DEVICE,
126 INCALL_MUSIC_UPLINK_PCM_DEVICE},
127 [USECASE_INCALL_MUSIC_UPLINK2] = {INCALL_MUSIC_UPLINK2_PCM_DEVICE,
128 INCALL_MUSIC_UPLINK2_PCM_DEVICE},
Gopikrishnaiah Anandanf538cef2013-10-28 14:06:03 -0700129 [USECASE_AUDIO_SPKR_CALIB_RX] = {SPKR_PROT_CALIB_RX_PCM_DEVICE, -1},
130 [USECASE_AUDIO_SPKR_CALIB_TX] = {-1, SPKR_PROT_CALIB_TX_PCM_DEVICE},
Eric Laurentb23d5282013-05-14 15:27:20 -0700131};
132
133/* Array to store sound devices */
134static const char * const device_table[SND_DEVICE_MAX] = {
135 [SND_DEVICE_NONE] = "none",
136 /* Playback sound devices */
137 [SND_DEVICE_OUT_HANDSET] = "handset",
138 [SND_DEVICE_OUT_SPEAKER] = "speaker",
139 [SND_DEVICE_OUT_SPEAKER_REVERSE] = "speaker-reverse",
140 [SND_DEVICE_OUT_HEADPHONES] = "headphones",
141 [SND_DEVICE_OUT_SPEAKER_AND_HEADPHONES] = "speaker-and-headphones",
142 [SND_DEVICE_OUT_VOICE_HANDSET] = "voice-handset",
143 [SND_DEVICE_OUT_VOICE_SPEAKER] = "voice-speaker",
144 [SND_DEVICE_OUT_VOICE_HEADPHONES] = "voice-headphones",
145 [SND_DEVICE_OUT_HDMI] = "hdmi",
146 [SND_DEVICE_OUT_SPEAKER_AND_HDMI] = "speaker-and-hdmi",
147 [SND_DEVICE_OUT_BT_SCO] = "bt-sco-headset",
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700148 [SND_DEVICE_OUT_BT_SCO_WB] = "bt-sco-headset-wb",
Eric Laurentb23d5282013-05-14 15:27:20 -0700149 [SND_DEVICE_OUT_VOICE_TTY_FULL_HEADPHONES] = "voice-tty-full-headphones",
150 [SND_DEVICE_OUT_VOICE_TTY_VCO_HEADPHONES] = "voice-tty-vco-headphones",
151 [SND_DEVICE_OUT_VOICE_TTY_HCO_HANDSET] = "voice-tty-hco-handset",
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700152 [SND_DEVICE_OUT_AFE_PROXY] = "afe-proxy",
153 [SND_DEVICE_OUT_USB_HEADSET] = "usb-headphones",
154 [SND_DEVICE_OUT_SPEAKER_AND_USB_HEADSET] = "speaker-and-usb-headphones",
Apoorv Raghuvanshi6e262842013-10-06 14:39:35 -0700155 [SND_DEVICE_OUT_TRANSMISSION_FM] = "transmission-fm",
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700156 [SND_DEVICE_OUT_ANC_HEADSET] = "anc-headphones",
157 [SND_DEVICE_OUT_ANC_FB_HEADSET] = "anc-fb-headphones",
158 [SND_DEVICE_OUT_VOICE_ANC_HEADSET] = "voice-anc-headphones",
159 [SND_DEVICE_OUT_VOICE_ANC_FB_HEADSET] = "voice-anc-fb-headphones",
160 [SND_DEVICE_OUT_SPEAKER_AND_ANC_HEADSET] = "speaker-and-anc-headphones",
161 [SND_DEVICE_OUT_ANC_HANDSET] = "anc-handset",
Gopikrishnaiah Anandanf538cef2013-10-28 14:06:03 -0700162 [SND_DEVICE_OUT_SPEAKER_PROTECTED] = "speaker-protected",
Eric Laurentb23d5282013-05-14 15:27:20 -0700163
164 /* Capture sound devices */
165 [SND_DEVICE_IN_HANDSET_MIC] = "handset-mic",
Eric Laurentb23d5282013-05-14 15:27:20 -0700166 [SND_DEVICE_IN_HANDSET_MIC_AEC] = "handset-mic",
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800167 [SND_DEVICE_IN_HANDSET_MIC_NS] = "handset-mic",
168 [SND_DEVICE_IN_HANDSET_MIC_AEC_NS] = "handset-mic",
169 [SND_DEVICE_IN_HANDSET_DMIC] = "dmic-endfire",
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800170 [SND_DEVICE_IN_HANDSET_DMIC_AEC] = "dmic-endfire",
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800171 [SND_DEVICE_IN_HANDSET_DMIC_NS] = "dmic-endfire",
172 [SND_DEVICE_IN_HANDSET_DMIC_AEC_NS] = "dmic-endfire",
173 [SND_DEVICE_IN_SPEAKER_MIC] = "speaker-mic",
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800174 [SND_DEVICE_IN_SPEAKER_MIC_AEC] = "speaker-mic",
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800175 [SND_DEVICE_IN_SPEAKER_MIC_NS] = "speaker-mic",
176 [SND_DEVICE_IN_SPEAKER_MIC_AEC_NS] = "speaker-mic",
177 [SND_DEVICE_IN_SPEAKER_DMIC] = "speaker-dmic-endfire",
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800178 [SND_DEVICE_IN_SPEAKER_DMIC_AEC] = "speaker-dmic-endfire",
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800179 [SND_DEVICE_IN_SPEAKER_DMIC_NS] = "speaker-dmic-endfire",
180 [SND_DEVICE_IN_SPEAKER_DMIC_AEC_NS] = "speaker-dmic-endfire",
181 [SND_DEVICE_IN_HEADSET_MIC] = "headset-mic",
182 [SND_DEVICE_IN_HEADSET_MIC_FLUENCE] = "headset-mic",
Eric Laurentb23d5282013-05-14 15:27:20 -0700183 [SND_DEVICE_IN_VOICE_SPEAKER_MIC] = "voice-speaker-mic",
184 [SND_DEVICE_IN_VOICE_HEADSET_MIC] = "voice-headset-mic",
185 [SND_DEVICE_IN_HDMI_MIC] = "hdmi-mic",
186 [SND_DEVICE_IN_BT_SCO_MIC] = "bt-sco-mic",
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700187 [SND_DEVICE_IN_BT_SCO_MIC_WB] = "bt-sco-mic-wb",
Eric Laurentb23d5282013-05-14 15:27:20 -0700188 [SND_DEVICE_IN_CAMCORDER_MIC] = "camcorder-mic",
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700189 [SND_DEVICE_IN_VOICE_DMIC] = "voice-dmic-ef",
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700190 [SND_DEVICE_IN_VOICE_SPEAKER_DMIC] = "voice-speaker-dmic-ef",
Eric Laurentb23d5282013-05-14 15:27:20 -0700191 [SND_DEVICE_IN_VOICE_TTY_FULL_HEADSET_MIC] = "voice-tty-full-headset-mic",
192 [SND_DEVICE_IN_VOICE_TTY_VCO_HANDSET_MIC] = "voice-tty-vco-handset-mic",
193 [SND_DEVICE_IN_VOICE_TTY_HCO_HEADSET_MIC] = "voice-tty-hco-headset-mic",
194 [SND_DEVICE_IN_VOICE_REC_MIC] = "voice-rec-mic",
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800195 [SND_DEVICE_IN_VOICE_REC_MIC_NS] = "voice-rec-mic",
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800196 [SND_DEVICE_IN_VOICE_REC_DMIC_STEREO] = "voice-rec-dmic-ef",
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700197 [SND_DEVICE_IN_VOICE_REC_DMIC_FLUENCE] = "voice-rec-dmic-ef-fluence",
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700198 [SND_DEVICE_IN_USB_HEADSET_MIC] = "usb-headset-mic",
Apoorv Raghuvanshi6e262842013-10-06 14:39:35 -0700199 [SND_DEVICE_IN_CAPTURE_FM] = "capture-fm",
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700200 [SND_DEVICE_IN_AANC_HANDSET_MIC] = "aanc-handset-mic",
Apoorv Raghuvanshi6178a3f2013-10-19 12:38:54 -0700201 [SND_DEVICE_IN_QUAD_MIC] = "quad-mic",
Apoorv Raghuvanshi6bd8dbf2013-10-19 18:37:52 -0700202 [SND_DEVICE_IN_HANDSET_STEREO_DMIC] = "handset-stereo-dmic-ef",
203 [SND_DEVICE_IN_SPEAKER_STEREO_DMIC] = "speaker-stereo-dmic-ef",
Gopikrishnaiah Anandanf538cef2013-10-28 14:06:03 -0700204 [SND_DEVICE_IN_CAPTURE_VI_FEEDBACK] = "vi-feedback",
Eric Laurentb23d5282013-05-14 15:27:20 -0700205};
206
207/* ACDB IDs (audio DSP path configuration IDs) for each sound device */
208static const int acdb_device_table[SND_DEVICE_MAX] = {
209 [SND_DEVICE_NONE] = -1,
210 [SND_DEVICE_OUT_HANDSET] = 7,
Vidyakumar Athotac29d4ab2013-11-14 16:58:02 -0800211 [SND_DEVICE_OUT_SPEAKER] = 14,
212 [SND_DEVICE_OUT_SPEAKER_REVERSE] = 14,
Eric Laurentb23d5282013-05-14 15:27:20 -0700213 [SND_DEVICE_OUT_HEADPHONES] = 10,
214 [SND_DEVICE_OUT_SPEAKER_AND_HEADPHONES] = 10,
215 [SND_DEVICE_OUT_VOICE_HANDSET] = 7,
Vidyakumar Athotac29d4ab2013-11-14 16:58:02 -0800216 [SND_DEVICE_OUT_VOICE_SPEAKER] = 14,
Eric Laurentb23d5282013-05-14 15:27:20 -0700217 [SND_DEVICE_OUT_VOICE_HEADPHONES] = 10,
218 [SND_DEVICE_OUT_HDMI] = 18,
Vidyakumar Athotac29d4ab2013-11-14 16:58:02 -0800219 [SND_DEVICE_OUT_SPEAKER_AND_HDMI] = 14,
Eric Laurentb23d5282013-05-14 15:27:20 -0700220 [SND_DEVICE_OUT_BT_SCO] = 22,
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700221 [SND_DEVICE_OUT_BT_SCO_WB] = 39,
Eric Laurentb23d5282013-05-14 15:27:20 -0700222 [SND_DEVICE_OUT_VOICE_TTY_FULL_HEADPHONES] = 17,
223 [SND_DEVICE_OUT_VOICE_TTY_VCO_HEADPHONES] = 17,
224 [SND_DEVICE_OUT_VOICE_TTY_HCO_HANDSET] = 37,
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700225 [SND_DEVICE_OUT_AFE_PROXY] = 0,
226 [SND_DEVICE_OUT_USB_HEADSET] = 0,
227 [SND_DEVICE_OUT_SPEAKER_AND_USB_HEADSET] = 14,
Apoorv Raghuvanshi6e262842013-10-06 14:39:35 -0700228 [SND_DEVICE_OUT_TRANSMISSION_FM] = 0,
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700229 [SND_DEVICE_OUT_ANC_HEADSET] = 26,
Ravi Kumar Alamanda48c921d2013-10-29 06:07:44 -0700230 [SND_DEVICE_OUT_ANC_FB_HEADSET] = 27,
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700231 [SND_DEVICE_OUT_VOICE_ANC_HEADSET] = 26,
Ravi Kumar Alamanda48c921d2013-10-29 06:07:44 -0700232 [SND_DEVICE_OUT_VOICE_ANC_FB_HEADSET] = 27,
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700233 [SND_DEVICE_OUT_SPEAKER_AND_ANC_HEADSET] = 26,
234 [SND_DEVICE_OUT_ANC_HANDSET] = 103,
Gopikrishnaiah Anandanf538cef2013-10-28 14:06:03 -0700235 [SND_DEVICE_OUT_SPEAKER_PROTECTED] = 101,
Eric Laurentb23d5282013-05-14 15:27:20 -0700236
237 [SND_DEVICE_IN_HANDSET_MIC] = 4,
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800238 [SND_DEVICE_IN_HANDSET_MIC_AEC] = 106,
239 [SND_DEVICE_IN_HANDSET_MIC_NS] = 107,
240 [SND_DEVICE_IN_HANDSET_MIC_AEC_NS] = 108,
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800241 [SND_DEVICE_IN_HANDSET_DMIC] = 41,
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800242 [SND_DEVICE_IN_HANDSET_DMIC_AEC] = 109,
243 [SND_DEVICE_IN_HANDSET_DMIC_NS] = 110,
244 [SND_DEVICE_IN_HANDSET_DMIC_AEC_NS] = 111,
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800245 [SND_DEVICE_IN_SPEAKER_MIC] = 11,
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800246 [SND_DEVICE_IN_SPEAKER_MIC_AEC] = 112,
247 [SND_DEVICE_IN_SPEAKER_MIC_NS] = 113,
248 [SND_DEVICE_IN_SPEAKER_MIC_AEC_NS] = 114,
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800249 [SND_DEVICE_IN_SPEAKER_DMIC] = 43,
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800250 [SND_DEVICE_IN_SPEAKER_DMIC_AEC] = 115,
251 [SND_DEVICE_IN_SPEAKER_DMIC_NS] = 116,
252 [SND_DEVICE_IN_SPEAKER_DMIC_AEC_NS] = 117,
Eric Laurentb23d5282013-05-14 15:27:20 -0700253 [SND_DEVICE_IN_HEADSET_MIC] = 8,
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800254 [SND_DEVICE_IN_HEADSET_MIC_FLUENCE] = 47,
Eric Laurentb23d5282013-05-14 15:27:20 -0700255 [SND_DEVICE_IN_VOICE_SPEAKER_MIC] = 11,
256 [SND_DEVICE_IN_VOICE_HEADSET_MIC] = 8,
257 [SND_DEVICE_IN_HDMI_MIC] = 4,
258 [SND_DEVICE_IN_BT_SCO_MIC] = 21,
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700259 [SND_DEVICE_IN_BT_SCO_MIC_WB] = 38,
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800260 [SND_DEVICE_IN_CAMCORDER_MIC] = 4,
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700261 [SND_DEVICE_IN_VOICE_DMIC] = 41,
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700262 [SND_DEVICE_IN_VOICE_SPEAKER_DMIC] = 43,
Eric Laurentb23d5282013-05-14 15:27:20 -0700263 [SND_DEVICE_IN_VOICE_TTY_FULL_HEADSET_MIC] = 16,
264 [SND_DEVICE_IN_VOICE_TTY_VCO_HANDSET_MIC] = 36,
265 [SND_DEVICE_IN_VOICE_TTY_HCO_HEADSET_MIC] = 16,
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800266 [SND_DEVICE_IN_VOICE_REC_MIC] = 4,
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800267 [SND_DEVICE_IN_VOICE_REC_MIC_NS] = 107,
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800268 [SND_DEVICE_IN_VOICE_REC_DMIC_STEREO] = 34,
269 [SND_DEVICE_IN_VOICE_REC_DMIC_FLUENCE] = 41,
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700270 [SND_DEVICE_IN_USB_HEADSET_MIC] = 44,
Apoorv Raghuvanshi6e262842013-10-06 14:39:35 -0700271 [SND_DEVICE_IN_CAPTURE_FM] = 0,
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700272 [SND_DEVICE_IN_AANC_HANDSET_MIC] = 104,
Apoorv Raghuvanshi6178a3f2013-10-19 12:38:54 -0700273 [SND_DEVICE_IN_QUAD_MIC] = 46,
Apoorv Raghuvanshi6bd8dbf2013-10-19 18:37:52 -0700274 [SND_DEVICE_IN_HANDSET_STEREO_DMIC] = 34,
275 [SND_DEVICE_IN_SPEAKER_STEREO_DMIC] = 35,
Gopikrishnaiah Anandanf538cef2013-10-28 14:06:03 -0700276 [SND_DEVICE_IN_CAPTURE_VI_FEEDBACK] = 102,
Eric Laurentb23d5282013-05-14 15:27:20 -0700277};
278
Haynes Mathew George7ff216f2013-09-11 19:51:41 -0700279#define DEEP_BUFFER_PLATFORM_DELAY (29*1000LL)
280#define LOW_LATENCY_PLATFORM_DELAY (13*1000LL)
281
Eric Laurentb23d5282013-05-14 15:27:20 -0700282static int set_echo_reference(struct mixer *mixer, const char* ec_ref)
283{
284 struct mixer_ctl *ctl;
285 const char *mixer_ctl_name = "EC_REF_RX";
286
287 ctl = mixer_get_ctl_by_name(mixer, mixer_ctl_name);
288 if (!ctl) {
289 ALOGE("%s: Could not get ctl for mixer cmd - %s",
290 __func__, mixer_ctl_name);
291 return -EINVAL;
292 }
293 ALOGV("Setting EC Reference: %s", ec_ref);
294 mixer_ctl_set_enum_by_string(ctl, ec_ref);
295 return 0;
296}
297
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800298static struct csd_data *open_csd_client()
299{
300 struct csd_data *csd = calloc(1, sizeof(struct csd_data));
301
302 csd->csd_client = dlopen(LIB_CSD_CLIENT, RTLD_NOW);
303 if (csd->csd_client == NULL) {
304 ALOGE("%s: DLOPEN failed for %s", __func__, LIB_CSD_CLIENT);
305 goto error;
306 } else {
307 ALOGV("%s: DLOPEN successful for %s", __func__, LIB_CSD_CLIENT);
308
309 csd->deinit = (deinit_t)dlsym(csd->csd_client,
310 "csd_client_deinit");
311 if (csd->deinit == NULL) {
312 ALOGE("%s: dlsym error %s for csd_client_deinit", __func__,
313 dlerror());
314 goto error;
315 }
316 csd->disable_device = (disable_device_t)dlsym(csd->csd_client,
317 "csd_client_disable_device");
318 if (csd->disable_device == NULL) {
319 ALOGE("%s: dlsym error %s for csd_client_disable_device",
320 __func__, dlerror());
321 goto error;
322 }
323 csd->enable_device = (enable_device_t)dlsym(csd->csd_client,
324 "csd_client_enable_device");
325 if (csd->enable_device == NULL) {
326 ALOGE("%s: dlsym error %s for csd_client_enable_device",
327 __func__, dlerror());
328 goto error;
329 }
330 csd->start_voice = (start_voice_t)dlsym(csd->csd_client,
331 "csd_client_start_voice");
332 if (csd->start_voice == NULL) {
333 ALOGE("%s: dlsym error %s for csd_client_start_voice",
334 __func__, dlerror());
335 goto error;
336 }
337 csd->stop_voice = (stop_voice_t)dlsym(csd->csd_client,
338 "csd_client_stop_voice");
339 if (csd->stop_voice == NULL) {
340 ALOGE("%s: dlsym error %s for csd_client_stop_voice",
341 __func__, dlerror());
342 goto error;
343 }
344 csd->volume = (volume_t)dlsym(csd->csd_client,
345 "csd_client_volume");
346 if (csd->volume == NULL) {
347 ALOGE("%s: dlsym error %s for csd_client_volume",
348 __func__, dlerror());
349 goto error;
350 }
351 csd->mic_mute = (mic_mute_t)dlsym(csd->csd_client,
352 "csd_client_mic_mute");
353 if (csd->mic_mute == NULL) {
354 ALOGE("%s: dlsym error %s for csd_client_mic_mute",
355 __func__, dlerror());
356 goto error;
357 }
358 csd->slow_talk = (slow_talk_t)dlsym(csd->csd_client,
359 "csd_client_slow_talk");
360 if (csd->slow_talk == NULL) {
361 ALOGE("%s: dlsym error %s for csd_client_slow_talk",
362 __func__, dlerror());
363 goto error;
364 }
365 csd->start_playback = (start_playback_t)dlsym(csd->csd_client,
366 "csd_client_start_playback");
367 if (csd->start_playback == NULL) {
368 ALOGE("%s: dlsym error %s for csd_client_start_playback",
369 __func__, dlerror());
370 goto error;
371 }
372 csd->stop_playback = (stop_playback_t)dlsym(csd->csd_client,
373 "csd_client_stop_playback");
374 if (csd->stop_playback == NULL) {
375 ALOGE("%s: dlsym error %s for csd_client_stop_playback",
376 __func__, dlerror());
377 goto error;
378 }
379 csd->start_record = (start_record_t)dlsym(csd->csd_client,
380 "csd_client_start_record");
381 if (csd->start_record == NULL) {
382 ALOGE("%s: dlsym error %s for csd_client_start_record",
383 __func__, dlerror());
384 goto error;
385 }
386 csd->stop_record = (stop_record_t)dlsym(csd->csd_client,
387 "csd_client_stop_record");
388 if (csd->stop_record == NULL) {
389 ALOGE("%s: dlsym error %s for csd_client_stop_record",
390 __func__, dlerror());
391 goto error;
392 }
393 csd->init = (init_t)dlsym(csd->csd_client, "csd_client_init");
394
395 if (csd->init == NULL) {
396 ALOGE("%s: dlsym error %s for csd_client_init",
397 __func__, dlerror());
398 goto error;
399 } else {
400 csd->init();
401 }
402 }
403 return csd;
404
405error:
406 free(csd);
407 csd = NULL;
408 return csd;
409}
410
411void close_csd_client(struct csd_data *csd)
412{
413 if (csd != NULL) {
414 csd->deinit();
415 dlclose(csd->csd_client);
416 free(csd);
417 csd = NULL;
418 }
419}
420
Eric Laurentb23d5282013-05-14 15:27:20 -0700421void *platform_init(struct audio_device *adev)
422{
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800423 char platform[PROPERTY_VALUE_MAX];
424 char baseband[PROPERTY_VALUE_MAX];
Eric Laurentb23d5282013-05-14 15:27:20 -0700425 char value[PROPERTY_VALUE_MAX];
426 struct platform_data *my_data;
Ravi Kumar Alamanda48c921d2013-10-29 06:07:44 -0700427 int retry_num = 0;
428 const char *snd_card_name;
sangwoo1b9f4b32013-06-21 18:22:55 -0700429
430 adev->mixer = mixer_open(MIXER_CARD);
431
432 while (!adev->mixer && retry_num < RETRY_NUMBER) {
433 usleep(RETRY_US);
434 adev->mixer = mixer_open(MIXER_CARD);
435 retry_num++;
436 }
437
438 if (!adev->mixer) {
439 ALOGE("Unable to open the mixer, aborting.");
440 return NULL;
441 }
442
Damir Didjustof1d46c72013-11-06 17:59:04 -0800443 if (audio_extn_read_xml(adev, MIXER_CARD, MIXER_XML_PATH,
444 MIXER_XML_PATH_AUXPCM) == -ENOSYS)
445 adev->audio_route = audio_route_init(MIXER_CARD, MIXER_XML_PATH);
446
sangwoo1b9f4b32013-06-21 18:22:55 -0700447 if (!adev->audio_route) {
448 ALOGE("%s: Failed to init audio route controls, aborting.", __func__);
449 return NULL;
450 }
Eric Laurentb23d5282013-05-14 15:27:20 -0700451
452 my_data = calloc(1, sizeof(struct platform_data));
453
Ravi Kumar Alamanda48c921d2013-10-29 06:07:44 -0700454 snd_card_name = mixer_get_name(adev->mixer);
455 my_data->hw_info = hw_info_init(snd_card_name);
456 if (!my_data->hw_info) {
457 ALOGE("%s: Failed to init hardware info", __func__);
458 }
459
Eric Laurentb23d5282013-05-14 15:27:20 -0700460 my_data->adev = adev;
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700461 my_data->btsco_sample_rate = SAMPLE_RATE_8KHZ;
Eric Laurentb23d5282013-05-14 15:27:20 -0700462 my_data->fluence_in_spkr_mode = false;
463 my_data->fluence_in_voice_call = false;
464 my_data->fluence_in_voice_rec = false;
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800465 my_data->fluence_in_audio_rec = false;
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700466 my_data->fluence_type = FLUENCE_NONE;
Eric Laurentb23d5282013-05-14 15:27:20 -0700467
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700468 property_get("ro.qc.sdk.audio.fluencetype", value, "");
469 if (!strncmp("fluencepro", value, sizeof("fluencepro"))) {
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800470 my_data->fluence_type = FLUENCE_QUAD_MIC | FLUENCE_DUAL_MIC;
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700471 } else if (!strncmp("fluence", value, sizeof("fluence"))) {
472 my_data->fluence_type = FLUENCE_DUAL_MIC;
473 } else {
474 my_data->fluence_type = FLUENCE_NONE;
Eric Laurentb23d5282013-05-14 15:27:20 -0700475 }
476
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700477 if (my_data->fluence_type != FLUENCE_NONE) {
Eric Laurentb23d5282013-05-14 15:27:20 -0700478 property_get("persist.audio.fluence.voicecall",value,"");
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700479 if (!strncmp("true", value, sizeof("true"))) {
Eric Laurentb23d5282013-05-14 15:27:20 -0700480 my_data->fluence_in_voice_call = true;
481 }
482
483 property_get("persist.audio.fluence.voicerec",value,"");
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700484 if (!strncmp("true", value, sizeof("true"))) {
Eric Laurentb23d5282013-05-14 15:27:20 -0700485 my_data->fluence_in_voice_rec = true;
486 }
487
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800488 property_get("persist.audio.fluence.audiorec",value,"");
489 if (!strncmp("true", value, sizeof("true"))) {
490 my_data->fluence_in_audio_rec = true;
491 }
492
Eric Laurentb23d5282013-05-14 15:27:20 -0700493 property_get("persist.audio.fluence.speaker",value,"");
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700494 if (!strncmp("true", value, sizeof("true"))) {
Eric Laurentb23d5282013-05-14 15:27:20 -0700495 my_data->fluence_in_spkr_mode = true;
496 }
497 }
498
499 my_data->acdb_handle = dlopen(LIB_ACDB_LOADER, RTLD_NOW);
500 if (my_data->acdb_handle == NULL) {
501 ALOGE("%s: DLOPEN failed for %s", __func__, LIB_ACDB_LOADER);
502 } else {
503 ALOGV("%s: DLOPEN successful for %s", __func__, LIB_ACDB_LOADER);
504 my_data->acdb_deallocate = (acdb_deallocate_t)dlsym(my_data->acdb_handle,
505 "acdb_loader_deallocate_ACDB");
506 my_data->acdb_send_audio_cal = (acdb_send_audio_cal_t)dlsym(my_data->acdb_handle,
507 "acdb_loader_send_audio_cal");
508 if (!my_data->acdb_send_audio_cal)
509 ALOGW("%s: Could not find the symbol acdb_send_audio_cal from %s",
510 __func__, LIB_ACDB_LOADER);
511 my_data->acdb_send_voice_cal = (acdb_send_voice_cal_t)dlsym(my_data->acdb_handle,
512 "acdb_loader_send_voice_cal");
513 my_data->acdb_init = (acdb_init_t)dlsym(my_data->acdb_handle,
514 "acdb_loader_init_ACDB");
515 if (my_data->acdb_init == NULL)
516 ALOGE("%s: dlsym error %s for acdb_loader_init_ACDB", __func__, dlerror());
517 else
518 my_data->acdb_init();
519 }
520
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800521 /* If platform is apq8084 and baseband is MDM, load CSD Client specific
522 * symbols. Voice call is handled by MDM and apps processor talks to
523 * MDM through CSD Client
524 */
525 property_get("ro.board.platform", platform, "");
526 property_get("ro.baseband", baseband, "");
527 if (!strncmp("apq8084", platform, sizeof("apq8084")) &&
528 !strncmp("mdm", baseband, sizeof("mdm"))) {
529 my_data->csd = open_csd_client();
530 }
531
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700532 /* init usb */
533 audio_extn_usb_init(adev);
534
Apoorv Raghuvanshi6178a3f2013-10-19 12:38:54 -0700535 /* Read one time ssr property */
536 audio_extn_ssr_update_enabled(adev);
Gopikrishnaiah Anandanf538cef2013-10-28 14:06:03 -0700537 audio_extn_spkr_prot_init(adev);
Eric Laurentb23d5282013-05-14 15:27:20 -0700538 return my_data;
539}
540
541void platform_deinit(void *platform)
542{
Ravi Kumar Alamanda48c921d2013-10-29 06:07:44 -0700543 struct platform_data *my_data = (struct platform_data *)platform;
544
545 hw_info_deinit(my_data->hw_info);
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800546 close_csd_client(my_data->csd);
547
Eric Laurentb23d5282013-05-14 15:27:20 -0700548 free(platform);
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700549 /* deinit usb */
550 audio_extn_usb_deinit();
Eric Laurentb23d5282013-05-14 15:27:20 -0700551}
552
553const char *platform_get_snd_device_name(snd_device_t snd_device)
554{
555 if (snd_device >= SND_DEVICE_MIN && snd_device < SND_DEVICE_MAX)
556 return device_table[snd_device];
557 else
558 return "";
559}
560
Ravi Kumar Alamanda48c921d2013-10-29 06:07:44 -0700561int platform_get_snd_device_name_extn(void *platform, snd_device_t snd_device,
562 char *device_name)
563{
564 struct platform_data *my_data = (struct platform_data *)platform;
565
566 if (snd_device >= SND_DEVICE_MIN && snd_device < SND_DEVICE_MAX) {
567 strlcpy(device_name, device_table[snd_device], DEVICE_NAME_MAX_SIZE);
568 hw_info_append_hw_type(my_data->hw_info, snd_device, device_name);
569 } else {
570 strlcpy(device_name, "", DEVICE_NAME_MAX_SIZE);
571 return -EINVAL;
572 }
573
574 return 0;
575}
576
Eric Laurentb23d5282013-05-14 15:27:20 -0700577void platform_add_backend_name(char *mixer_path, snd_device_t snd_device)
578{
579 if (snd_device == SND_DEVICE_IN_BT_SCO_MIC)
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700580 strlcat(mixer_path, " bt-sco", MIXER_PATH_MAX_LENGTH);
581 else if (snd_device == SND_DEVICE_IN_BT_SCO_MIC_WB)
582 strlcat(mixer_path, " bt-sco-wb", MIXER_PATH_MAX_LENGTH);
Eric Laurentb23d5282013-05-14 15:27:20 -0700583 else if(snd_device == SND_DEVICE_OUT_BT_SCO)
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700584 strlcat(mixer_path, " bt-sco", MIXER_PATH_MAX_LENGTH);
585 else if(snd_device == SND_DEVICE_OUT_BT_SCO_WB)
586 strlcat(mixer_path, " bt-sco-wb", MIXER_PATH_MAX_LENGTH);
Eric Laurentb23d5282013-05-14 15:27:20 -0700587 else if (snd_device == SND_DEVICE_OUT_HDMI)
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700588 strlcat(mixer_path, " hdmi", MIXER_PATH_MAX_LENGTH);
Eric Laurentb23d5282013-05-14 15:27:20 -0700589 else if (snd_device == SND_DEVICE_OUT_SPEAKER_AND_HDMI)
590 strcat(mixer_path, " speaker-and-hdmi");
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700591 else if (snd_device == SND_DEVICE_OUT_AFE_PROXY)
592 strlcat(mixer_path, " afe-proxy", MIXER_PATH_MAX_LENGTH);
593 else if (snd_device == SND_DEVICE_OUT_USB_HEADSET)
594 strlcat(mixer_path, " usb-headphones", MIXER_PATH_MAX_LENGTH);
595 else if (snd_device == SND_DEVICE_OUT_SPEAKER_AND_USB_HEADSET)
596 strlcat(mixer_path, " speaker-and-usb-headphones",
597 MIXER_PATH_MAX_LENGTH);
598 else if (snd_device == SND_DEVICE_IN_USB_HEADSET_MIC)
599 strlcat(mixer_path, " usb-headset-mic", MIXER_PATH_MAX_LENGTH);
Apoorv Raghuvanshi6e262842013-10-06 14:39:35 -0700600 else if (snd_device == SND_DEVICE_IN_CAPTURE_FM)
601 strlcat(mixer_path, " capture-fm", MIXER_PATH_MAX_LENGTH);
602 else if (snd_device == SND_DEVICE_OUT_TRANSMISSION_FM)
603 strlcat(mixer_path, " transmission-fm", MIXER_PATH_MAX_LENGTH);
Eric Laurentb23d5282013-05-14 15:27:20 -0700604}
605
606int platform_get_pcm_device_id(audio_usecase_t usecase, int device_type)
607{
608 int device_id;
609 if (device_type == PCM_PLAYBACK)
610 device_id = pcm_device_table[usecase][0];
611 else
612 device_id = pcm_device_table[usecase][1];
613 return device_id;
614}
615
616int platform_send_audio_calibration(void *platform, snd_device_t snd_device)
617{
618 struct platform_data *my_data = (struct platform_data *)platform;
619 int acdb_dev_id, acdb_dev_type;
620
621 acdb_dev_id = acdb_device_table[snd_device];
622 if (acdb_dev_id < 0) {
623 ALOGE("%s: Could not find acdb id for device(%d)",
624 __func__, snd_device);
625 return -EINVAL;
626 }
627 if (my_data->acdb_send_audio_cal) {
Eric Laurent994a6932013-07-17 11:51:42 -0700628 ("%s: sending audio calibration for snd_device(%d) acdb_id(%d)",
Eric Laurentb23d5282013-05-14 15:27:20 -0700629 __func__, snd_device, acdb_dev_id);
630 if (snd_device >= SND_DEVICE_OUT_BEGIN &&
631 snd_device < SND_DEVICE_OUT_END)
632 acdb_dev_type = ACDB_DEV_TYPE_OUT;
633 else
634 acdb_dev_type = ACDB_DEV_TYPE_IN;
635 my_data->acdb_send_audio_cal(acdb_dev_id, acdb_dev_type);
636 }
637 return 0;
638}
639
640int platform_switch_voice_call_device_pre(void *platform)
641{
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800642 struct platform_data *my_data = (struct platform_data *)platform;
643 int ret = 0;
644
645 if (my_data->csd != NULL) {
646 /* This must be called before disabling mixer controls on APQ side */
647 ret = my_data->csd->disable_device();
648 if (ret < 0) {
649 ALOGE("%s: csd_client_disable_device, failed, error %d",
650 __func__, ret);
651 }
652 }
653 return ret;
Eric Laurentb23d5282013-05-14 15:27:20 -0700654}
655
656int platform_switch_voice_call_device_post(void *platform,
657 snd_device_t out_snd_device,
658 snd_device_t in_snd_device)
659{
660 struct platform_data *my_data = (struct platform_data *)platform;
661 int acdb_rx_id, acdb_tx_id;
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800662 int ret = 0;
663
664 acdb_rx_id = acdb_device_table[out_snd_device];
665 acdb_tx_id = acdb_device_table[in_snd_device];
Eric Laurentb23d5282013-05-14 15:27:20 -0700666
667 if (my_data->acdb_send_voice_cal == NULL) {
668 ALOGE("%s: dlsym error for acdb_send_voice_call", __func__);
669 } else {
Eric Laurentb23d5282013-05-14 15:27:20 -0700670 if (acdb_rx_id > 0 && acdb_tx_id > 0)
671 my_data->acdb_send_voice_cal(acdb_rx_id, acdb_tx_id);
672 else
673 ALOGE("%s: Incorrect ACDB IDs (rx: %d tx: %d)", __func__,
674 acdb_rx_id, acdb_tx_id);
675 }
676
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800677 if (my_data->csd != NULL) {
678 if (acdb_rx_id > 0 || acdb_tx_id > 0) {
679 ret = my_data->csd->enable_device(acdb_rx_id, acdb_tx_id,
680 my_data->adev->acdb_settings);
681 if (ret < 0) {
682 ALOGE("%s: csd_enable_device, failed, error %d",
683 __func__, ret);
684 }
685 } else {
686 ALOGE("%s: Incorrect ACDB IDs (rx: %d tx: %d)", __func__,
687 acdb_rx_id, acdb_tx_id);
688 }
689 }
690 return ret;
Eric Laurentb23d5282013-05-14 15:27:20 -0700691}
692
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800693int platform_start_voice_call(void *platform, uint32_t vsid)
Eric Laurentb23d5282013-05-14 15:27:20 -0700694{
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800695 struct platform_data *my_data = (struct platform_data *)platform;
696 int ret = 0;
697
698 if (my_data->csd != NULL) {
699 ret = my_data->csd->start_voice(vsid);
700 if (ret < 0) {
701 ALOGE("%s: csd_start_voice error %d\n", __func__, ret);
702 }
703 }
704 return ret;
Eric Laurentb23d5282013-05-14 15:27:20 -0700705}
706
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800707int platform_stop_voice_call(void *platform, uint32_t vsid)
Eric Laurentb23d5282013-05-14 15:27:20 -0700708{
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800709 struct platform_data *my_data = (struct platform_data *)platform;
710 int ret = 0;
711
712 if (my_data->csd != NULL) {
713 ret = my_data->csd->stop_voice(vsid);
714 if (ret < 0) {
715 ALOGE("%s: csd_stop_voice error %d\n", __func__, ret);
716 }
717 }
718 return ret;
Eric Laurentb23d5282013-05-14 15:27:20 -0700719}
720
721int platform_set_voice_volume(void *platform, int volume)
722{
723 struct platform_data *my_data = (struct platform_data *)platform;
724 struct audio_device *adev = my_data->adev;
725 struct mixer_ctl *ctl;
sangwoo53b2cf02013-07-25 19:18:44 -0700726 const char *mixer_ctl_name = "Voice Rx Gain";
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800727 int vol_index = 0, ret = 0;
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700728 uint32_t set_values[ ] = {0,
729 ALL_SESSION_VSID,
730 DEFAULT_VOLUME_RAMP_DURATION_MS};
Eric Laurentb23d5282013-05-14 15:27:20 -0700731
732 // Voice volume levels are mapped to adsp volume levels as follows.
733 // 100 -> 5, 80 -> 4, 60 -> 3, 40 -> 2, 20 -> 1 0 -> 0
734 // But this values don't changed in kernel. So, below change is need.
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700735 vol_index = (int)percent_to_index(volume, MIN_VOL_INDEX, MAX_VOL_INDEX);
736 set_values[0] = vol_index;
Eric Laurentb23d5282013-05-14 15:27:20 -0700737
738 ctl = mixer_get_ctl_by_name(adev->mixer, mixer_ctl_name);
739 if (!ctl) {
740 ALOGE("%s: Could not get ctl for mixer cmd - %s",
741 __func__, mixer_ctl_name);
742 return -EINVAL;
743 }
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700744 ALOGV("Setting voice volume index: %d", set_values[0]);
745 mixer_ctl_set_array(ctl, set_values, ARRAY_SIZE(set_values));
Eric Laurentb23d5282013-05-14 15:27:20 -0700746
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800747 if (my_data->csd != NULL) {
748 ret = my_data->csd->volume(ALL_SESSION_VSID, volume);
749 if (ret < 0) {
750 ALOGE("%s: csd_volume error %d", __func__, ret);
751 }
752 }
753 return ret;
Eric Laurentb23d5282013-05-14 15:27:20 -0700754}
755
756int platform_set_mic_mute(void *platform, bool state)
757{
758 struct platform_data *my_data = (struct platform_data *)platform;
759 struct audio_device *adev = my_data->adev;
760 struct mixer_ctl *ctl;
761 const char *mixer_ctl_name = "Voice Tx Mute";
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800762 int ret = 0;
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700763 uint32_t set_values[ ] = {0,
764 ALL_SESSION_VSID,
765 DEFAULT_VOLUME_RAMP_DURATION_MS};
Eric Laurentb23d5282013-05-14 15:27:20 -0700766
767 if (adev->mode == AUDIO_MODE_IN_CALL) {
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700768 set_values[0] = state;
Eric Laurentb23d5282013-05-14 15:27:20 -0700769 ctl = mixer_get_ctl_by_name(adev->mixer, mixer_ctl_name);
770 if (!ctl) {
771 ALOGE("%s: Could not get ctl for mixer cmd - %s",
772 __func__, mixer_ctl_name);
773 return -EINVAL;
774 }
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700775 ALOGV("Setting voice mute state: %d", state);
776 mixer_ctl_set_array(ctl, set_values, ARRAY_SIZE(set_values));
Eric Laurentb23d5282013-05-14 15:27:20 -0700777
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800778 if (my_data->csd != NULL) {
779 ret = my_data->csd->mic_mute(ALL_SESSION_VSID, state);
780 if (ret < 0) {
781 ALOGE("%s: csd_mic_mute error %d", __func__, ret);
782 }
783 }
784 }
785 return ret;
Eric Laurentb23d5282013-05-14 15:27:20 -0700786}
787
788snd_device_t platform_get_output_snd_device(void *platform, audio_devices_t devices)
789{
790 struct platform_data *my_data = (struct platform_data *)platform;
791 struct audio_device *adev = my_data->adev;
792 audio_mode_t mode = adev->mode;
793 snd_device_t snd_device = SND_DEVICE_NONE;
794
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700795 audio_channel_mask_t channel_mask = (adev->active_input == NULL) ?
796 AUDIO_CHANNEL_IN_MONO : adev->active_input->channel_mask;
797 int channel_count = popcount(channel_mask);
798
Eric Laurentb23d5282013-05-14 15:27:20 -0700799 ALOGV("%s: enter: output devices(%#x)", __func__, devices);
800 if (devices == AUDIO_DEVICE_NONE ||
801 devices & AUDIO_DEVICE_BIT_IN) {
802 ALOGV("%s: Invalid output devices (%#x)", __func__, devices);
803 goto exit;
804 }
805
806 if (mode == AUDIO_MODE_IN_CALL) {
807 if (devices & AUDIO_DEVICE_OUT_WIRED_HEADPHONE ||
808 devices & AUDIO_DEVICE_OUT_WIRED_HEADSET) {
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700809 if (adev->voice.tty_mode == TTY_MODE_FULL) {
Eric Laurentb23d5282013-05-14 15:27:20 -0700810 snd_device = SND_DEVICE_OUT_VOICE_TTY_FULL_HEADPHONES;
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700811 } else if (adev->voice.tty_mode == TTY_MODE_VCO) {
Eric Laurentb23d5282013-05-14 15:27:20 -0700812 snd_device = SND_DEVICE_OUT_VOICE_TTY_VCO_HEADPHONES;
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700813 } else if (adev->voice.tty_mode == TTY_MODE_HCO) {
Eric Laurentb23d5282013-05-14 15:27:20 -0700814 snd_device = SND_DEVICE_OUT_VOICE_TTY_HCO_HANDSET;
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700815 } else if (audio_extn_get_anc_enabled()) {
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700816 if (audio_extn_should_use_fb_anc())
817 snd_device = SND_DEVICE_OUT_VOICE_ANC_FB_HEADSET;
818 else
819 snd_device = SND_DEVICE_OUT_VOICE_ANC_HEADSET;
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700820 } else {
Eric Laurentb23d5282013-05-14 15:27:20 -0700821 snd_device = SND_DEVICE_OUT_VOICE_HEADPHONES;
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700822 }
Eric Laurentb23d5282013-05-14 15:27:20 -0700823 } else if (devices & AUDIO_DEVICE_OUT_ALL_SCO) {
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700824 if (my_data->btsco_sample_rate == SAMPLE_RATE_16KHZ)
825 snd_device = SND_DEVICE_OUT_BT_SCO_WB;
826 else
827 snd_device = SND_DEVICE_OUT_BT_SCO;
Eric Laurentb23d5282013-05-14 15:27:20 -0700828 } else if (devices & AUDIO_DEVICE_OUT_SPEAKER) {
829 snd_device = SND_DEVICE_OUT_VOICE_SPEAKER;
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700830 } else if (devices & AUDIO_DEVICE_OUT_ANLG_DOCK_HEADSET ||
831 devices & AUDIO_DEVICE_OUT_DGTL_DOCK_HEADSET) {
832 snd_device = SND_DEVICE_OUT_USB_HEADSET;
Apoorv Raghuvanshi6e262842013-10-06 14:39:35 -0700833 } else if (devices & AUDIO_DEVICE_OUT_FM_TX) {
834 snd_device = SND_DEVICE_OUT_TRANSMISSION_FM;
Eric Laurentb23d5282013-05-14 15:27:20 -0700835 } else if (devices & AUDIO_DEVICE_OUT_EARPIECE) {
Ravi Kumar Alamandaceb40822013-11-06 11:01:47 -0800836 if (audio_extn_should_use_handset_anc(channel_count))
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700837 snd_device = SND_DEVICE_OUT_ANC_HANDSET;
Eric Laurentb23d5282013-05-14 15:27:20 -0700838 else
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700839 snd_device = SND_DEVICE_OUT_VOICE_HANDSET;
Eric Laurentb23d5282013-05-14 15:27:20 -0700840 }
841 if (snd_device != SND_DEVICE_NONE) {
842 goto exit;
843 }
844 }
845
846 if (popcount(devices) == 2) {
847 if (devices == (AUDIO_DEVICE_OUT_WIRED_HEADPHONE |
848 AUDIO_DEVICE_OUT_SPEAKER)) {
849 snd_device = SND_DEVICE_OUT_SPEAKER_AND_HEADPHONES;
850 } else if (devices == (AUDIO_DEVICE_OUT_WIRED_HEADSET |
851 AUDIO_DEVICE_OUT_SPEAKER)) {
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700852 if (audio_extn_get_anc_enabled())
853 snd_device = SND_DEVICE_OUT_SPEAKER_AND_ANC_HEADSET;
854 else
855 snd_device = SND_DEVICE_OUT_SPEAKER_AND_HEADPHONES;
Eric Laurentb23d5282013-05-14 15:27:20 -0700856 } else if (devices == (AUDIO_DEVICE_OUT_AUX_DIGITAL |
857 AUDIO_DEVICE_OUT_SPEAKER)) {
858 snd_device = SND_DEVICE_OUT_SPEAKER_AND_HDMI;
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700859 } else if (devices == (AUDIO_DEVICE_OUT_ANLG_DOCK_HEADSET |
860 AUDIO_DEVICE_OUT_SPEAKER)) {
861 snd_device = SND_DEVICE_OUT_SPEAKER_AND_USB_HEADSET;
Eric Laurentb23d5282013-05-14 15:27:20 -0700862 } else {
863 ALOGE("%s: Invalid combo device(%#x)", __func__, devices);
864 goto exit;
865 }
866 if (snd_device != SND_DEVICE_NONE) {
867 goto exit;
868 }
869 }
870
871 if (popcount(devices) != 1) {
872 ALOGE("%s: Invalid output devices(%#x)", __func__, devices);
873 goto exit;
874 }
875
876 if (devices & AUDIO_DEVICE_OUT_WIRED_HEADPHONE ||
877 devices & AUDIO_DEVICE_OUT_WIRED_HEADSET) {
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700878 if (devices & AUDIO_DEVICE_OUT_WIRED_HEADSET
879 && audio_extn_get_anc_enabled()) {
880 if (audio_extn_should_use_fb_anc())
881 snd_device = SND_DEVICE_OUT_ANC_FB_HEADSET;
882 else
883 snd_device = SND_DEVICE_OUT_ANC_HEADSET;
884 }
885 else
886 snd_device = SND_DEVICE_OUT_HEADPHONES;
Eric Laurentb23d5282013-05-14 15:27:20 -0700887 } else if (devices & AUDIO_DEVICE_OUT_SPEAKER) {
888 if (adev->speaker_lr_swap)
889 snd_device = SND_DEVICE_OUT_SPEAKER_REVERSE;
890 else
891 snd_device = SND_DEVICE_OUT_SPEAKER;
892 } else if (devices & AUDIO_DEVICE_OUT_ALL_SCO) {
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700893 if (my_data->btsco_sample_rate == SAMPLE_RATE_16KHZ)
894 snd_device = SND_DEVICE_OUT_BT_SCO_WB;
895 else
896 snd_device = SND_DEVICE_OUT_BT_SCO;
Eric Laurentb23d5282013-05-14 15:27:20 -0700897 } else if (devices & AUDIO_DEVICE_OUT_AUX_DIGITAL) {
898 snd_device = SND_DEVICE_OUT_HDMI ;
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700899 } else if (devices & AUDIO_DEVICE_OUT_ANLG_DOCK_HEADSET ||
900 devices & AUDIO_DEVICE_OUT_DGTL_DOCK_HEADSET) {
901 snd_device = SND_DEVICE_OUT_USB_HEADSET;
Apoorv Raghuvanshi6e262842013-10-06 14:39:35 -0700902 } else if (devices & AUDIO_DEVICE_OUT_FM_TX) {
903 snd_device = SND_DEVICE_OUT_TRANSMISSION_FM;
Eric Laurentb23d5282013-05-14 15:27:20 -0700904 } else if (devices & AUDIO_DEVICE_OUT_EARPIECE) {
905 snd_device = SND_DEVICE_OUT_HANDSET;
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700906 } else if (devices & AUDIO_DEVICE_OUT_PROXY) {
907 ALOGD("%s: setting sink capability for Proxy", __func__);
908 audio_extn_set_afe_proxy_channel_mixer(adev);
909 snd_device = SND_DEVICE_OUT_AFE_PROXY;
Eric Laurentb23d5282013-05-14 15:27:20 -0700910 } else {
911 ALOGE("%s: Unknown device(s) %#x", __func__, devices);
912 }
913exit:
914 ALOGV("%s: exit: snd_device(%s)", __func__, device_table[snd_device]);
915 return snd_device;
916}
917
918snd_device_t platform_get_input_snd_device(void *platform, audio_devices_t out_device)
919{
920 struct platform_data *my_data = (struct platform_data *)platform;
921 struct audio_device *adev = my_data->adev;
922 audio_source_t source = (adev->active_input == NULL) ?
923 AUDIO_SOURCE_DEFAULT : adev->active_input->source;
924
925 audio_mode_t mode = adev->mode;
926 audio_devices_t in_device = ((adev->active_input == NULL) ?
927 AUDIO_DEVICE_NONE : adev->active_input->device)
928 & ~AUDIO_DEVICE_BIT_IN;
929 audio_channel_mask_t channel_mask = (adev->active_input == NULL) ?
930 AUDIO_CHANNEL_IN_MONO : adev->active_input->channel_mask;
931 snd_device_t snd_device = SND_DEVICE_NONE;
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700932 int channel_count = popcount(channel_mask);
Eric Laurentb23d5282013-05-14 15:27:20 -0700933
934 ALOGV("%s: enter: out_device(%#x) in_device(%#x)",
935 __func__, out_device, in_device);
936 if (mode == AUDIO_MODE_IN_CALL) {
937 if (out_device == AUDIO_DEVICE_NONE) {
938 ALOGE("%s: No output device set for voice call", __func__);
939 goto exit;
940 }
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700941 if (adev->voice.tty_mode != TTY_MODE_OFF) {
Eric Laurentb23d5282013-05-14 15:27:20 -0700942 if (out_device & AUDIO_DEVICE_OUT_WIRED_HEADPHONE ||
943 out_device & AUDIO_DEVICE_OUT_WIRED_HEADSET) {
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700944 switch (adev->voice.tty_mode) {
Eric Laurentb23d5282013-05-14 15:27:20 -0700945 case TTY_MODE_FULL:
946 snd_device = SND_DEVICE_IN_VOICE_TTY_FULL_HEADSET_MIC;
947 break;
948 case TTY_MODE_VCO:
949 snd_device = SND_DEVICE_IN_VOICE_TTY_VCO_HANDSET_MIC;
950 break;
951 case TTY_MODE_HCO:
952 snd_device = SND_DEVICE_IN_VOICE_TTY_HCO_HEADSET_MIC;
953 break;
954 default:
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700955 ALOGE("%s: Invalid TTY mode (%#x)",
956 __func__, adev->voice.tty_mode);
Eric Laurentb23d5282013-05-14 15:27:20 -0700957 }
958 goto exit;
959 }
960 }
961 if (out_device & AUDIO_DEVICE_OUT_EARPIECE ||
962 out_device & AUDIO_DEVICE_OUT_WIRED_HEADPHONE) {
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700963 if (out_device & AUDIO_DEVICE_OUT_EARPIECE &&
964 audio_extn_should_use_handset_anc(channel_count)) {
965 snd_device = SND_DEVICE_IN_AANC_HANDSET_MIC;
966 } else if (my_data->fluence_type == FLUENCE_NONE ||
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700967 my_data->fluence_in_voice_call == false) {
Eric Laurentb23d5282013-05-14 15:27:20 -0700968 snd_device = SND_DEVICE_IN_HANDSET_MIC;
969 } else {
Ravi Kumar Alamandaceb40822013-11-06 11:01:47 -0800970 snd_device = SND_DEVICE_IN_VOICE_DMIC;
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700971 adev->acdb_settings |= DMIC_FLAG;
Eric Laurentb23d5282013-05-14 15:27:20 -0700972 }
973 } else if (out_device & AUDIO_DEVICE_OUT_WIRED_HEADSET) {
974 snd_device = SND_DEVICE_IN_VOICE_HEADSET_MIC;
975 } else if (out_device & AUDIO_DEVICE_OUT_ALL_SCO) {
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700976 if (my_data->btsco_sample_rate == SAMPLE_RATE_16KHZ)
977 snd_device = SND_DEVICE_IN_BT_SCO_MIC_WB;
978 else
979 snd_device = SND_DEVICE_IN_BT_SCO_MIC;
Eric Laurentb23d5282013-05-14 15:27:20 -0700980 } else if (out_device & AUDIO_DEVICE_OUT_SPEAKER) {
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700981 if (my_data->fluence_type != FLUENCE_NONE &&
982 my_data->fluence_in_voice_call &&
983 my_data->fluence_in_spkr_mode) {
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800984 if(my_data->fluence_type & FLUENCE_QUAD_MIC) {
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700985 adev->acdb_settings |= QMIC_FLAG;
986 snd_device = SND_DEVICE_IN_VOICE_SPEAKER_QMIC;
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800987 } else {
988 adev->acdb_settings |= DMIC_FLAG;
989 snd_device = SND_DEVICE_IN_VOICE_SPEAKER_DMIC;
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700990 }
Eric Laurentb23d5282013-05-14 15:27:20 -0700991 } else {
992 snd_device = SND_DEVICE_IN_VOICE_SPEAKER_MIC;
993 }
994 }
995 } else if (source == AUDIO_SOURCE_CAMCORDER) {
996 if (in_device & AUDIO_DEVICE_IN_BUILTIN_MIC ||
997 in_device & AUDIO_DEVICE_IN_BACK_MIC) {
998 snd_device = SND_DEVICE_IN_CAMCORDER_MIC;
999 }
1000 } else if (source == AUDIO_SOURCE_VOICE_RECOGNITION) {
1001 if (in_device & AUDIO_DEVICE_IN_BUILTIN_MIC) {
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -08001002 if (channel_count == 2) {
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -08001003 snd_device = SND_DEVICE_IN_VOICE_REC_DMIC_STEREO;
Mingming Yin8e5a4f62013-10-07 15:23:41 -07001004 adev->acdb_settings |= DMIC_FLAG;
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -08001005 } else if (adev->active_input->enable_ns)
1006 snd_device = SND_DEVICE_IN_VOICE_REC_MIC_NS;
1007 else if (my_data->fluence_type != FLUENCE_NONE &&
1008 my_data->fluence_in_voice_rec) {
1009 snd_device = SND_DEVICE_IN_VOICE_REC_DMIC_FLUENCE;
1010 adev->acdb_settings |= DMIC_FLAG;
1011 } else {
1012 snd_device = SND_DEVICE_IN_VOICE_REC_MIC;
1013 }
Eric Laurentb23d5282013-05-14 15:27:20 -07001014 }
1015 } else if (source == AUDIO_SOURCE_VOICE_COMMUNICATION) {
1016 if (out_device & AUDIO_DEVICE_OUT_SPEAKER)
1017 in_device = AUDIO_DEVICE_IN_BACK_MIC;
1018 if (adev->active_input) {
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -08001019 if (adev->active_input->enable_aec &&
1020 adev->active_input->enable_ns) {
Eric Laurentb23d5282013-05-14 15:27:20 -07001021 if (in_device & AUDIO_DEVICE_IN_BACK_MIC) {
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -08001022 if (my_data->fluence_type & FLUENCE_DUAL_MIC &&
1023 my_data->fluence_in_spkr_mode) {
1024 snd_device = SND_DEVICE_IN_SPEAKER_DMIC_AEC_NS;
1025 adev->acdb_settings |= DMIC_FLAG;
1026 } else
1027 snd_device = SND_DEVICE_IN_SPEAKER_MIC_AEC_NS;
Eric Laurentb23d5282013-05-14 15:27:20 -07001028 } else if (in_device & AUDIO_DEVICE_IN_BUILTIN_MIC) {
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -08001029 if (my_data->fluence_type & FLUENCE_DUAL_MIC) {
1030 snd_device = SND_DEVICE_IN_HANDSET_DMIC_AEC_NS;
1031 adev->acdb_settings |= DMIC_FLAG;
1032 } else
1033 snd_device = SND_DEVICE_IN_HANDSET_MIC_AEC_NS;
Eric Laurentb23d5282013-05-14 15:27:20 -07001034 } else if (in_device & AUDIO_DEVICE_IN_WIRED_HEADSET) {
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -08001035 snd_device = SND_DEVICE_IN_HEADSET_MIC_FLUENCE;
Eric Laurentb23d5282013-05-14 15:27:20 -07001036 }
1037 set_echo_reference(adev->mixer, "SLIM_RX");
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -08001038 } else if (adev->active_input->enable_aec) {
1039 if (in_device & AUDIO_DEVICE_IN_BACK_MIC) {
1040 if (my_data->fluence_type & FLUENCE_DUAL_MIC) {
1041 snd_device = SND_DEVICE_IN_SPEAKER_DMIC_AEC;
1042 adev->acdb_settings |= DMIC_FLAG;
1043 } else
1044 snd_device = SND_DEVICE_IN_SPEAKER_MIC_AEC;
1045 } else if (in_device & AUDIO_DEVICE_IN_BUILTIN_MIC) {
1046 if (my_data->fluence_type & FLUENCE_DUAL_MIC) {
1047 snd_device = SND_DEVICE_IN_HANDSET_DMIC_AEC;
1048 adev->acdb_settings |= DMIC_FLAG;
1049 } else
1050 snd_device = SND_DEVICE_IN_HANDSET_MIC_AEC;
1051 } else if (in_device & AUDIO_DEVICE_IN_WIRED_HEADSET) {
1052 snd_device = SND_DEVICE_IN_HEADSET_MIC_FLUENCE;
1053 }
1054 set_echo_reference(adev->mixer, "SLIM_RX");
1055 } else if (adev->active_input->enable_ns) {
1056 if (in_device & AUDIO_DEVICE_IN_BACK_MIC) {
1057 if (my_data->fluence_type & FLUENCE_DUAL_MIC) {
1058 snd_device = SND_DEVICE_IN_SPEAKER_DMIC_NS;
1059 adev->acdb_settings |= DMIC_FLAG;
1060 } else
1061 snd_device = SND_DEVICE_IN_SPEAKER_MIC_NS;
1062 } else if (in_device & AUDIO_DEVICE_IN_BUILTIN_MIC) {
1063 if (my_data->fluence_type & FLUENCE_DUAL_MIC) {
1064 snd_device = SND_DEVICE_IN_HANDSET_DMIC_NS;
1065 adev->acdb_settings |= DMIC_FLAG;
1066 } else
1067 snd_device = SND_DEVICE_IN_HANDSET_MIC_NS;
1068 } else if (in_device & AUDIO_DEVICE_IN_WIRED_HEADSET) {
1069 snd_device = SND_DEVICE_IN_HEADSET_MIC_FLUENCE;
1070 }
1071 set_echo_reference(adev->mixer, "NONE");
Eric Laurentb23d5282013-05-14 15:27:20 -07001072 } else
1073 set_echo_reference(adev->mixer, "NONE");
1074 }
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -08001075 } else if (source == AUDIO_SOURCE_MIC) {
Apoorv Raghuvanshic0536542013-11-14 16:25:59 -08001076 if (in_device & AUDIO_DEVICE_IN_BUILTIN_MIC &&
1077 channel_count == 1 ) {
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -08001078 if(my_data->fluence_type & FLUENCE_DUAL_MIC &&
1079 my_data->fluence_in_audio_rec)
1080 snd_device = SND_DEVICE_IN_HANDSET_DMIC;
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -08001081 }
Mingming Yinab429782013-11-07 11:16:55 -08001082 } else if (source == AUDIO_SOURCE_FM_RX ||
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -08001083 source == AUDIO_SOURCE_FM_RX_A2DP) {
Preetam Singh Ranawatde84f1a2013-11-01 14:58:16 -07001084 snd_device = SND_DEVICE_IN_CAPTURE_FM;
Eric Laurentb23d5282013-05-14 15:27:20 -07001085 } else if (source == AUDIO_SOURCE_DEFAULT) {
1086 goto exit;
1087 }
1088
1089
1090 if (snd_device != SND_DEVICE_NONE) {
1091 goto exit;
1092 }
1093
1094 if (in_device != AUDIO_DEVICE_NONE &&
1095 !(in_device & AUDIO_DEVICE_IN_VOICE_CALL) &&
1096 !(in_device & AUDIO_DEVICE_IN_COMMUNICATION)) {
1097 if (in_device & AUDIO_DEVICE_IN_BUILTIN_MIC) {
Apoorv Raghuvanshi6178a3f2013-10-19 12:38:54 -07001098 if (audio_extn_ssr_get_enabled() && channel_count == 6)
1099 snd_device = SND_DEVICE_IN_QUAD_MIC;
Ravi Kumar Alamandafae42112013-11-07 23:31:54 -08001100 else if (channel_count == 2)
Apoorv Raghuvanshi6bd8dbf2013-10-19 18:37:52 -07001101 snd_device = SND_DEVICE_IN_HANDSET_STEREO_DMIC;
Apoorv Raghuvanshi6178a3f2013-10-19 12:38:54 -07001102 else
1103 snd_device = SND_DEVICE_IN_HANDSET_MIC;
Eric Laurentb23d5282013-05-14 15:27:20 -07001104 } else if (in_device & AUDIO_DEVICE_IN_BACK_MIC) {
1105 snd_device = SND_DEVICE_IN_SPEAKER_MIC;
1106 } else if (in_device & AUDIO_DEVICE_IN_WIRED_HEADSET) {
1107 snd_device = SND_DEVICE_IN_HEADSET_MIC;
1108 } else if (in_device & AUDIO_DEVICE_IN_BLUETOOTH_SCO_HEADSET) {
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001109 if (my_data->btsco_sample_rate == SAMPLE_RATE_16KHZ)
1110 snd_device = SND_DEVICE_IN_BT_SCO_MIC_WB;
1111 else
1112 snd_device = SND_DEVICE_IN_BT_SCO_MIC;
Eric Laurentb23d5282013-05-14 15:27:20 -07001113 } else if (in_device & AUDIO_DEVICE_IN_AUX_DIGITAL) {
1114 snd_device = SND_DEVICE_IN_HDMI_MIC;
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -07001115 } else if (in_device & AUDIO_DEVICE_IN_ANLG_DOCK_HEADSET ||
1116 in_device & AUDIO_DEVICE_IN_DGTL_DOCK_HEADSET) {
1117 snd_device = SND_DEVICE_IN_USB_HEADSET_MIC;
Apoorv Raghuvanshi6e262842013-10-06 14:39:35 -07001118 } else if (in_device & AUDIO_DEVICE_IN_FM_RX) {
1119 snd_device = SND_DEVICE_IN_CAPTURE_FM;
Eric Laurentb23d5282013-05-14 15:27:20 -07001120 } else {
1121 ALOGE("%s: Unknown input device(s) %#x", __func__, in_device);
1122 ALOGW("%s: Using default handset-mic", __func__);
1123 snd_device = SND_DEVICE_IN_HANDSET_MIC;
1124 }
1125 } else {
1126 if (out_device & AUDIO_DEVICE_OUT_EARPIECE) {
1127 snd_device = SND_DEVICE_IN_HANDSET_MIC;
1128 } else if (out_device & AUDIO_DEVICE_OUT_WIRED_HEADSET) {
1129 snd_device = SND_DEVICE_IN_HEADSET_MIC;
1130 } else if (out_device & AUDIO_DEVICE_OUT_SPEAKER) {
Apoorv Raghuvanshi6bd8dbf2013-10-19 18:37:52 -07001131 if (channel_count > 1)
1132 snd_device = SND_DEVICE_IN_SPEAKER_STEREO_DMIC;
1133 else
1134 snd_device = SND_DEVICE_IN_SPEAKER_MIC;
Eric Laurentb23d5282013-05-14 15:27:20 -07001135 } else if (out_device & AUDIO_DEVICE_OUT_WIRED_HEADPHONE) {
1136 snd_device = SND_DEVICE_IN_HANDSET_MIC;
1137 } else if (out_device & AUDIO_DEVICE_OUT_BLUETOOTH_SCO_HEADSET) {
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001138 if (my_data->btsco_sample_rate == SAMPLE_RATE_16KHZ)
1139 snd_device = SND_DEVICE_IN_BT_SCO_MIC_WB;
1140 else
1141 snd_device = SND_DEVICE_IN_BT_SCO_MIC;
Eric Laurentb23d5282013-05-14 15:27:20 -07001142 } else if (out_device & AUDIO_DEVICE_OUT_AUX_DIGITAL) {
1143 snd_device = SND_DEVICE_IN_HDMI_MIC;
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -07001144 } else if (out_device & AUDIO_DEVICE_OUT_ANLG_DOCK_HEADSET ||
1145 out_device & AUDIO_DEVICE_OUT_DGTL_DOCK_HEADSET) {
1146 snd_device = SND_DEVICE_IN_USB_HEADSET_MIC;
Eric Laurentb23d5282013-05-14 15:27:20 -07001147 } else {
1148 ALOGE("%s: Unknown output device(s) %#x", __func__, out_device);
1149 ALOGW("%s: Using default handset-mic", __func__);
1150 snd_device = SND_DEVICE_IN_HANDSET_MIC;
1151 }
1152 }
1153exit:
1154 ALOGV("%s: exit: in_snd_device(%s)", __func__, device_table[snd_device]);
1155 return snd_device;
1156}
1157
1158int platform_set_hdmi_channels(void *platform, int channel_count)
1159{
1160 struct platform_data *my_data = (struct platform_data *)platform;
1161 struct audio_device *adev = my_data->adev;
1162 struct mixer_ctl *ctl;
1163 const char *channel_cnt_str = NULL;
1164 const char *mixer_ctl_name = "HDMI_RX Channels";
1165 switch (channel_count) {
1166 case 8:
1167 channel_cnt_str = "Eight"; break;
1168 case 7:
1169 channel_cnt_str = "Seven"; break;
1170 case 6:
1171 channel_cnt_str = "Six"; break;
1172 case 5:
1173 channel_cnt_str = "Five"; break;
1174 case 4:
1175 channel_cnt_str = "Four"; break;
1176 case 3:
1177 channel_cnt_str = "Three"; break;
1178 default:
1179 channel_cnt_str = "Two"; break;
1180 }
1181 ctl = mixer_get_ctl_by_name(adev->mixer, mixer_ctl_name);
1182 if (!ctl) {
1183 ALOGE("%s: Could not get ctl for mixer cmd - %s",
1184 __func__, mixer_ctl_name);
1185 return -EINVAL;
1186 }
1187 ALOGV("HDMI channel count: %s", channel_cnt_str);
1188 mixer_ctl_set_enum_by_string(ctl, channel_cnt_str);
1189 return 0;
1190}
1191
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001192int platform_edid_get_max_channels(void *platform)
Eric Laurentb23d5282013-05-14 15:27:20 -07001193{
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001194 struct platform_data *my_data = (struct platform_data *)platform;
1195 struct audio_device *adev = my_data->adev;
Eric Laurentb23d5282013-05-14 15:27:20 -07001196 char block[MAX_SAD_BLOCKS * SAD_BLOCK_SIZE];
1197 char *sad = block;
1198 int num_audio_blocks;
1199 int channel_count;
1200 int max_channels = 0;
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001201 int i, ret, count;
Eric Laurentb23d5282013-05-14 15:27:20 -07001202
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001203 struct mixer_ctl *ctl;
1204
1205 ctl = mixer_get_ctl_by_name(adev->mixer, AUDIO_DATA_BLOCK_MIXER_CTL);
1206 if (!ctl) {
1207 ALOGE("%s: Could not get ctl for mixer cmd - %s",
1208 __func__, AUDIO_DATA_BLOCK_MIXER_CTL);
Eric Laurentb23d5282013-05-14 15:27:20 -07001209 return 0;
1210 }
1211
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001212 mixer_ctl_update(ctl);
1213
1214 count = mixer_ctl_get_num_values(ctl);
Eric Laurentb23d5282013-05-14 15:27:20 -07001215
1216 /* Read SAD blocks, clamping the maximum size for safety */
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001217 if (count > (int)sizeof(block))
1218 count = (int)sizeof(block);
Eric Laurentb23d5282013-05-14 15:27:20 -07001219
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001220 ret = mixer_ctl_get_array(ctl, block, count);
1221 if (ret != 0) {
1222 ALOGE("%s: mixer_ctl_get_array() failed to get EDID info", __func__);
1223 return 0;
1224 }
Eric Laurentb23d5282013-05-14 15:27:20 -07001225
1226 /* Calculate the number of SAD blocks */
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001227 num_audio_blocks = count / SAD_BLOCK_SIZE;
Eric Laurentb23d5282013-05-14 15:27:20 -07001228
1229 for (i = 0; i < num_audio_blocks; i++) {
1230 /* Only consider LPCM blocks */
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001231 if ((sad[0] >> 3) != EDID_FORMAT_LPCM) {
1232 sad += 3;
Eric Laurentb23d5282013-05-14 15:27:20 -07001233 continue;
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001234 }
Eric Laurentb23d5282013-05-14 15:27:20 -07001235
1236 channel_count = (sad[0] & 0x7) + 1;
1237 if (channel_count > max_channels)
1238 max_channels = channel_count;
1239
1240 /* Advance to next block */
1241 sad += 3;
1242 }
1243
1244 return max_channels;
1245}
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001246
1247static int platform_set_slowtalk(struct platform_data *my_data, bool state)
1248{
1249 int ret = 0;
1250 struct audio_device *adev = my_data->adev;
1251 struct mixer_ctl *ctl;
1252 const char *mixer_ctl_name = "Slowtalk Enable";
1253 uint32_t set_values[ ] = {0,
1254 ALL_SESSION_VSID};
1255
1256 set_values[0] = state;
1257 ctl = mixer_get_ctl_by_name(adev->mixer, mixer_ctl_name);
1258 if (!ctl) {
1259 ALOGE("%s: Could not get ctl for mixer cmd - %s",
1260 __func__, mixer_ctl_name);
1261 ret = -EINVAL;
1262 } else {
1263 ALOGV("Setting slowtalk state: %d", state);
1264 ret = mixer_ctl_set_array(ctl, set_values, ARRAY_SIZE(set_values));
1265 my_data->slowtalk = state;
1266 }
1267
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -08001268 if (my_data->csd != NULL) {
1269 ret = my_data->csd->slow_talk(ALL_SESSION_VSID, state);
1270 if (ret < 0) {
1271 ALOGE("%s: csd_client_disable_device, failed, error %d",
1272 __func__, ret);
1273 }
1274 }
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001275 return ret;
1276}
1277
1278int platform_set_parameters(void *platform, struct str_parms *parms)
1279{
1280 struct platform_data *my_data = (struct platform_data *)platform;
1281 char *str;
1282 char value[32];
1283 int val;
1284 int ret = 0;
1285
1286 ALOGV("%s: enter: %s", __func__, str_parms_to_str(parms));
1287
1288 ret = str_parms_get_int(parms, AUDIO_PARAMETER_KEY_BTSCO, &val);
1289 if (ret >= 0) {
1290 str_parms_del(parms, AUDIO_PARAMETER_KEY_BTSCO);
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001291 my_data->btsco_sample_rate = val;
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001292 }
1293
1294 ret = str_parms_get_int(parms, AUDIO_PARAMETER_KEY_SLOWTALK, &val);
1295 if (ret >= 0) {
1296 str_parms_del(parms, AUDIO_PARAMETER_KEY_SLOWTALK);
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001297 ret = platform_set_slowtalk(my_data, val);
1298 if (ret)
1299 ALOGE("%s: Failed to set slow talk err: %d", __func__, ret);
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001300 }
1301
1302 ALOGV("%s: exit with code(%d)", __func__, ret);
1303 return ret;
1304}
1305
Shiv Maliyappanahallida107642013-10-17 11:16:13 -07001306int platform_set_incall_recoding_session_id(void *platform,
1307 uint32_t session_id)
1308{
1309 int ret = 0;
1310 struct platform_data *my_data = (struct platform_data *)platform;
1311 struct audio_device *adev = my_data->adev;
1312 struct mixer_ctl *ctl;
1313 const char *mixer_ctl_name = "Voc VSID";
1314 int num_ctl_values;
1315 int i;
1316
1317 ctl = mixer_get_ctl_by_name(adev->mixer, mixer_ctl_name);
1318 if (!ctl) {
1319 ALOGE("%s: Could not get ctl for mixer cmd - %s",
1320 __func__, mixer_ctl_name);
1321 ret = -EINVAL;
1322 } else {
1323 num_ctl_values = mixer_ctl_get_num_values(ctl);
1324 for (i = 0; i < num_ctl_values; i++) {
1325 if (mixer_ctl_set_value(ctl, i, session_id)) {
1326 ALOGV("Error: invalid session_id: %x", session_id);
1327 ret = -EINVAL;
1328 break;
1329 }
1330 }
1331 }
1332
1333 return ret;
1334}
1335
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001336void platform_get_parameters(void *platform,
1337 struct str_parms *query,
1338 struct str_parms *reply)
1339{
1340 struct platform_data *my_data = (struct platform_data *)platform;
1341 char *str = NULL;
1342 char value[256] = {0};
1343 int ret;
1344 int fluence_type;
1345
1346 ret = str_parms_get_str(query, AUDIO_PARAMETER_KEY_FLUENCE_TYPE,
1347 value, sizeof(value));
1348 if (ret >= 0) {
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -08001349 if (my_data->fluence_type & FLUENCE_QUAD_MIC) {
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001350 strlcpy(value, "fluencepro", sizeof(value));
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -08001351 } else if (my_data->fluence_type & FLUENCE_DUAL_MIC) {
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001352 strlcpy(value, "fluence", sizeof(value));
1353 } else {
1354 strlcpy(value, "none", sizeof(value));
1355 }
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001356
1357 str_parms_add_str(reply, AUDIO_PARAMETER_KEY_FLUENCE_TYPE, value);
1358 }
1359
1360 ALOGV("%s: exit: returns - %s", __func__, str_parms_to_str(reply));
1361}
1362
Haynes Mathew George7ff216f2013-09-11 19:51:41 -07001363/* Delay in Us */
1364int64_t platform_render_latency(audio_usecase_t usecase)
1365{
1366 switch (usecase) {
1367 case USECASE_AUDIO_PLAYBACK_DEEP_BUFFER:
1368 return DEEP_BUFFER_PLATFORM_DELAY;
1369 case USECASE_AUDIO_PLAYBACK_LOW_LATENCY:
1370 return LOW_LATENCY_PLATFORM_DELAY;
1371 default:
1372 return 0;
1373 }
1374}
Preetam Singh Ranawatde84f1a2013-11-01 14:58:16 -07001375
1376int platform_get_usecase_from_source(int source)
1377{
Mingming Yinab429782013-11-07 11:16:55 -08001378 ALOGV("%s: input source :%d", __func__, source);
1379 if(source == AUDIO_SOURCE_FM_RX_A2DP)
Preetam Singh Ranawatde84f1a2013-11-01 14:58:16 -07001380 return USECASE_AUDIO_RECORD_FM_VIRTUAL;
1381 else
1382 return USECASE_AUDIO_RECORD;
1383}