blob: d562a0c7ec7bac4ca6a4e21638a651f40b978180 [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
Apoorv Raghuvanshi84fa2fe2013-12-04 11:57:47 -080056#define MAX_SND_CARD 8
sangwoo1b9f4b32013-06-21 18:22:55 -070057
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -070058#define SAMPLE_RATE_8KHZ 8000
59#define SAMPLE_RATE_16KHZ 16000
60
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -070061#define AUDIO_PARAMETER_KEY_FLUENCE_TYPE "fluence"
62#define AUDIO_PARAMETER_KEY_BTSCO "bt_samplerate"
63#define AUDIO_PARAMETER_KEY_SLOWTALK "st_enable"
Ben Rombergerc1dc70d2013-12-19 15:11:17 -080064#define AUDIO_PARAMETER_KEY_VOLUME_BOOST "volume_boost"
65
66enum {
67 VOICE_FEATURE_SET_DEFAULT,
68 VOICE_FEATURE_SET_VOLUME_BOOST
69};
sangwoo53b2cf02013-07-25 19:18:44 -070070
Eric Laurentb23d5282013-05-14 15:27:20 -070071struct audio_block_header
72{
73 int reserved;
74 int length;
75};
76
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -080077/* Audio calibration related functions */
Eric Laurentb23d5282013-05-14 15:27:20 -070078typedef void (*acdb_deallocate_t)();
79typedef int (*acdb_init_t)();
80typedef void (*acdb_send_audio_cal_t)(int, int);
81typedef void (*acdb_send_voice_cal_t)(int, int);
Ben Rombergerc1dc70d2013-12-19 15:11:17 -080082typedef int (*acdb_reload_vocvoltable_t)(int);
Eric Laurentb23d5282013-05-14 15:27:20 -070083
Eric Laurentb23d5282013-05-14 15:27:20 -070084struct platform_data {
85 struct audio_device *adev;
86 bool fluence_in_spkr_mode;
87 bool fluence_in_voice_call;
88 bool fluence_in_voice_rec;
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -080089 bool fluence_in_audio_rec;
Mingming Yin8e5a4f62013-10-07 15:23:41 -070090 int fluence_type;
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -070091 int btsco_sample_rate;
92 bool slowtalk;
Ravi Kumar Alamanda48c921d2013-10-29 06:07:44 -070093 /* Audio calibration related functions */
Ben Rombergerc1dc70d2013-12-19 15:11:17 -080094 void *acdb_handle;
95 int voice_feature_set;
96 acdb_init_t acdb_init;
97 acdb_deallocate_t acdb_deallocate;
98 acdb_send_audio_cal_t acdb_send_audio_cal;
99 acdb_send_voice_cal_t acdb_send_voice_cal;
100 acdb_reload_vocvoltable_t acdb_reload_vocvoltable;
Ravi Kumar Alamanda48c921d2013-10-29 06:07:44 -0700101
102 void *hw_info;
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800103 struct csd_data *csd;
Eric Laurentb23d5282013-05-14 15:27:20 -0700104};
105
106static const int pcm_device_table[AUDIO_USECASE_MAX][2] = {
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700107 [USECASE_AUDIO_PLAYBACK_DEEP_BUFFER] = {DEEP_BUFFER_PCM_DEVICE,
108 DEEP_BUFFER_PCM_DEVICE},
109 [USECASE_AUDIO_PLAYBACK_LOW_LATENCY] = {LOWLATENCY_PCM_DEVICE,
Preetam Singh Ranawatde84f1a2013-11-01 14:58:16 -0700110 LOWLATENCY_PCM_DEVICE},
111 [USECASE_AUDIO_PLAYBACK_MULTI_CH] = {MULTIMEDIA2_PCM_DEVICE,
112 MULTIMEDIA2_PCM_DEVICE},
Krishnankutty Kolathappillya43f96e2013-11-01 12:17:53 -0700113 [USECASE_AUDIO_PLAYBACK_OFFLOAD] =
114 {PLAYBACK_OFFLOAD_DEVICE, PLAYBACK_OFFLOAD_DEVICE},
Shiv Maliyappanahallida107642013-10-17 11:16:13 -0700115 [USECASE_AUDIO_RECORD] = {AUDIO_RECORD_PCM_DEVICE, AUDIO_RECORD_PCM_DEVICE},
Mingming Yine62d7842013-10-25 16:26:03 -0700116 [USECASE_AUDIO_RECORD_COMPRESS] = {COMPRESS_CAPTURE_DEVICE, COMPRESS_CAPTURE_DEVICE},
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700117 [USECASE_AUDIO_RECORD_LOW_LATENCY] = {LOWLATENCY_PCM_DEVICE,
118 LOWLATENCY_PCM_DEVICE},
Preetam Singh Ranawatde84f1a2013-11-01 14:58:16 -0700119 [USECASE_AUDIO_RECORD_FM_VIRTUAL] = {MULTIMEDIA2_PCM_DEVICE,
120 MULTIMEDIA2_PCM_DEVICE},
Apoorv Raghuvanshi6e262842013-10-06 14:39:35 -0700121 [USECASE_AUDIO_PLAYBACK_FM] = {FM_PLAYBACK_PCM_DEVICE, FM_CAPTURE_PCM_DEVICE},
Vimal Puthanveed5b4d3f12013-11-05 15:57:39 -0800122 [USECASE_AUDIO_HFP_SCO] = {HFP_PCM_RX, HFP_SCO_RX},
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700123 [USECASE_VOICE_CALL] = {VOICE_CALL_PCM_DEVICE, VOICE_CALL_PCM_DEVICE},
124 [USECASE_VOICE2_CALL] = {VOICE2_CALL_PCM_DEVICE, VOICE2_CALL_PCM_DEVICE},
125 [USECASE_VOLTE_CALL] = {VOLTE_CALL_PCM_DEVICE, VOLTE_CALL_PCM_DEVICE},
126 [USECASE_QCHAT_CALL] = {QCHAT_CALL_PCM_DEVICE, QCHAT_CALL_PCM_DEVICE},
Narsinga Rao Chella05573b72013-11-15 15:21:40 -0800127 [USECASE_COMPRESS_VOIP_CALL] = {COMPRESS_VOIP_CALL_PCM_DEVICE, COMPRESS_VOIP_CALL_PCM_DEVICE},
Shiv Maliyappanahallida107642013-10-17 11:16:13 -0700128 [USECASE_INCALL_REC_UPLINK] = {AUDIO_RECORD_PCM_DEVICE,
129 AUDIO_RECORD_PCM_DEVICE},
130 [USECASE_INCALL_REC_DOWNLINK] = {AUDIO_RECORD_PCM_DEVICE,
131 AUDIO_RECORD_PCM_DEVICE},
132 [USECASE_INCALL_REC_UPLINK_AND_DOWNLINK] = {AUDIO_RECORD_PCM_DEVICE,
133 AUDIO_RECORD_PCM_DEVICE},
Helen Zenge56b4852013-12-03 16:54:40 -0800134 [USECASE_INCALL_REC_UPLINK_COMPRESS] = {COMPRESS_CAPTURE_DEVICE,
135 COMPRESS_CAPTURE_DEVICE},
136 [USECASE_INCALL_REC_DOWNLINK_COMPRESS] = {COMPRESS_CAPTURE_DEVICE,
137 COMPRESS_CAPTURE_DEVICE},
138 [USECASE_INCALL_REC_UPLINK_AND_DOWNLINK_COMPRESS] = {COMPRESS_CAPTURE_DEVICE,
139 COMPRESS_CAPTURE_DEVICE},
Shiv Maliyappanahallif3b9a422013-10-22 16:38:08 -0700140 [USECASE_INCALL_MUSIC_UPLINK] = {INCALL_MUSIC_UPLINK_PCM_DEVICE,
141 INCALL_MUSIC_UPLINK_PCM_DEVICE},
142 [USECASE_INCALL_MUSIC_UPLINK2] = {INCALL_MUSIC_UPLINK2_PCM_DEVICE,
143 INCALL_MUSIC_UPLINK2_PCM_DEVICE},
Gopikrishnaiah Anandanf538cef2013-10-28 14:06:03 -0700144 [USECASE_AUDIO_SPKR_CALIB_RX] = {SPKR_PROT_CALIB_RX_PCM_DEVICE, -1},
145 [USECASE_AUDIO_SPKR_CALIB_TX] = {-1, SPKR_PROT_CALIB_TX_PCM_DEVICE},
Eric Laurentb23d5282013-05-14 15:27:20 -0700146};
147
148/* Array to store sound devices */
149static const char * const device_table[SND_DEVICE_MAX] = {
150 [SND_DEVICE_NONE] = "none",
151 /* Playback sound devices */
152 [SND_DEVICE_OUT_HANDSET] = "handset",
153 [SND_DEVICE_OUT_SPEAKER] = "speaker",
154 [SND_DEVICE_OUT_SPEAKER_REVERSE] = "speaker-reverse",
155 [SND_DEVICE_OUT_HEADPHONES] = "headphones",
156 [SND_DEVICE_OUT_SPEAKER_AND_HEADPHONES] = "speaker-and-headphones",
157 [SND_DEVICE_OUT_VOICE_HANDSET] = "voice-handset",
158 [SND_DEVICE_OUT_VOICE_SPEAKER] = "voice-speaker",
159 [SND_DEVICE_OUT_VOICE_HEADPHONES] = "voice-headphones",
160 [SND_DEVICE_OUT_HDMI] = "hdmi",
161 [SND_DEVICE_OUT_SPEAKER_AND_HDMI] = "speaker-and-hdmi",
162 [SND_DEVICE_OUT_BT_SCO] = "bt-sco-headset",
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700163 [SND_DEVICE_OUT_BT_SCO_WB] = "bt-sco-headset-wb",
Eric Laurentb23d5282013-05-14 15:27:20 -0700164 [SND_DEVICE_OUT_VOICE_TTY_FULL_HEADPHONES] = "voice-tty-full-headphones",
165 [SND_DEVICE_OUT_VOICE_TTY_VCO_HEADPHONES] = "voice-tty-vco-headphones",
166 [SND_DEVICE_OUT_VOICE_TTY_HCO_HANDSET] = "voice-tty-hco-handset",
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700167 [SND_DEVICE_OUT_AFE_PROXY] = "afe-proxy",
168 [SND_DEVICE_OUT_USB_HEADSET] = "usb-headphones",
169 [SND_DEVICE_OUT_SPEAKER_AND_USB_HEADSET] = "speaker-and-usb-headphones",
Apoorv Raghuvanshi6e262842013-10-06 14:39:35 -0700170 [SND_DEVICE_OUT_TRANSMISSION_FM] = "transmission-fm",
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700171 [SND_DEVICE_OUT_ANC_HEADSET] = "anc-headphones",
172 [SND_DEVICE_OUT_ANC_FB_HEADSET] = "anc-fb-headphones",
173 [SND_DEVICE_OUT_VOICE_ANC_HEADSET] = "voice-anc-headphones",
174 [SND_DEVICE_OUT_VOICE_ANC_FB_HEADSET] = "voice-anc-fb-headphones",
175 [SND_DEVICE_OUT_SPEAKER_AND_ANC_HEADSET] = "speaker-and-anc-headphones",
176 [SND_DEVICE_OUT_ANC_HANDSET] = "anc-handset",
Gopikrishnaiah Anandanf538cef2013-10-28 14:06:03 -0700177 [SND_DEVICE_OUT_SPEAKER_PROTECTED] = "speaker-protected",
Eric Laurentb23d5282013-05-14 15:27:20 -0700178
179 /* Capture sound devices */
180 [SND_DEVICE_IN_HANDSET_MIC] = "handset-mic",
Eric Laurentb23d5282013-05-14 15:27:20 -0700181 [SND_DEVICE_IN_HANDSET_MIC_AEC] = "handset-mic",
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800182 [SND_DEVICE_IN_HANDSET_MIC_NS] = "handset-mic",
183 [SND_DEVICE_IN_HANDSET_MIC_AEC_NS] = "handset-mic",
184 [SND_DEVICE_IN_HANDSET_DMIC] = "dmic-endfire",
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800185 [SND_DEVICE_IN_HANDSET_DMIC_AEC] = "dmic-endfire",
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800186 [SND_DEVICE_IN_HANDSET_DMIC_NS] = "dmic-endfire",
187 [SND_DEVICE_IN_HANDSET_DMIC_AEC_NS] = "dmic-endfire",
188 [SND_DEVICE_IN_SPEAKER_MIC] = "speaker-mic",
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800189 [SND_DEVICE_IN_SPEAKER_MIC_AEC] = "speaker-mic",
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800190 [SND_DEVICE_IN_SPEAKER_MIC_NS] = "speaker-mic",
191 [SND_DEVICE_IN_SPEAKER_MIC_AEC_NS] = "speaker-mic",
192 [SND_DEVICE_IN_SPEAKER_DMIC] = "speaker-dmic-endfire",
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800193 [SND_DEVICE_IN_SPEAKER_DMIC_AEC] = "speaker-dmic-endfire",
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800194 [SND_DEVICE_IN_SPEAKER_DMIC_NS] = "speaker-dmic-endfire",
195 [SND_DEVICE_IN_SPEAKER_DMIC_AEC_NS] = "speaker-dmic-endfire",
196 [SND_DEVICE_IN_HEADSET_MIC] = "headset-mic",
197 [SND_DEVICE_IN_HEADSET_MIC_FLUENCE] = "headset-mic",
Eric Laurentb23d5282013-05-14 15:27:20 -0700198 [SND_DEVICE_IN_VOICE_SPEAKER_MIC] = "voice-speaker-mic",
199 [SND_DEVICE_IN_VOICE_HEADSET_MIC] = "voice-headset-mic",
200 [SND_DEVICE_IN_HDMI_MIC] = "hdmi-mic",
201 [SND_DEVICE_IN_BT_SCO_MIC] = "bt-sco-mic",
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700202 [SND_DEVICE_IN_BT_SCO_MIC_WB] = "bt-sco-mic-wb",
Eric Laurentb23d5282013-05-14 15:27:20 -0700203 [SND_DEVICE_IN_CAMCORDER_MIC] = "camcorder-mic",
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700204 [SND_DEVICE_IN_VOICE_DMIC] = "voice-dmic-ef",
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700205 [SND_DEVICE_IN_VOICE_SPEAKER_DMIC] = "voice-speaker-dmic-ef",
Eric Laurentb23d5282013-05-14 15:27:20 -0700206 [SND_DEVICE_IN_VOICE_TTY_FULL_HEADSET_MIC] = "voice-tty-full-headset-mic",
207 [SND_DEVICE_IN_VOICE_TTY_VCO_HANDSET_MIC] = "voice-tty-vco-handset-mic",
208 [SND_DEVICE_IN_VOICE_TTY_HCO_HEADSET_MIC] = "voice-tty-hco-headset-mic",
209 [SND_DEVICE_IN_VOICE_REC_MIC] = "voice-rec-mic",
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800210 [SND_DEVICE_IN_VOICE_REC_MIC_NS] = "voice-rec-mic",
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800211 [SND_DEVICE_IN_VOICE_REC_DMIC_STEREO] = "voice-rec-dmic-ef",
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700212 [SND_DEVICE_IN_VOICE_REC_DMIC_FLUENCE] = "voice-rec-dmic-ef-fluence",
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700213 [SND_DEVICE_IN_USB_HEADSET_MIC] = "usb-headset-mic",
Apoorv Raghuvanshi6e262842013-10-06 14:39:35 -0700214 [SND_DEVICE_IN_CAPTURE_FM] = "capture-fm",
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700215 [SND_DEVICE_IN_AANC_HANDSET_MIC] = "aanc-handset-mic",
Apoorv Raghuvanshi6178a3f2013-10-19 12:38:54 -0700216 [SND_DEVICE_IN_QUAD_MIC] = "quad-mic",
Apoorv Raghuvanshi6bd8dbf2013-10-19 18:37:52 -0700217 [SND_DEVICE_IN_HANDSET_STEREO_DMIC] = "handset-stereo-dmic-ef",
218 [SND_DEVICE_IN_SPEAKER_STEREO_DMIC] = "speaker-stereo-dmic-ef",
Gopikrishnaiah Anandanf538cef2013-10-28 14:06:03 -0700219 [SND_DEVICE_IN_CAPTURE_VI_FEEDBACK] = "vi-feedback",
Eric Laurentb23d5282013-05-14 15:27:20 -0700220};
221
222/* ACDB IDs (audio DSP path configuration IDs) for each sound device */
223static const int acdb_device_table[SND_DEVICE_MAX] = {
224 [SND_DEVICE_NONE] = -1,
225 [SND_DEVICE_OUT_HANDSET] = 7,
Vidyakumar Athotac29d4ab2013-11-14 16:58:02 -0800226 [SND_DEVICE_OUT_SPEAKER] = 14,
227 [SND_DEVICE_OUT_SPEAKER_REVERSE] = 14,
Eric Laurentb23d5282013-05-14 15:27:20 -0700228 [SND_DEVICE_OUT_HEADPHONES] = 10,
229 [SND_DEVICE_OUT_SPEAKER_AND_HEADPHONES] = 10,
230 [SND_DEVICE_OUT_VOICE_HANDSET] = 7,
Vidyakumar Athotac29d4ab2013-11-14 16:58:02 -0800231 [SND_DEVICE_OUT_VOICE_SPEAKER] = 14,
Eric Laurentb23d5282013-05-14 15:27:20 -0700232 [SND_DEVICE_OUT_VOICE_HEADPHONES] = 10,
233 [SND_DEVICE_OUT_HDMI] = 18,
Vidyakumar Athotac29d4ab2013-11-14 16:58:02 -0800234 [SND_DEVICE_OUT_SPEAKER_AND_HDMI] = 14,
Eric Laurentb23d5282013-05-14 15:27:20 -0700235 [SND_DEVICE_OUT_BT_SCO] = 22,
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700236 [SND_DEVICE_OUT_BT_SCO_WB] = 39,
Eric Laurentb23d5282013-05-14 15:27:20 -0700237 [SND_DEVICE_OUT_VOICE_TTY_FULL_HEADPHONES] = 17,
238 [SND_DEVICE_OUT_VOICE_TTY_VCO_HEADPHONES] = 17,
239 [SND_DEVICE_OUT_VOICE_TTY_HCO_HANDSET] = 37,
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700240 [SND_DEVICE_OUT_AFE_PROXY] = 0,
241 [SND_DEVICE_OUT_USB_HEADSET] = 0,
242 [SND_DEVICE_OUT_SPEAKER_AND_USB_HEADSET] = 14,
Apoorv Raghuvanshi6e262842013-10-06 14:39:35 -0700243 [SND_DEVICE_OUT_TRANSMISSION_FM] = 0,
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700244 [SND_DEVICE_OUT_ANC_HEADSET] = 26,
Ravi Kumar Alamanda48c921d2013-10-29 06:07:44 -0700245 [SND_DEVICE_OUT_ANC_FB_HEADSET] = 27,
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700246 [SND_DEVICE_OUT_VOICE_ANC_HEADSET] = 26,
Ravi Kumar Alamanda48c921d2013-10-29 06:07:44 -0700247 [SND_DEVICE_OUT_VOICE_ANC_FB_HEADSET] = 27,
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700248 [SND_DEVICE_OUT_SPEAKER_AND_ANC_HEADSET] = 26,
249 [SND_DEVICE_OUT_ANC_HANDSET] = 103,
Gopikrishnaiah Anandanf538cef2013-10-28 14:06:03 -0700250 [SND_DEVICE_OUT_SPEAKER_PROTECTED] = 101,
Eric Laurentb23d5282013-05-14 15:27:20 -0700251
252 [SND_DEVICE_IN_HANDSET_MIC] = 4,
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800253 [SND_DEVICE_IN_HANDSET_MIC_AEC] = 106,
254 [SND_DEVICE_IN_HANDSET_MIC_NS] = 107,
255 [SND_DEVICE_IN_HANDSET_MIC_AEC_NS] = 108,
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800256 [SND_DEVICE_IN_HANDSET_DMIC] = 41,
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800257 [SND_DEVICE_IN_HANDSET_DMIC_AEC] = 109,
258 [SND_DEVICE_IN_HANDSET_DMIC_NS] = 110,
259 [SND_DEVICE_IN_HANDSET_DMIC_AEC_NS] = 111,
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800260 [SND_DEVICE_IN_SPEAKER_MIC] = 11,
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800261 [SND_DEVICE_IN_SPEAKER_MIC_AEC] = 112,
262 [SND_DEVICE_IN_SPEAKER_MIC_NS] = 113,
263 [SND_DEVICE_IN_SPEAKER_MIC_AEC_NS] = 114,
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800264 [SND_DEVICE_IN_SPEAKER_DMIC] = 43,
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800265 [SND_DEVICE_IN_SPEAKER_DMIC_AEC] = 115,
266 [SND_DEVICE_IN_SPEAKER_DMIC_NS] = 116,
267 [SND_DEVICE_IN_SPEAKER_DMIC_AEC_NS] = 117,
Eric Laurentb23d5282013-05-14 15:27:20 -0700268 [SND_DEVICE_IN_HEADSET_MIC] = 8,
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800269 [SND_DEVICE_IN_HEADSET_MIC_FLUENCE] = 47,
Eric Laurentb23d5282013-05-14 15:27:20 -0700270 [SND_DEVICE_IN_VOICE_SPEAKER_MIC] = 11,
271 [SND_DEVICE_IN_VOICE_HEADSET_MIC] = 8,
272 [SND_DEVICE_IN_HDMI_MIC] = 4,
273 [SND_DEVICE_IN_BT_SCO_MIC] = 21,
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700274 [SND_DEVICE_IN_BT_SCO_MIC_WB] = 38,
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800275 [SND_DEVICE_IN_CAMCORDER_MIC] = 4,
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700276 [SND_DEVICE_IN_VOICE_DMIC] = 41,
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700277 [SND_DEVICE_IN_VOICE_SPEAKER_DMIC] = 43,
Eric Laurentb23d5282013-05-14 15:27:20 -0700278 [SND_DEVICE_IN_VOICE_TTY_FULL_HEADSET_MIC] = 16,
279 [SND_DEVICE_IN_VOICE_TTY_VCO_HANDSET_MIC] = 36,
280 [SND_DEVICE_IN_VOICE_TTY_HCO_HEADSET_MIC] = 16,
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800281 [SND_DEVICE_IN_VOICE_REC_MIC] = 4,
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -0800282 [SND_DEVICE_IN_VOICE_REC_MIC_NS] = 107,
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800283 [SND_DEVICE_IN_VOICE_REC_DMIC_STEREO] = 34,
284 [SND_DEVICE_IN_VOICE_REC_DMIC_FLUENCE] = 41,
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700285 [SND_DEVICE_IN_USB_HEADSET_MIC] = 44,
Apoorv Raghuvanshi6e262842013-10-06 14:39:35 -0700286 [SND_DEVICE_IN_CAPTURE_FM] = 0,
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700287 [SND_DEVICE_IN_AANC_HANDSET_MIC] = 104,
Apoorv Raghuvanshi6178a3f2013-10-19 12:38:54 -0700288 [SND_DEVICE_IN_QUAD_MIC] = 46,
Apoorv Raghuvanshi6bd8dbf2013-10-19 18:37:52 -0700289 [SND_DEVICE_IN_HANDSET_STEREO_DMIC] = 34,
290 [SND_DEVICE_IN_SPEAKER_STEREO_DMIC] = 35,
Gopikrishnaiah Anandanf538cef2013-10-28 14:06:03 -0700291 [SND_DEVICE_IN_CAPTURE_VI_FEEDBACK] = 102,
Eric Laurentb23d5282013-05-14 15:27:20 -0700292};
293
Haynes Mathew George7ff216f2013-09-11 19:51:41 -0700294#define DEEP_BUFFER_PLATFORM_DELAY (29*1000LL)
295#define LOW_LATENCY_PLATFORM_DELAY (13*1000LL)
296
Eric Laurentb23d5282013-05-14 15:27:20 -0700297static int set_echo_reference(struct mixer *mixer, const char* ec_ref)
298{
299 struct mixer_ctl *ctl;
300 const char *mixer_ctl_name = "EC_REF_RX";
301
302 ctl = mixer_get_ctl_by_name(mixer, mixer_ctl_name);
303 if (!ctl) {
304 ALOGE("%s: Could not get ctl for mixer cmd - %s",
305 __func__, mixer_ctl_name);
306 return -EINVAL;
307 }
308 ALOGV("Setting EC Reference: %s", ec_ref);
309 mixer_ctl_set_enum_by_string(ctl, ec_ref);
310 return 0;
311}
312
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800313static struct csd_data *open_csd_client()
314{
315 struct csd_data *csd = calloc(1, sizeof(struct csd_data));
316
317 csd->csd_client = dlopen(LIB_CSD_CLIENT, RTLD_NOW);
318 if (csd->csd_client == NULL) {
319 ALOGE("%s: DLOPEN failed for %s", __func__, LIB_CSD_CLIENT);
320 goto error;
321 } else {
322 ALOGV("%s: DLOPEN successful for %s", __func__, LIB_CSD_CLIENT);
323
324 csd->deinit = (deinit_t)dlsym(csd->csd_client,
325 "csd_client_deinit");
326 if (csd->deinit == NULL) {
327 ALOGE("%s: dlsym error %s for csd_client_deinit", __func__,
328 dlerror());
329 goto error;
330 }
331 csd->disable_device = (disable_device_t)dlsym(csd->csd_client,
332 "csd_client_disable_device");
333 if (csd->disable_device == NULL) {
334 ALOGE("%s: dlsym error %s for csd_client_disable_device",
335 __func__, dlerror());
336 goto error;
337 }
338 csd->enable_device = (enable_device_t)dlsym(csd->csd_client,
339 "csd_client_enable_device");
340 if (csd->enable_device == NULL) {
341 ALOGE("%s: dlsym error %s for csd_client_enable_device",
342 __func__, dlerror());
343 goto error;
344 }
345 csd->start_voice = (start_voice_t)dlsym(csd->csd_client,
346 "csd_client_start_voice");
347 if (csd->start_voice == NULL) {
348 ALOGE("%s: dlsym error %s for csd_client_start_voice",
349 __func__, dlerror());
350 goto error;
351 }
352 csd->stop_voice = (stop_voice_t)dlsym(csd->csd_client,
353 "csd_client_stop_voice");
354 if (csd->stop_voice == NULL) {
355 ALOGE("%s: dlsym error %s for csd_client_stop_voice",
356 __func__, dlerror());
357 goto error;
358 }
359 csd->volume = (volume_t)dlsym(csd->csd_client,
360 "csd_client_volume");
361 if (csd->volume == NULL) {
362 ALOGE("%s: dlsym error %s for csd_client_volume",
363 __func__, dlerror());
364 goto error;
365 }
366 csd->mic_mute = (mic_mute_t)dlsym(csd->csd_client,
367 "csd_client_mic_mute");
368 if (csd->mic_mute == NULL) {
369 ALOGE("%s: dlsym error %s for csd_client_mic_mute",
370 __func__, dlerror());
371 goto error;
372 }
373 csd->slow_talk = (slow_talk_t)dlsym(csd->csd_client,
374 "csd_client_slow_talk");
375 if (csd->slow_talk == NULL) {
376 ALOGE("%s: dlsym error %s for csd_client_slow_talk",
377 __func__, dlerror());
378 goto error;
379 }
380 csd->start_playback = (start_playback_t)dlsym(csd->csd_client,
381 "csd_client_start_playback");
382 if (csd->start_playback == NULL) {
383 ALOGE("%s: dlsym error %s for csd_client_start_playback",
384 __func__, dlerror());
385 goto error;
386 }
387 csd->stop_playback = (stop_playback_t)dlsym(csd->csd_client,
388 "csd_client_stop_playback");
389 if (csd->stop_playback == NULL) {
390 ALOGE("%s: dlsym error %s for csd_client_stop_playback",
391 __func__, dlerror());
392 goto error;
393 }
394 csd->start_record = (start_record_t)dlsym(csd->csd_client,
395 "csd_client_start_record");
396 if (csd->start_record == NULL) {
397 ALOGE("%s: dlsym error %s for csd_client_start_record",
398 __func__, dlerror());
399 goto error;
400 }
401 csd->stop_record = (stop_record_t)dlsym(csd->csd_client,
402 "csd_client_stop_record");
403 if (csd->stop_record == NULL) {
404 ALOGE("%s: dlsym error %s for csd_client_stop_record",
405 __func__, dlerror());
406 goto error;
407 }
408 csd->init = (init_t)dlsym(csd->csd_client, "csd_client_init");
409
410 if (csd->init == NULL) {
411 ALOGE("%s: dlsym error %s for csd_client_init",
412 __func__, dlerror());
413 goto error;
414 } else {
415 csd->init();
416 }
417 }
418 return csd;
419
420error:
421 free(csd);
422 csd = NULL;
423 return csd;
424}
425
426void close_csd_client(struct csd_data *csd)
427{
428 if (csd != NULL) {
429 csd->deinit();
430 dlclose(csd->csd_client);
431 free(csd);
432 csd = NULL;
433 }
434}
435
Eric Laurentb23d5282013-05-14 15:27:20 -0700436void *platform_init(struct audio_device *adev)
437{
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800438 char platform[PROPERTY_VALUE_MAX];
439 char baseband[PROPERTY_VALUE_MAX];
Eric Laurentb23d5282013-05-14 15:27:20 -0700440 char value[PROPERTY_VALUE_MAX];
Apoorv Raghuvanshi84fa2fe2013-12-04 11:57:47 -0800441 struct platform_data *my_data = NULL;
442 int retry_num = 0, snd_card_num = 0;
Ravi Kumar Alamanda48c921d2013-10-29 06:07:44 -0700443 const char *snd_card_name;
sangwoo1b9f4b32013-06-21 18:22:55 -0700444
Eric Laurentb23d5282013-05-14 15:27:20 -0700445 my_data = calloc(1, sizeof(struct platform_data));
446
Apoorv Raghuvanshi84fa2fe2013-12-04 11:57:47 -0800447 while (snd_card_num < MAX_SND_CARD) {
448 adev->mixer = mixer_open(snd_card_num);
449
450 while (!adev->mixer && retry_num < RETRY_NUMBER) {
451 usleep(RETRY_US);
452 adev->mixer = mixer_open(snd_card_num);
453 retry_num++;
454 }
455
456 if (!adev->mixer) {
457 ALOGE("%s: Unable to open the mixer card: %d", __func__,
458 snd_card_num);
459 retry_num = 0;
460 snd_card_num++;
461 continue;
462 }
463
464 snd_card_name = mixer_get_name(adev->mixer);
465 ALOGV("%s: snd_card_name: %s", __func__, snd_card_name);
466
467 my_data->hw_info = hw_info_init(snd_card_name);
468 if (!my_data->hw_info) {
469 ALOGE("%s: Failed to init hardware info", __func__);
470 } else {
471 if (audio_extn_read_xml(adev, snd_card_num, MIXER_XML_PATH,
472 MIXER_XML_PATH_AUXPCM) == -ENOSYS)
473 adev->audio_route = audio_route_init(snd_card_num,
474 MIXER_XML_PATH);
475 if (!adev->audio_route) {
476 ALOGE("%s: Failed to init audio route controls, aborting.",
477 __func__);
478 free(my_data);
479 return NULL;
480 }
481 adev->snd_card = snd_card_num;
482 ALOGD("%s: Opened sound card:%d", __func__, snd_card_num);
483 break;
484 }
485 retry_num = 0;
486 snd_card_num++;
487 }
488
489 if (snd_card_num >= MAX_SND_CARD) {
490 ALOGE("%s: Unable to find correct sound card, aborting.", __func__);
491 free(my_data);
492 return NULL;
Ravi Kumar Alamanda48c921d2013-10-29 06:07:44 -0700493 }
494
Eric Laurentb23d5282013-05-14 15:27:20 -0700495 my_data->adev = adev;
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700496 my_data->btsco_sample_rate = SAMPLE_RATE_8KHZ;
Eric Laurentb23d5282013-05-14 15:27:20 -0700497 my_data->fluence_in_spkr_mode = false;
498 my_data->fluence_in_voice_call = false;
499 my_data->fluence_in_voice_rec = false;
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800500 my_data->fluence_in_audio_rec = false;
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700501 my_data->fluence_type = FLUENCE_NONE;
Eric Laurentb23d5282013-05-14 15:27:20 -0700502
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700503 property_get("ro.qc.sdk.audio.fluencetype", value, "");
504 if (!strncmp("fluencepro", value, sizeof("fluencepro"))) {
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800505 my_data->fluence_type = FLUENCE_QUAD_MIC | FLUENCE_DUAL_MIC;
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700506 } else if (!strncmp("fluence", value, sizeof("fluence"))) {
507 my_data->fluence_type = FLUENCE_DUAL_MIC;
508 } else {
509 my_data->fluence_type = FLUENCE_NONE;
Eric Laurentb23d5282013-05-14 15:27:20 -0700510 }
511
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700512 if (my_data->fluence_type != FLUENCE_NONE) {
Eric Laurentb23d5282013-05-14 15:27:20 -0700513 property_get("persist.audio.fluence.voicecall",value,"");
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700514 if (!strncmp("true", value, sizeof("true"))) {
Eric Laurentb23d5282013-05-14 15:27:20 -0700515 my_data->fluence_in_voice_call = true;
516 }
517
518 property_get("persist.audio.fluence.voicerec",value,"");
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700519 if (!strncmp("true", value, sizeof("true"))) {
Eric Laurentb23d5282013-05-14 15:27:20 -0700520 my_data->fluence_in_voice_rec = true;
521 }
522
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -0800523 property_get("persist.audio.fluence.audiorec",value,"");
524 if (!strncmp("true", value, sizeof("true"))) {
525 my_data->fluence_in_audio_rec = true;
526 }
527
Eric Laurentb23d5282013-05-14 15:27:20 -0700528 property_get("persist.audio.fluence.speaker",value,"");
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700529 if (!strncmp("true", value, sizeof("true"))) {
Eric Laurentb23d5282013-05-14 15:27:20 -0700530 my_data->fluence_in_spkr_mode = true;
531 }
532 }
533
Ben Rombergerc1dc70d2013-12-19 15:11:17 -0800534 my_data->voice_feature_set = VOICE_FEATURE_SET_DEFAULT;
Eric Laurentb23d5282013-05-14 15:27:20 -0700535 my_data->acdb_handle = dlopen(LIB_ACDB_LOADER, RTLD_NOW);
536 if (my_data->acdb_handle == NULL) {
537 ALOGE("%s: DLOPEN failed for %s", __func__, LIB_ACDB_LOADER);
538 } else {
539 ALOGV("%s: DLOPEN successful for %s", __func__, LIB_ACDB_LOADER);
540 my_data->acdb_deallocate = (acdb_deallocate_t)dlsym(my_data->acdb_handle,
541 "acdb_loader_deallocate_ACDB");
Ben Rombergerc1dc70d2013-12-19 15:11:17 -0800542 if (!my_data->acdb_deallocate)
543 ALOGE("%s: Could not find the symbol acdb_loader_deallocate_ACDB from %s",
544 __func__, LIB_ACDB_LOADER);
545
Eric Laurentb23d5282013-05-14 15:27:20 -0700546 my_data->acdb_send_audio_cal = (acdb_send_audio_cal_t)dlsym(my_data->acdb_handle,
547 "acdb_loader_send_audio_cal");
548 if (!my_data->acdb_send_audio_cal)
Ben Rombergerc1dc70d2013-12-19 15:11:17 -0800549 ALOGE("%s: Could not find the symbol acdb_send_audio_cal from %s",
Eric Laurentb23d5282013-05-14 15:27:20 -0700550 __func__, LIB_ACDB_LOADER);
Ben Rombergerc1dc70d2013-12-19 15:11:17 -0800551
Eric Laurentb23d5282013-05-14 15:27:20 -0700552 my_data->acdb_send_voice_cal = (acdb_send_voice_cal_t)dlsym(my_data->acdb_handle,
553 "acdb_loader_send_voice_cal");
Ben Rombergerc1dc70d2013-12-19 15:11:17 -0800554 if (!my_data->acdb_send_voice_cal)
555 ALOGE("%s: Could not find the symbol acdb_loader_send_voice_cal from %s",
556 __func__, LIB_ACDB_LOADER);
557
558 my_data->acdb_reload_vocvoltable = (acdb_reload_vocvoltable_t)dlsym(my_data->acdb_handle,
559 "acdb_loader_reload_vocvoltable");
560 if (!my_data->acdb_reload_vocvoltable)
561 ALOGE("%s: Could not find the symbol acdb_loader_reload_vocvoltable from %s",
562 __func__, LIB_ACDB_LOADER);
563
Eric Laurentb23d5282013-05-14 15:27:20 -0700564 my_data->acdb_init = (acdb_init_t)dlsym(my_data->acdb_handle,
565 "acdb_loader_init_ACDB");
566 if (my_data->acdb_init == NULL)
567 ALOGE("%s: dlsym error %s for acdb_loader_init_ACDB", __func__, dlerror());
568 else
569 my_data->acdb_init();
570 }
571
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800572 /* If platform is apq8084 and baseband is MDM, load CSD Client specific
573 * symbols. Voice call is handled by MDM and apps processor talks to
574 * MDM through CSD Client
575 */
576 property_get("ro.board.platform", platform, "");
577 property_get("ro.baseband", baseband, "");
578 if (!strncmp("apq8084", platform, sizeof("apq8084")) &&
579 !strncmp("mdm", baseband, sizeof("mdm"))) {
580 my_data->csd = open_csd_client();
581 }
582
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700583 /* init usb */
584 audio_extn_usb_init(adev);
Apoorv Raghuvanshi84fa2fe2013-12-04 11:57:47 -0800585 /* update sound cards appropriately */
586 audio_extn_usb_set_proxy_sound_card(adev->snd_card);
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700587
Apoorv Raghuvanshi6178a3f2013-10-19 12:38:54 -0700588 /* Read one time ssr property */
589 audio_extn_ssr_update_enabled(adev);
Gopikrishnaiah Anandanf538cef2013-10-28 14:06:03 -0700590 audio_extn_spkr_prot_init(adev);
Eric Laurentb23d5282013-05-14 15:27:20 -0700591 return my_data;
592}
593
594void platform_deinit(void *platform)
595{
Ravi Kumar Alamanda48c921d2013-10-29 06:07:44 -0700596 struct platform_data *my_data = (struct platform_data *)platform;
597
598 hw_info_deinit(my_data->hw_info);
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800599 close_csd_client(my_data->csd);
600
Eric Laurentb23d5282013-05-14 15:27:20 -0700601 free(platform);
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700602 /* deinit usb */
603 audio_extn_usb_deinit();
Eric Laurentb23d5282013-05-14 15:27:20 -0700604}
605
606const char *platform_get_snd_device_name(snd_device_t snd_device)
607{
608 if (snd_device >= SND_DEVICE_MIN && snd_device < SND_DEVICE_MAX)
609 return device_table[snd_device];
610 else
611 return "";
612}
613
Ravi Kumar Alamanda48c921d2013-10-29 06:07:44 -0700614int platform_get_snd_device_name_extn(void *platform, snd_device_t snd_device,
615 char *device_name)
616{
617 struct platform_data *my_data = (struct platform_data *)platform;
618
619 if (snd_device >= SND_DEVICE_MIN && snd_device < SND_DEVICE_MAX) {
620 strlcpy(device_name, device_table[snd_device], DEVICE_NAME_MAX_SIZE);
621 hw_info_append_hw_type(my_data->hw_info, snd_device, device_name);
622 } else {
623 strlcpy(device_name, "", DEVICE_NAME_MAX_SIZE);
624 return -EINVAL;
625 }
626
627 return 0;
628}
629
Eric Laurentb23d5282013-05-14 15:27:20 -0700630void platform_add_backend_name(char *mixer_path, snd_device_t snd_device)
631{
632 if (snd_device == SND_DEVICE_IN_BT_SCO_MIC)
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700633 strlcat(mixer_path, " bt-sco", MIXER_PATH_MAX_LENGTH);
634 else if (snd_device == SND_DEVICE_IN_BT_SCO_MIC_WB)
635 strlcat(mixer_path, " bt-sco-wb", MIXER_PATH_MAX_LENGTH);
Eric Laurentb23d5282013-05-14 15:27:20 -0700636 else if(snd_device == SND_DEVICE_OUT_BT_SCO)
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700637 strlcat(mixer_path, " bt-sco", MIXER_PATH_MAX_LENGTH);
638 else if(snd_device == SND_DEVICE_OUT_BT_SCO_WB)
639 strlcat(mixer_path, " bt-sco-wb", MIXER_PATH_MAX_LENGTH);
Eric Laurentb23d5282013-05-14 15:27:20 -0700640 else if (snd_device == SND_DEVICE_OUT_HDMI)
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700641 strlcat(mixer_path, " hdmi", MIXER_PATH_MAX_LENGTH);
Eric Laurentb23d5282013-05-14 15:27:20 -0700642 else if (snd_device == SND_DEVICE_OUT_SPEAKER_AND_HDMI)
643 strcat(mixer_path, " speaker-and-hdmi");
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700644 else if (snd_device == SND_DEVICE_OUT_AFE_PROXY)
645 strlcat(mixer_path, " afe-proxy", MIXER_PATH_MAX_LENGTH);
646 else if (snd_device == SND_DEVICE_OUT_USB_HEADSET)
647 strlcat(mixer_path, " usb-headphones", MIXER_PATH_MAX_LENGTH);
648 else if (snd_device == SND_DEVICE_OUT_SPEAKER_AND_USB_HEADSET)
649 strlcat(mixer_path, " speaker-and-usb-headphones",
650 MIXER_PATH_MAX_LENGTH);
651 else if (snd_device == SND_DEVICE_IN_USB_HEADSET_MIC)
652 strlcat(mixer_path, " usb-headset-mic", MIXER_PATH_MAX_LENGTH);
Apoorv Raghuvanshi6e262842013-10-06 14:39:35 -0700653 else if (snd_device == SND_DEVICE_IN_CAPTURE_FM)
654 strlcat(mixer_path, " capture-fm", MIXER_PATH_MAX_LENGTH);
655 else if (snd_device == SND_DEVICE_OUT_TRANSMISSION_FM)
656 strlcat(mixer_path, " transmission-fm", MIXER_PATH_MAX_LENGTH);
Eric Laurentb23d5282013-05-14 15:27:20 -0700657}
658
659int platform_get_pcm_device_id(audio_usecase_t usecase, int device_type)
660{
661 int device_id;
662 if (device_type == PCM_PLAYBACK)
663 device_id = pcm_device_table[usecase][0];
664 else
665 device_id = pcm_device_table[usecase][1];
666 return device_id;
667}
668
669int platform_send_audio_calibration(void *platform, snd_device_t snd_device)
670{
671 struct platform_data *my_data = (struct platform_data *)platform;
672 int acdb_dev_id, acdb_dev_type;
673
674 acdb_dev_id = acdb_device_table[snd_device];
675 if (acdb_dev_id < 0) {
676 ALOGE("%s: Could not find acdb id for device(%d)",
677 __func__, snd_device);
678 return -EINVAL;
679 }
680 if (my_data->acdb_send_audio_cal) {
Eric Laurent994a6932013-07-17 11:51:42 -0700681 ("%s: sending audio calibration for snd_device(%d) acdb_id(%d)",
Eric Laurentb23d5282013-05-14 15:27:20 -0700682 __func__, snd_device, acdb_dev_id);
683 if (snd_device >= SND_DEVICE_OUT_BEGIN &&
684 snd_device < SND_DEVICE_OUT_END)
685 acdb_dev_type = ACDB_DEV_TYPE_OUT;
686 else
687 acdb_dev_type = ACDB_DEV_TYPE_IN;
688 my_data->acdb_send_audio_cal(acdb_dev_id, acdb_dev_type);
689 }
690 return 0;
691}
692
693int platform_switch_voice_call_device_pre(void *platform)
694{
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800695 struct platform_data *my_data = (struct platform_data *)platform;
696 int ret = 0;
697
Vidyakumar Athota1fd21792013-11-15 14:50:57 -0800698 if (my_data->csd != NULL &&
699 my_data->adev->mode == AUDIO_MODE_IN_CALL) {
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800700 /* This must be called before disabling mixer controls on APQ side */
701 ret = my_data->csd->disable_device();
702 if (ret < 0) {
703 ALOGE("%s: csd_client_disable_device, failed, error %d",
704 __func__, ret);
705 }
706 }
707 return ret;
Eric Laurentb23d5282013-05-14 15:27:20 -0700708}
709
710int platform_switch_voice_call_device_post(void *platform,
711 snd_device_t out_snd_device,
712 snd_device_t in_snd_device)
713{
714 struct platform_data *my_data = (struct platform_data *)platform;
715 int acdb_rx_id, acdb_tx_id;
716
717 if (my_data->acdb_send_voice_cal == NULL) {
718 ALOGE("%s: dlsym error for acdb_send_voice_call", __func__);
719 } else {
Vidyakumar Athota1fd21792013-11-15 14:50:57 -0800720 acdb_rx_id = acdb_device_table[out_snd_device];
721 acdb_tx_id = acdb_device_table[in_snd_device];
722
Eric Laurentb23d5282013-05-14 15:27:20 -0700723 if (acdb_rx_id > 0 && acdb_tx_id > 0)
724 my_data->acdb_send_voice_cal(acdb_rx_id, acdb_tx_id);
725 else
726 ALOGE("%s: Incorrect ACDB IDs (rx: %d tx: %d)", __func__,
727 acdb_rx_id, acdb_tx_id);
728 }
729
Vidyakumar Athota1fd21792013-11-15 14:50:57 -0800730 return 0;
731}
732
733int platform_switch_voice_call_usecase_route_post(void *platform,
734 snd_device_t out_snd_device,
735 snd_device_t in_snd_device)
736{
737 struct platform_data *my_data = (struct platform_data *)platform;
738 int acdb_rx_id, acdb_tx_id;
739 int ret = 0;
740
741 acdb_rx_id = acdb_device_table[out_snd_device];
742 acdb_tx_id = acdb_device_table[in_snd_device];
743
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800744 if (my_data->csd != NULL) {
Vidyakumar Athota1fd21792013-11-15 14:50:57 -0800745 if (acdb_rx_id > 0 && acdb_tx_id > 0) {
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800746 ret = my_data->csd->enable_device(acdb_rx_id, acdb_tx_id,
747 my_data->adev->acdb_settings);
748 if (ret < 0) {
749 ALOGE("%s: csd_enable_device, failed, error %d",
750 __func__, ret);
751 }
752 } else {
753 ALOGE("%s: Incorrect ACDB IDs (rx: %d tx: %d)", __func__,
754 acdb_rx_id, acdb_tx_id);
755 }
756 }
757 return ret;
Eric Laurentb23d5282013-05-14 15:27:20 -0700758}
759
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800760int platform_start_voice_call(void *platform, uint32_t vsid)
Eric Laurentb23d5282013-05-14 15:27:20 -0700761{
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800762 struct platform_data *my_data = (struct platform_data *)platform;
763 int ret = 0;
764
765 if (my_data->csd != NULL) {
766 ret = my_data->csd->start_voice(vsid);
767 if (ret < 0) {
768 ALOGE("%s: csd_start_voice error %d\n", __func__, ret);
769 }
770 }
771 return ret;
Eric Laurentb23d5282013-05-14 15:27:20 -0700772}
773
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800774int platform_stop_voice_call(void *platform, uint32_t vsid)
Eric Laurentb23d5282013-05-14 15:27:20 -0700775{
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800776 struct platform_data *my_data = (struct platform_data *)platform;
777 int ret = 0;
778
779 if (my_data->csd != NULL) {
780 ret = my_data->csd->stop_voice(vsid);
781 if (ret < 0) {
782 ALOGE("%s: csd_stop_voice error %d\n", __func__, ret);
783 }
784 }
785 return ret;
Eric Laurentb23d5282013-05-14 15:27:20 -0700786}
787
788int platform_set_voice_volume(void *platform, int volume)
789{
790 struct platform_data *my_data = (struct platform_data *)platform;
791 struct audio_device *adev = my_data->adev;
792 struct mixer_ctl *ctl;
sangwoo53b2cf02013-07-25 19:18:44 -0700793 const char *mixer_ctl_name = "Voice Rx Gain";
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800794 int vol_index = 0, ret = 0;
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700795 uint32_t set_values[ ] = {0,
796 ALL_SESSION_VSID,
797 DEFAULT_VOLUME_RAMP_DURATION_MS};
Eric Laurentb23d5282013-05-14 15:27:20 -0700798
799 // Voice volume levels are mapped to adsp volume levels as follows.
800 // 100 -> 5, 80 -> 4, 60 -> 3, 40 -> 2, 20 -> 1 0 -> 0
801 // But this values don't changed in kernel. So, below change is need.
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700802 vol_index = (int)percent_to_index(volume, MIN_VOL_INDEX, MAX_VOL_INDEX);
803 set_values[0] = vol_index;
Eric Laurentb23d5282013-05-14 15:27:20 -0700804
805 ctl = mixer_get_ctl_by_name(adev->mixer, mixer_ctl_name);
806 if (!ctl) {
807 ALOGE("%s: Could not get ctl for mixer cmd - %s",
808 __func__, mixer_ctl_name);
809 return -EINVAL;
810 }
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700811 ALOGV("Setting voice volume index: %d", set_values[0]);
812 mixer_ctl_set_array(ctl, set_values, ARRAY_SIZE(set_values));
Eric Laurentb23d5282013-05-14 15:27:20 -0700813
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800814 if (my_data->csd != NULL) {
815 ret = my_data->csd->volume(ALL_SESSION_VSID, volume);
816 if (ret < 0) {
817 ALOGE("%s: csd_volume error %d", __func__, ret);
818 }
819 }
820 return ret;
Eric Laurentb23d5282013-05-14 15:27:20 -0700821}
822
823int platform_set_mic_mute(void *platform, bool state)
824{
825 struct platform_data *my_data = (struct platform_data *)platform;
826 struct audio_device *adev = my_data->adev;
827 struct mixer_ctl *ctl;
828 const char *mixer_ctl_name = "Voice Tx Mute";
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800829 int ret = 0;
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700830 uint32_t set_values[ ] = {0,
831 ALL_SESSION_VSID,
832 DEFAULT_VOLUME_RAMP_DURATION_MS};
Eric Laurentb23d5282013-05-14 15:27:20 -0700833
Narsinga Rao Chella05573b72013-11-15 15:21:40 -0800834 set_values[0] = state;
835 ctl = mixer_get_ctl_by_name(adev->mixer, mixer_ctl_name);
836 if (!ctl) {
837 ALOGE("%s: Could not get ctl for mixer cmd - %s",
838 __func__, mixer_ctl_name);
839 return -EINVAL;
840 }
841 ALOGV("Setting voice mute state: %d", state);
842 mixer_ctl_set_array(ctl, set_values, ARRAY_SIZE(set_values));
Eric Laurentb23d5282013-05-14 15:27:20 -0700843
Narsinga Rao Chella05573b72013-11-15 15:21:40 -0800844 if (my_data->csd != NULL) {
845 ret = my_data->csd->mic_mute(ALL_SESSION_VSID, state);
846 if (ret < 0) {
847 ALOGE("%s: csd_mic_mute error %d", __func__, ret);
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -0800848 }
849 }
850 return ret;
Eric Laurentb23d5282013-05-14 15:27:20 -0700851}
852
853snd_device_t platform_get_output_snd_device(void *platform, audio_devices_t devices)
854{
855 struct platform_data *my_data = (struct platform_data *)platform;
856 struct audio_device *adev = my_data->adev;
857 audio_mode_t mode = adev->mode;
858 snd_device_t snd_device = SND_DEVICE_NONE;
859
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700860 audio_channel_mask_t channel_mask = (adev->active_input == NULL) ?
861 AUDIO_CHANNEL_IN_MONO : adev->active_input->channel_mask;
862 int channel_count = popcount(channel_mask);
863
Eric Laurentb23d5282013-05-14 15:27:20 -0700864 ALOGV("%s: enter: output devices(%#x)", __func__, devices);
865 if (devices == AUDIO_DEVICE_NONE ||
866 devices & AUDIO_DEVICE_BIT_IN) {
867 ALOGV("%s: Invalid output devices (%#x)", __func__, devices);
868 goto exit;
869 }
870
Narsinga Rao Chella05573b72013-11-15 15:21:40 -0800871 if ((mode == AUDIO_MODE_IN_CALL) ||
872 voice_extn_compress_voip_is_active(adev)) {
Eric Laurentb23d5282013-05-14 15:27:20 -0700873 if (devices & AUDIO_DEVICE_OUT_WIRED_HEADPHONE ||
874 devices & AUDIO_DEVICE_OUT_WIRED_HEADSET) {
Narsinga Rao Chella05573b72013-11-15 15:21:40 -0800875 if ((adev->voice.tty_mode != TTY_MODE_OFF) &&
876 !voice_extn_compress_voip_is_active(adev)) {
877 switch (adev->voice.tty_mode) {
878 case TTY_MODE_FULL:
879 snd_device = SND_DEVICE_OUT_VOICE_TTY_FULL_HEADPHONES;
880 break;
881 case TTY_MODE_VCO:
882 snd_device = SND_DEVICE_OUT_VOICE_TTY_VCO_HEADPHONES;
883 break;
884 case TTY_MODE_HCO:
885 snd_device = SND_DEVICE_OUT_VOICE_TTY_HCO_HANDSET;
886 break;
887 default:
888 ALOGE("%s: Invalid TTY mode (%#x)",
889 __func__, adev->voice.tty_mode);
890 }
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700891 } else if (audio_extn_get_anc_enabled()) {
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700892 if (audio_extn_should_use_fb_anc())
893 snd_device = SND_DEVICE_OUT_VOICE_ANC_FB_HEADSET;
894 else
895 snd_device = SND_DEVICE_OUT_VOICE_ANC_HEADSET;
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700896 } else {
Eric Laurentb23d5282013-05-14 15:27:20 -0700897 snd_device = SND_DEVICE_OUT_VOICE_HEADPHONES;
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700898 }
Eric Laurentb23d5282013-05-14 15:27:20 -0700899 } 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_SPEAKER) {
905 snd_device = SND_DEVICE_OUT_VOICE_SPEAKER;
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) {
Ravi Kumar Alamandaceb40822013-11-06 11:01:47 -0800912 if (audio_extn_should_use_handset_anc(channel_count))
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700913 snd_device = SND_DEVICE_OUT_ANC_HANDSET;
Eric Laurentb23d5282013-05-14 15:27:20 -0700914 else
Mingming Yin8e5a4f62013-10-07 15:23:41 -0700915 snd_device = SND_DEVICE_OUT_VOICE_HANDSET;
Eric Laurentb23d5282013-05-14 15:27:20 -0700916 }
917 if (snd_device != SND_DEVICE_NONE) {
918 goto exit;
919 }
920 }
921
922 if (popcount(devices) == 2) {
923 if (devices == (AUDIO_DEVICE_OUT_WIRED_HEADPHONE |
924 AUDIO_DEVICE_OUT_SPEAKER)) {
925 snd_device = SND_DEVICE_OUT_SPEAKER_AND_HEADPHONES;
926 } else if (devices == (AUDIO_DEVICE_OUT_WIRED_HEADSET |
927 AUDIO_DEVICE_OUT_SPEAKER)) {
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700928 if (audio_extn_get_anc_enabled())
929 snd_device = SND_DEVICE_OUT_SPEAKER_AND_ANC_HEADSET;
930 else
931 snd_device = SND_DEVICE_OUT_SPEAKER_AND_HEADPHONES;
Eric Laurentb23d5282013-05-14 15:27:20 -0700932 } else if (devices == (AUDIO_DEVICE_OUT_AUX_DIGITAL |
933 AUDIO_DEVICE_OUT_SPEAKER)) {
934 snd_device = SND_DEVICE_OUT_SPEAKER_AND_HDMI;
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700935 } else if (devices == (AUDIO_DEVICE_OUT_ANLG_DOCK_HEADSET |
936 AUDIO_DEVICE_OUT_SPEAKER)) {
937 snd_device = SND_DEVICE_OUT_SPEAKER_AND_USB_HEADSET;
Eric Laurentb23d5282013-05-14 15:27:20 -0700938 } else {
939 ALOGE("%s: Invalid combo device(%#x)", __func__, devices);
940 goto exit;
941 }
942 if (snd_device != SND_DEVICE_NONE) {
943 goto exit;
944 }
945 }
946
947 if (popcount(devices) != 1) {
948 ALOGE("%s: Invalid output devices(%#x)", __func__, devices);
949 goto exit;
950 }
951
952 if (devices & AUDIO_DEVICE_OUT_WIRED_HEADPHONE ||
953 devices & AUDIO_DEVICE_OUT_WIRED_HEADSET) {
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -0700954 if (devices & AUDIO_DEVICE_OUT_WIRED_HEADSET
955 && audio_extn_get_anc_enabled()) {
956 if (audio_extn_should_use_fb_anc())
957 snd_device = SND_DEVICE_OUT_ANC_FB_HEADSET;
958 else
959 snd_device = SND_DEVICE_OUT_ANC_HEADSET;
960 }
961 else
962 snd_device = SND_DEVICE_OUT_HEADPHONES;
Eric Laurentb23d5282013-05-14 15:27:20 -0700963 } else if (devices & AUDIO_DEVICE_OUT_SPEAKER) {
964 if (adev->speaker_lr_swap)
965 snd_device = SND_DEVICE_OUT_SPEAKER_REVERSE;
966 else
967 snd_device = SND_DEVICE_OUT_SPEAKER;
968 } else if (devices & AUDIO_DEVICE_OUT_ALL_SCO) {
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -0700969 if (my_data->btsco_sample_rate == SAMPLE_RATE_16KHZ)
970 snd_device = SND_DEVICE_OUT_BT_SCO_WB;
971 else
972 snd_device = SND_DEVICE_OUT_BT_SCO;
Eric Laurentb23d5282013-05-14 15:27:20 -0700973 } else if (devices & AUDIO_DEVICE_OUT_AUX_DIGITAL) {
974 snd_device = SND_DEVICE_OUT_HDMI ;
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700975 } else if (devices & AUDIO_DEVICE_OUT_ANLG_DOCK_HEADSET ||
976 devices & AUDIO_DEVICE_OUT_DGTL_DOCK_HEADSET) {
977 snd_device = SND_DEVICE_OUT_USB_HEADSET;
Apoorv Raghuvanshi6e262842013-10-06 14:39:35 -0700978 } else if (devices & AUDIO_DEVICE_OUT_FM_TX) {
979 snd_device = SND_DEVICE_OUT_TRANSMISSION_FM;
Eric Laurentb23d5282013-05-14 15:27:20 -0700980 } else if (devices & AUDIO_DEVICE_OUT_EARPIECE) {
981 snd_device = SND_DEVICE_OUT_HANDSET;
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -0700982 } else if (devices & AUDIO_DEVICE_OUT_PROXY) {
983 ALOGD("%s: setting sink capability for Proxy", __func__);
984 audio_extn_set_afe_proxy_channel_mixer(adev);
985 snd_device = SND_DEVICE_OUT_AFE_PROXY;
Eric Laurentb23d5282013-05-14 15:27:20 -0700986 } else {
987 ALOGE("%s: Unknown device(s) %#x", __func__, devices);
988 }
989exit:
990 ALOGV("%s: exit: snd_device(%s)", __func__, device_table[snd_device]);
991 return snd_device;
992}
993
994snd_device_t platform_get_input_snd_device(void *platform, audio_devices_t out_device)
995{
996 struct platform_data *my_data = (struct platform_data *)platform;
997 struct audio_device *adev = my_data->adev;
998 audio_source_t source = (adev->active_input == NULL) ?
999 AUDIO_SOURCE_DEFAULT : adev->active_input->source;
1000
1001 audio_mode_t mode = adev->mode;
1002 audio_devices_t in_device = ((adev->active_input == NULL) ?
1003 AUDIO_DEVICE_NONE : adev->active_input->device)
1004 & ~AUDIO_DEVICE_BIT_IN;
1005 audio_channel_mask_t channel_mask = (adev->active_input == NULL) ?
1006 AUDIO_CHANNEL_IN_MONO : adev->active_input->channel_mask;
1007 snd_device_t snd_device = SND_DEVICE_NONE;
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -07001008 int channel_count = popcount(channel_mask);
Eric Laurentb23d5282013-05-14 15:27:20 -07001009
1010 ALOGV("%s: enter: out_device(%#x) in_device(%#x)",
1011 __func__, out_device, in_device);
Helen Zeng067b96b2013-11-26 12:10:29 -08001012 if ((out_device != AUDIO_DEVICE_NONE) && ((mode == AUDIO_MODE_IN_CALL) ||
1013 voice_extn_compress_voip_is_active(adev))) {
Narsinga Rao Chella05573b72013-11-15 15:21:40 -08001014 if ((adev->voice.tty_mode != TTY_MODE_OFF) &&
1015 !voice_extn_compress_voip_is_active(adev)) {
Eric Laurentb23d5282013-05-14 15:27:20 -07001016 if (out_device & AUDIO_DEVICE_OUT_WIRED_HEADPHONE ||
1017 out_device & AUDIO_DEVICE_OUT_WIRED_HEADSET) {
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001018 switch (adev->voice.tty_mode) {
Eric Laurentb23d5282013-05-14 15:27:20 -07001019 case TTY_MODE_FULL:
1020 snd_device = SND_DEVICE_IN_VOICE_TTY_FULL_HEADSET_MIC;
1021 break;
1022 case TTY_MODE_VCO:
1023 snd_device = SND_DEVICE_IN_VOICE_TTY_VCO_HANDSET_MIC;
1024 break;
1025 case TTY_MODE_HCO:
1026 snd_device = SND_DEVICE_IN_VOICE_TTY_HCO_HEADSET_MIC;
1027 break;
1028 default:
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001029 ALOGE("%s: Invalid TTY mode (%#x)",
1030 __func__, adev->voice.tty_mode);
Eric Laurentb23d5282013-05-14 15:27:20 -07001031 }
1032 goto exit;
1033 }
1034 }
1035 if (out_device & AUDIO_DEVICE_OUT_EARPIECE ||
1036 out_device & AUDIO_DEVICE_OUT_WIRED_HEADPHONE) {
Apoorv Raghuvanshi9eaf94e2013-10-04 16:13:44 -07001037 if (out_device & AUDIO_DEVICE_OUT_EARPIECE &&
1038 audio_extn_should_use_handset_anc(channel_count)) {
1039 snd_device = SND_DEVICE_IN_AANC_HANDSET_MIC;
1040 } else if (my_data->fluence_type == FLUENCE_NONE ||
Mingming Yin8e5a4f62013-10-07 15:23:41 -07001041 my_data->fluence_in_voice_call == false) {
Eric Laurentb23d5282013-05-14 15:27:20 -07001042 snd_device = SND_DEVICE_IN_HANDSET_MIC;
Laxminath Kasam7294f522013-12-11 05:07:00 +05301043 set_echo_reference(adev->mixer, EC_REF_RX);
Eric Laurentb23d5282013-05-14 15:27:20 -07001044 } else {
Ravi Kumar Alamandaceb40822013-11-06 11:01:47 -08001045 snd_device = SND_DEVICE_IN_VOICE_DMIC;
Mingming Yin8e5a4f62013-10-07 15:23:41 -07001046 adev->acdb_settings |= DMIC_FLAG;
Eric Laurentb23d5282013-05-14 15:27:20 -07001047 }
1048 } else if (out_device & AUDIO_DEVICE_OUT_WIRED_HEADSET) {
1049 snd_device = SND_DEVICE_IN_VOICE_HEADSET_MIC;
1050 } else if (out_device & AUDIO_DEVICE_OUT_ALL_SCO) {
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001051 if (my_data->btsco_sample_rate == SAMPLE_RATE_16KHZ)
1052 snd_device = SND_DEVICE_IN_BT_SCO_MIC_WB;
1053 else
1054 snd_device = SND_DEVICE_IN_BT_SCO_MIC;
Eric Laurentb23d5282013-05-14 15:27:20 -07001055 } else if (out_device & AUDIO_DEVICE_OUT_SPEAKER) {
Mingming Yin8e5a4f62013-10-07 15:23:41 -07001056 if (my_data->fluence_type != FLUENCE_NONE &&
1057 my_data->fluence_in_voice_call &&
1058 my_data->fluence_in_spkr_mode) {
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -08001059 if(my_data->fluence_type & FLUENCE_QUAD_MIC) {
Mingming Yin8e5a4f62013-10-07 15:23:41 -07001060 adev->acdb_settings |= QMIC_FLAG;
1061 snd_device = SND_DEVICE_IN_VOICE_SPEAKER_QMIC;
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -08001062 } else {
1063 adev->acdb_settings |= DMIC_FLAG;
1064 snd_device = SND_DEVICE_IN_VOICE_SPEAKER_DMIC;
Mingming Yin8e5a4f62013-10-07 15:23:41 -07001065 }
Eric Laurentb23d5282013-05-14 15:27:20 -07001066 } else {
1067 snd_device = SND_DEVICE_IN_VOICE_SPEAKER_MIC;
1068 }
1069 }
1070 } else if (source == AUDIO_SOURCE_CAMCORDER) {
1071 if (in_device & AUDIO_DEVICE_IN_BUILTIN_MIC ||
1072 in_device & AUDIO_DEVICE_IN_BACK_MIC) {
1073 snd_device = SND_DEVICE_IN_CAMCORDER_MIC;
1074 }
1075 } else if (source == AUDIO_SOURCE_VOICE_RECOGNITION) {
1076 if (in_device & AUDIO_DEVICE_IN_BUILTIN_MIC) {
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -08001077 if (channel_count == 2) {
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -08001078 snd_device = SND_DEVICE_IN_VOICE_REC_DMIC_STEREO;
Mingming Yin8e5a4f62013-10-07 15:23:41 -07001079 adev->acdb_settings |= DMIC_FLAG;
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -08001080 } else if (adev->active_input->enable_ns)
1081 snd_device = SND_DEVICE_IN_VOICE_REC_MIC_NS;
1082 else if (my_data->fluence_type != FLUENCE_NONE &&
1083 my_data->fluence_in_voice_rec) {
1084 snd_device = SND_DEVICE_IN_VOICE_REC_DMIC_FLUENCE;
1085 adev->acdb_settings |= DMIC_FLAG;
1086 } else {
1087 snd_device = SND_DEVICE_IN_VOICE_REC_MIC;
1088 }
Eric Laurentb23d5282013-05-14 15:27:20 -07001089 }
1090 } else if (source == AUDIO_SOURCE_VOICE_COMMUNICATION) {
1091 if (out_device & AUDIO_DEVICE_OUT_SPEAKER)
1092 in_device = AUDIO_DEVICE_IN_BACK_MIC;
1093 if (adev->active_input) {
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -08001094 if (adev->active_input->enable_aec &&
1095 adev->active_input->enable_ns) {
Eric Laurentb23d5282013-05-14 15:27:20 -07001096 if (in_device & AUDIO_DEVICE_IN_BACK_MIC) {
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -08001097 if (my_data->fluence_type & FLUENCE_DUAL_MIC &&
1098 my_data->fluence_in_spkr_mode) {
1099 snd_device = SND_DEVICE_IN_SPEAKER_DMIC_AEC_NS;
1100 adev->acdb_settings |= DMIC_FLAG;
1101 } else
1102 snd_device = SND_DEVICE_IN_SPEAKER_MIC_AEC_NS;
Eric Laurentb23d5282013-05-14 15:27:20 -07001103 } else if (in_device & AUDIO_DEVICE_IN_BUILTIN_MIC) {
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -08001104 if (my_data->fluence_type & FLUENCE_DUAL_MIC) {
1105 snd_device = SND_DEVICE_IN_HANDSET_DMIC_AEC_NS;
1106 adev->acdb_settings |= DMIC_FLAG;
1107 } else
1108 snd_device = SND_DEVICE_IN_HANDSET_MIC_AEC_NS;
Eric Laurentb23d5282013-05-14 15:27:20 -07001109 } else if (in_device & AUDIO_DEVICE_IN_WIRED_HEADSET) {
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -08001110 snd_device = SND_DEVICE_IN_HEADSET_MIC_FLUENCE;
Eric Laurentb23d5282013-05-14 15:27:20 -07001111 }
Laxminath Kasam7294f522013-12-11 05:07:00 +05301112 set_echo_reference(adev->mixer, EC_REF_RX);
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -08001113 } else if (adev->active_input->enable_aec) {
1114 if (in_device & AUDIO_DEVICE_IN_BACK_MIC) {
1115 if (my_data->fluence_type & FLUENCE_DUAL_MIC) {
1116 snd_device = SND_DEVICE_IN_SPEAKER_DMIC_AEC;
1117 adev->acdb_settings |= DMIC_FLAG;
1118 } else
1119 snd_device = SND_DEVICE_IN_SPEAKER_MIC_AEC;
1120 } else if (in_device & AUDIO_DEVICE_IN_BUILTIN_MIC) {
1121 if (my_data->fluence_type & FLUENCE_DUAL_MIC) {
1122 snd_device = SND_DEVICE_IN_HANDSET_DMIC_AEC;
1123 adev->acdb_settings |= DMIC_FLAG;
1124 } else
1125 snd_device = SND_DEVICE_IN_HANDSET_MIC_AEC;
1126 } else if (in_device & AUDIO_DEVICE_IN_WIRED_HEADSET) {
1127 snd_device = SND_DEVICE_IN_HEADSET_MIC_FLUENCE;
1128 }
Laxminath Kasam7294f522013-12-11 05:07:00 +05301129 set_echo_reference(adev->mixer, EC_REF_RX);
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -08001130 } else if (adev->active_input->enable_ns) {
1131 if (in_device & AUDIO_DEVICE_IN_BACK_MIC) {
1132 if (my_data->fluence_type & FLUENCE_DUAL_MIC) {
1133 snd_device = SND_DEVICE_IN_SPEAKER_DMIC_NS;
1134 adev->acdb_settings |= DMIC_FLAG;
1135 } else
1136 snd_device = SND_DEVICE_IN_SPEAKER_MIC_NS;
1137 } else if (in_device & AUDIO_DEVICE_IN_BUILTIN_MIC) {
1138 if (my_data->fluence_type & FLUENCE_DUAL_MIC) {
1139 snd_device = SND_DEVICE_IN_HANDSET_DMIC_NS;
1140 adev->acdb_settings |= DMIC_FLAG;
1141 } else
1142 snd_device = SND_DEVICE_IN_HANDSET_MIC_NS;
1143 } else if (in_device & AUDIO_DEVICE_IN_WIRED_HEADSET) {
1144 snd_device = SND_DEVICE_IN_HEADSET_MIC_FLUENCE;
1145 }
1146 set_echo_reference(adev->mixer, "NONE");
Eric Laurentb23d5282013-05-14 15:27:20 -07001147 } else
1148 set_echo_reference(adev->mixer, "NONE");
1149 }
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -08001150 } else if (source == AUDIO_SOURCE_MIC) {
Apoorv Raghuvanshic0536542013-11-14 16:25:59 -08001151 if (in_device & AUDIO_DEVICE_IN_BUILTIN_MIC &&
1152 channel_count == 1 ) {
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -08001153 if(my_data->fluence_type & FLUENCE_DUAL_MIC &&
1154 my_data->fluence_in_audio_rec)
1155 snd_device = SND_DEVICE_IN_HANDSET_DMIC;
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -08001156 }
Mingming Yinab429782013-11-07 11:16:55 -08001157 } else if (source == AUDIO_SOURCE_FM_RX ||
Ravi Kumar Alamanda198185e2013-11-07 15:42:19 -08001158 source == AUDIO_SOURCE_FM_RX_A2DP) {
Preetam Singh Ranawatde84f1a2013-11-01 14:58:16 -07001159 snd_device = SND_DEVICE_IN_CAPTURE_FM;
Eric Laurentb23d5282013-05-14 15:27:20 -07001160 } else if (source == AUDIO_SOURCE_DEFAULT) {
1161 goto exit;
1162 }
1163
1164
1165 if (snd_device != SND_DEVICE_NONE) {
1166 goto exit;
1167 }
1168
1169 if (in_device != AUDIO_DEVICE_NONE &&
1170 !(in_device & AUDIO_DEVICE_IN_VOICE_CALL) &&
1171 !(in_device & AUDIO_DEVICE_IN_COMMUNICATION)) {
1172 if (in_device & AUDIO_DEVICE_IN_BUILTIN_MIC) {
Apoorv Raghuvanshi6178a3f2013-10-19 12:38:54 -07001173 if (audio_extn_ssr_get_enabled() && channel_count == 6)
1174 snd_device = SND_DEVICE_IN_QUAD_MIC;
Ravi Kumar Alamandafae42112013-11-07 23:31:54 -08001175 else if (channel_count == 2)
Apoorv Raghuvanshi6bd8dbf2013-10-19 18:37:52 -07001176 snd_device = SND_DEVICE_IN_HANDSET_STEREO_DMIC;
Apoorv Raghuvanshi6178a3f2013-10-19 12:38:54 -07001177 else
1178 snd_device = SND_DEVICE_IN_HANDSET_MIC;
Eric Laurentb23d5282013-05-14 15:27:20 -07001179 } else if (in_device & AUDIO_DEVICE_IN_BACK_MIC) {
1180 snd_device = SND_DEVICE_IN_SPEAKER_MIC;
1181 } else if (in_device & AUDIO_DEVICE_IN_WIRED_HEADSET) {
1182 snd_device = SND_DEVICE_IN_HEADSET_MIC;
1183 } else if (in_device & AUDIO_DEVICE_IN_BLUETOOTH_SCO_HEADSET) {
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001184 if (my_data->btsco_sample_rate == SAMPLE_RATE_16KHZ)
1185 snd_device = SND_DEVICE_IN_BT_SCO_MIC_WB;
1186 else
1187 snd_device = SND_DEVICE_IN_BT_SCO_MIC;
Eric Laurentb23d5282013-05-14 15:27:20 -07001188 } else if (in_device & AUDIO_DEVICE_IN_AUX_DIGITAL) {
1189 snd_device = SND_DEVICE_IN_HDMI_MIC;
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -07001190 } else if (in_device & AUDIO_DEVICE_IN_ANLG_DOCK_HEADSET ||
1191 in_device & AUDIO_DEVICE_IN_DGTL_DOCK_HEADSET) {
1192 snd_device = SND_DEVICE_IN_USB_HEADSET_MIC;
Apoorv Raghuvanshi6e262842013-10-06 14:39:35 -07001193 } else if (in_device & AUDIO_DEVICE_IN_FM_RX) {
1194 snd_device = SND_DEVICE_IN_CAPTURE_FM;
Eric Laurentb23d5282013-05-14 15:27:20 -07001195 } else {
1196 ALOGE("%s: Unknown input device(s) %#x", __func__, in_device);
1197 ALOGW("%s: Using default handset-mic", __func__);
1198 snd_device = SND_DEVICE_IN_HANDSET_MIC;
1199 }
1200 } else {
1201 if (out_device & AUDIO_DEVICE_OUT_EARPIECE) {
1202 snd_device = SND_DEVICE_IN_HANDSET_MIC;
1203 } else if (out_device & AUDIO_DEVICE_OUT_WIRED_HEADSET) {
1204 snd_device = SND_DEVICE_IN_HEADSET_MIC;
1205 } else if (out_device & AUDIO_DEVICE_OUT_SPEAKER) {
Apoorv Raghuvanshi6bd8dbf2013-10-19 18:37:52 -07001206 if (channel_count > 1)
1207 snd_device = SND_DEVICE_IN_SPEAKER_STEREO_DMIC;
1208 else
1209 snd_device = SND_DEVICE_IN_SPEAKER_MIC;
Eric Laurentb23d5282013-05-14 15:27:20 -07001210 } else if (out_device & AUDIO_DEVICE_OUT_WIRED_HEADPHONE) {
1211 snd_device = SND_DEVICE_IN_HANDSET_MIC;
1212 } else if (out_device & AUDIO_DEVICE_OUT_BLUETOOTH_SCO_HEADSET) {
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001213 if (my_data->btsco_sample_rate == SAMPLE_RATE_16KHZ)
1214 snd_device = SND_DEVICE_IN_BT_SCO_MIC_WB;
1215 else
1216 snd_device = SND_DEVICE_IN_BT_SCO_MIC;
Eric Laurentb23d5282013-05-14 15:27:20 -07001217 } else if (out_device & AUDIO_DEVICE_OUT_AUX_DIGITAL) {
1218 snd_device = SND_DEVICE_IN_HDMI_MIC;
Apoorv Raghuvanshi5792d4b2013-10-07 18:40:05 -07001219 } else if (out_device & AUDIO_DEVICE_OUT_ANLG_DOCK_HEADSET ||
1220 out_device & AUDIO_DEVICE_OUT_DGTL_DOCK_HEADSET) {
1221 snd_device = SND_DEVICE_IN_USB_HEADSET_MIC;
Eric Laurentb23d5282013-05-14 15:27:20 -07001222 } else {
1223 ALOGE("%s: Unknown output device(s) %#x", __func__, out_device);
1224 ALOGW("%s: Using default handset-mic", __func__);
1225 snd_device = SND_DEVICE_IN_HANDSET_MIC;
1226 }
1227 }
1228exit:
1229 ALOGV("%s: exit: in_snd_device(%s)", __func__, device_table[snd_device]);
1230 return snd_device;
1231}
1232
1233int platform_set_hdmi_channels(void *platform, int channel_count)
1234{
1235 struct platform_data *my_data = (struct platform_data *)platform;
1236 struct audio_device *adev = my_data->adev;
1237 struct mixer_ctl *ctl;
1238 const char *channel_cnt_str = NULL;
1239 const char *mixer_ctl_name = "HDMI_RX Channels";
1240 switch (channel_count) {
1241 case 8:
1242 channel_cnt_str = "Eight"; break;
1243 case 7:
1244 channel_cnt_str = "Seven"; break;
1245 case 6:
1246 channel_cnt_str = "Six"; break;
1247 case 5:
1248 channel_cnt_str = "Five"; break;
1249 case 4:
1250 channel_cnt_str = "Four"; break;
1251 case 3:
1252 channel_cnt_str = "Three"; break;
1253 default:
1254 channel_cnt_str = "Two"; break;
1255 }
1256 ctl = mixer_get_ctl_by_name(adev->mixer, mixer_ctl_name);
1257 if (!ctl) {
1258 ALOGE("%s: Could not get ctl for mixer cmd - %s",
1259 __func__, mixer_ctl_name);
1260 return -EINVAL;
1261 }
1262 ALOGV("HDMI channel count: %s", channel_cnt_str);
1263 mixer_ctl_set_enum_by_string(ctl, channel_cnt_str);
1264 return 0;
1265}
1266
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001267int platform_edid_get_max_channels(void *platform)
Eric Laurentb23d5282013-05-14 15:27:20 -07001268{
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001269 struct platform_data *my_data = (struct platform_data *)platform;
1270 struct audio_device *adev = my_data->adev;
Eric Laurentb23d5282013-05-14 15:27:20 -07001271 char block[MAX_SAD_BLOCKS * SAD_BLOCK_SIZE];
1272 char *sad = block;
1273 int num_audio_blocks;
1274 int channel_count;
1275 int max_channels = 0;
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001276 int i, ret, count;
Eric Laurentb23d5282013-05-14 15:27:20 -07001277
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001278 struct mixer_ctl *ctl;
1279
1280 ctl = mixer_get_ctl_by_name(adev->mixer, AUDIO_DATA_BLOCK_MIXER_CTL);
1281 if (!ctl) {
1282 ALOGE("%s: Could not get ctl for mixer cmd - %s",
1283 __func__, AUDIO_DATA_BLOCK_MIXER_CTL);
Eric Laurentb23d5282013-05-14 15:27:20 -07001284 return 0;
1285 }
1286
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001287 mixer_ctl_update(ctl);
1288
1289 count = mixer_ctl_get_num_values(ctl);
Eric Laurentb23d5282013-05-14 15:27:20 -07001290
1291 /* Read SAD blocks, clamping the maximum size for safety */
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001292 if (count > (int)sizeof(block))
1293 count = (int)sizeof(block);
Eric Laurentb23d5282013-05-14 15:27:20 -07001294
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001295 ret = mixer_ctl_get_array(ctl, block, count);
1296 if (ret != 0) {
1297 ALOGE("%s: mixer_ctl_get_array() failed to get EDID info", __func__);
1298 return 0;
1299 }
Eric Laurentb23d5282013-05-14 15:27:20 -07001300
1301 /* Calculate the number of SAD blocks */
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001302 num_audio_blocks = count / SAD_BLOCK_SIZE;
Eric Laurentb23d5282013-05-14 15:27:20 -07001303
1304 for (i = 0; i < num_audio_blocks; i++) {
1305 /* Only consider LPCM blocks */
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001306 if ((sad[0] >> 3) != EDID_FORMAT_LPCM) {
1307 sad += 3;
Eric Laurentb23d5282013-05-14 15:27:20 -07001308 continue;
Haynes Mathew George47cd4cb2013-07-19 11:58:50 -07001309 }
Eric Laurentb23d5282013-05-14 15:27:20 -07001310
1311 channel_count = (sad[0] & 0x7) + 1;
1312 if (channel_count > max_channels)
1313 max_channels = channel_count;
1314
1315 /* Advance to next block */
1316 sad += 3;
1317 }
1318
1319 return max_channels;
1320}
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001321
1322static int platform_set_slowtalk(struct platform_data *my_data, bool state)
1323{
1324 int ret = 0;
1325 struct audio_device *adev = my_data->adev;
1326 struct mixer_ctl *ctl;
1327 const char *mixer_ctl_name = "Slowtalk Enable";
1328 uint32_t set_values[ ] = {0,
1329 ALL_SESSION_VSID};
1330
1331 set_values[0] = state;
1332 ctl = mixer_get_ctl_by_name(adev->mixer, mixer_ctl_name);
1333 if (!ctl) {
1334 ALOGE("%s: Could not get ctl for mixer cmd - %s",
1335 __func__, mixer_ctl_name);
1336 ret = -EINVAL;
1337 } else {
1338 ALOGV("Setting slowtalk state: %d", state);
1339 ret = mixer_ctl_set_array(ctl, set_values, ARRAY_SIZE(set_values));
1340 my_data->slowtalk = state;
1341 }
1342
Vidyakumar Athotad9d9ff32013-11-13 11:46:52 -08001343 if (my_data->csd != NULL) {
1344 ret = my_data->csd->slow_talk(ALL_SESSION_VSID, state);
1345 if (ret < 0) {
1346 ALOGE("%s: csd_client_disable_device, failed, error %d",
1347 __func__, ret);
1348 }
1349 }
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001350 return ret;
1351}
1352
1353int platform_set_parameters(void *platform, struct str_parms *parms)
1354{
1355 struct platform_data *my_data = (struct platform_data *)platform;
1356 char *str;
1357 char value[32];
1358 int val;
1359 int ret = 0;
1360
1361 ALOGV("%s: enter: %s", __func__, str_parms_to_str(parms));
1362
1363 ret = str_parms_get_int(parms, AUDIO_PARAMETER_KEY_BTSCO, &val);
1364 if (ret >= 0) {
1365 str_parms_del(parms, AUDIO_PARAMETER_KEY_BTSCO);
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001366 my_data->btsco_sample_rate = val;
Vidyakumar Athota67c42712013-12-04 14:03:06 -08001367 if (val == SAMPLE_RATE_16KHZ) {
1368 audio_route_apply_path(my_data->adev->audio_route,
1369 "bt-sco-wb-samplerate");
1370 audio_route_update_mixer(my_data->adev->audio_route);
1371 }
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001372 }
1373
1374 ret = str_parms_get_int(parms, AUDIO_PARAMETER_KEY_SLOWTALK, &val);
1375 if (ret >= 0) {
1376 str_parms_del(parms, AUDIO_PARAMETER_KEY_SLOWTALK);
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001377 ret = platform_set_slowtalk(my_data, val);
1378 if (ret)
1379 ALOGE("%s: Failed to set slow talk err: %d", __func__, ret);
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001380 }
1381
Ben Rombergerc1dc70d2013-12-19 15:11:17 -08001382 ret = str_parms_get_str(parms, AUDIO_PARAMETER_KEY_VOLUME_BOOST,
1383 value, sizeof(value));
1384 if (ret >= 0) {
1385 str_parms_del(parms, AUDIO_PARAMETER_KEY_VOLUME_BOOST);
1386
1387 if (my_data->acdb_reload_vocvoltable == NULL) {
1388 ALOGE("%s: acdb_reload_vocvoltable is NULL", __func__);
1389 } else if (!strcmp(value, "on")) {
1390 if (!my_data->acdb_reload_vocvoltable(VOICE_FEATURE_SET_VOLUME_BOOST)) {
1391 my_data->voice_feature_set = 1;
1392 }
1393 } else {
1394 if (!my_data->acdb_reload_vocvoltable(VOICE_FEATURE_SET_DEFAULT)) {
1395 my_data->voice_feature_set = 0;
1396 }
1397 }
1398 }
1399
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001400 ALOGV("%s: exit with code(%d)", __func__, ret);
1401 return ret;
1402}
1403
Vidyakumar Athota2850d532013-11-19 16:02:12 -08001404int platform_set_incall_recording_session_id(void *platform,
1405 uint32_t session_id, int rec_mode)
Shiv Maliyappanahallida107642013-10-17 11:16:13 -07001406{
1407 int ret = 0;
1408 struct platform_data *my_data = (struct platform_data *)platform;
1409 struct audio_device *adev = my_data->adev;
1410 struct mixer_ctl *ctl;
1411 const char *mixer_ctl_name = "Voc VSID";
1412 int num_ctl_values;
1413 int i;
1414
1415 ctl = mixer_get_ctl_by_name(adev->mixer, mixer_ctl_name);
1416 if (!ctl) {
1417 ALOGE("%s: Could not get ctl for mixer cmd - %s",
1418 __func__, mixer_ctl_name);
1419 ret = -EINVAL;
1420 } else {
1421 num_ctl_values = mixer_ctl_get_num_values(ctl);
1422 for (i = 0; i < num_ctl_values; i++) {
1423 if (mixer_ctl_set_value(ctl, i, session_id)) {
1424 ALOGV("Error: invalid session_id: %x", session_id);
1425 ret = -EINVAL;
1426 break;
1427 }
1428 }
1429 }
1430
Vidyakumar Athota2850d532013-11-19 16:02:12 -08001431 if (my_data->csd != NULL) {
1432 ret = my_data->csd->start_record(ALL_SESSION_VSID, rec_mode);
1433 if (ret < 0) {
1434 ALOGE("%s: csd_client_start_record failed, error %d",
1435 __func__, ret);
1436 }
1437 }
1438
1439 return ret;
1440}
1441
1442int platform_stop_incall_recording_usecase(void *platform)
1443{
1444 int ret = 0;
1445 struct platform_data *my_data = (struct platform_data *)platform;
1446
1447 if (my_data->csd != NULL) {
1448 ret = my_data->csd->stop_record(ALL_SESSION_VSID);
1449 if (ret < 0) {
1450 ALOGE("%s: csd_client_stop_record failed, error %d",
1451 __func__, ret);
1452 }
1453 }
1454
1455 return ret;
1456}
1457
1458int platform_start_incall_music_usecase(void *platform)
1459{
1460 int ret = 0;
1461 struct platform_data *my_data = (struct platform_data *)platform;
1462
1463 if (my_data->csd != NULL) {
1464 ret = my_data->csd->start_playback(ALL_SESSION_VSID);
1465 if (ret < 0) {
1466 ALOGE("%s: csd_client_start_playback failed, error %d",
1467 __func__, ret);
1468 }
1469 }
1470
1471 return ret;
1472}
1473
1474int platform_stop_incall_music_usecase(void *platform)
1475{
1476 int ret = 0;
1477 struct platform_data *my_data = (struct platform_data *)platform;
1478
1479 if (my_data->csd != NULL) {
1480 ret = my_data->csd->stop_playback(ALL_SESSION_VSID);
1481 if (ret < 0) {
1482 ALOGE("%s: csd_client_stop_playback failed, error %d",
1483 __func__, ret);
1484 }
1485 }
1486
Shiv Maliyappanahallida107642013-10-17 11:16:13 -07001487 return ret;
1488}
1489
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001490void platform_get_parameters(void *platform,
1491 struct str_parms *query,
1492 struct str_parms *reply)
1493{
1494 struct platform_data *my_data = (struct platform_data *)platform;
1495 char *str = NULL;
1496 char value[256] = {0};
1497 int ret;
1498 int fluence_type;
1499
1500 ret = str_parms_get_str(query, AUDIO_PARAMETER_KEY_FLUENCE_TYPE,
1501 value, sizeof(value));
1502 if (ret >= 0) {
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -08001503 if (my_data->fluence_type & FLUENCE_QUAD_MIC) {
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001504 strlcpy(value, "fluencepro", sizeof(value));
Ravi Kumar Alamandab034ddb2013-11-06 15:52:18 -08001505 } else if (my_data->fluence_type & FLUENCE_DUAL_MIC) {
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001506 strlcpy(value, "fluence", sizeof(value));
1507 } else {
1508 strlcpy(value, "none", sizeof(value));
1509 }
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001510
1511 str_parms_add_str(reply, AUDIO_PARAMETER_KEY_FLUENCE_TYPE, value);
1512 }
1513
Shiv Maliyappanahalli9d899292013-11-20 14:43:01 -08001514 memset(value, 0, sizeof(value));
1515 ret = str_parms_get_str(query, AUDIO_PARAMETER_KEY_SLOWTALK,
1516 value, sizeof(value));
1517 if (ret >= 0) {
1518 str_parms_add_int(reply, AUDIO_PARAMETER_KEY_SLOWTALK,
1519 my_data->slowtalk);
1520 }
1521
Ben Rombergerc1dc70d2013-12-19 15:11:17 -08001522 ret = str_parms_get_str(query, AUDIO_PARAMETER_KEY_VOLUME_BOOST,
1523 value, sizeof(value));
1524 if (ret >= 0) {
1525 if (my_data->voice_feature_set == VOICE_FEATURE_SET_VOLUME_BOOST) {
1526 strlcpy(value, "on", sizeof(value));
1527 } else {
1528 strlcpy(value, "off", sizeof(value));
1529 }
1530
1531 str_parms_add_str(reply, AUDIO_PARAMETER_KEY_VOLUME_BOOST, value);
1532 }
1533
Shiv Maliyappanahalli34b585f2013-10-01 15:49:05 -07001534 ALOGV("%s: exit: returns - %s", __func__, str_parms_to_str(reply));
1535}
1536
Haynes Mathew George7ff216f2013-09-11 19:51:41 -07001537/* Delay in Us */
1538int64_t platform_render_latency(audio_usecase_t usecase)
1539{
1540 switch (usecase) {
1541 case USECASE_AUDIO_PLAYBACK_DEEP_BUFFER:
1542 return DEEP_BUFFER_PLATFORM_DELAY;
1543 case USECASE_AUDIO_PLAYBACK_LOW_LATENCY:
1544 return LOW_LATENCY_PLATFORM_DELAY;
1545 default:
1546 return 0;
1547 }
1548}
Preetam Singh Ranawatde84f1a2013-11-01 14:58:16 -07001549
Mingming Yine62d7842013-10-25 16:26:03 -07001550int platform_update_usecase_from_source(int source, int usecase)
Preetam Singh Ranawatde84f1a2013-11-01 14:58:16 -07001551{
Mingming Yinab429782013-11-07 11:16:55 -08001552 ALOGV("%s: input source :%d", __func__, source);
1553 if(source == AUDIO_SOURCE_FM_RX_A2DP)
Mingming Yine62d7842013-10-25 16:26:03 -07001554 usecase = USECASE_AUDIO_RECORD_FM_VIRTUAL;
1555 return usecase;
Preetam Singh Ranawatde84f1a2013-11-01 14:58:16 -07001556}
Kiran Kandide144c82013-11-20 15:58:32 -08001557
1558bool platform_listen_update_status(snd_device_t snd_device)
1559{
1560 if ((snd_device >= SND_DEVICE_IN_BEGIN) &&
1561 (snd_device < SND_DEVICE_IN_END) &&
1562 (snd_device != SND_DEVICE_IN_CAPTURE_FM) &&
1563 (snd_device != SND_DEVICE_IN_CAPTURE_VI_FEEDBACK))
1564 return true;
1565 else
1566 return false;
1567}