waipio.c 52 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (c) 2016-2021, The Linux Foundation. All rights reserved.
  4. */
  5. #include <linux/clk.h>
  6. #include <linux/delay.h>
  7. #include <linux/gpio.h>
  8. #include <linux/of_gpio.h>
  9. #include <linux/platform_device.h>
  10. #include <linux/slab.h>
  11. #include <linux/io.h>
  12. #include <linux/module.h>
  13. #include <linux/input.h>
  14. #include <linux/of_device.h>
  15. #include <linux/soc/qcom/fsa4480-i2c.h>
  16. #include <linux/pm_qos.h>
  17. #include <sound/control.h>
  18. #include <sound/core.h>
  19. #include <sound/soc.h>
  20. #include <sound/soc-dapm.h>
  21. #include <sound/pcm.h>
  22. #include <sound/pcm_params.h>
  23. #include <sound/info.h>
  24. #include <soc/snd_event.h>
  25. #include <dsp/audio_prm.h>
  26. #include <soc/swr-common.h>
  27. #include <soc/soundwire.h>
  28. #include "device_event.h"
  29. #include "asoc/msm-cdc-pinctrl.h"
  30. #include "asoc/wcd-mbhc-v2.h"
  31. #include "codecs/wcd938x/wcd938x-mbhc.h"
  32. #include "codecs/wsa883x/wsa883x.h"
  33. #include "codecs/wcd938x/wcd938x.h"
  34. #include "codecs/lpass-cdc/lpass-cdc.h"
  35. #include <bindings/audio-codec-port-types.h>
  36. #include "codecs/lpass-cdc/lpass-cdc-wsa-macro.h"
  37. #include "waipio-port-config.h"
  38. #include "msm-audio-defs.h"
  39. #include "msm_common.h"
  40. #include "msm_dailink.h"
  41. #define DRV_NAME "waipio-asoc-snd"
  42. #define __CHIPSET__ "WAIPIO "
  43. #define MSM_DAILINK_NAME(name) (__CHIPSET__#name)
  44. #define WCD9XXX_MBHC_DEF_RLOADS 5
  45. #define WCD9XXX_MBHC_DEF_BUTTONS 8
  46. #define CODEC_EXT_CLK_RATE 9600000
  47. #define DEV_NAME_STR_LEN 32
  48. #define WCD_MBHC_HS_V_MAX 1600
  49. #define MSM_LL_QOS_VALUE 300 /* time in us to ensure LPM doesn't go in C3/C4 */
  50. #define WCN_CDC_SLIM_RX_CH_MAX 2
  51. #define WCN_CDC_SLIM_TX_CH_MAX 2
  52. #define WCN_CDC_SLIM_TX_CH_MAX_LITO 3
  53. /* Number of WSAs */
  54. #define MONO_SPEAKER 1
  55. #define STEREO_SPEAKER 2
  56. #define QUAD_SPEAKER 4
  57. struct msm_asoc_mach_data {
  58. struct snd_info_entry *codec_root;
  59. struct msm_common_pdata *common_pdata;
  60. int usbc_en2_gpio; /* used by gpio driver API */
  61. struct device_node *dmic01_gpio_p; /* used by pinctrl API */
  62. struct device_node *dmic23_gpio_p; /* used by pinctrl API */
  63. struct device_node *dmic45_gpio_p; /* used by pinctrl API */
  64. struct pinctrl *usbc_en2_gpio_p; /* used by pinctrl API */
  65. bool is_afe_config_done;
  66. struct device_node *fsa_handle;
  67. struct clk *lpass_audio_hw_vote;
  68. int core_audio_vote_count;
  69. u32 wsa_max_devs;
  70. int wcd_disabled;
  71. int (*get_dev_num)(struct snd_soc_component *);
  72. int backend_used;
  73. struct prm_earpa_hw_intf_config upd_config;
  74. };
  75. static bool is_initial_boot;
  76. static bool codec_reg_done;
  77. static struct snd_soc_card snd_soc_card_waipio_msm;
  78. static int dmic_0_1_gpio_cnt;
  79. static int dmic_2_3_gpio_cnt;
  80. static int dmic_4_5_gpio_cnt;
  81. static void *def_wcd_mbhc_cal(void);
  82. static int msm_rx_tx_codec_init(struct snd_soc_pcm_runtime*);
  83. static int msm_int_wsa_init(struct snd_soc_pcm_runtime*);
  84. /*
  85. * Need to report LINEIN
  86. * if R/L channel impedance is larger than 5K ohm
  87. */
  88. static struct wcd_mbhc_config wcd_mbhc_cfg = {
  89. .read_fw_bin = false,
  90. .calibration = NULL,
  91. .detect_extn_cable = true,
  92. .mono_stero_detection = false,
  93. .swap_gnd_mic = NULL,
  94. .hs_ext_micbias = true,
  95. .key_code[0] = KEY_MEDIA,
  96. .key_code[1] = KEY_VOICECOMMAND,
  97. .key_code[2] = KEY_VOLUMEUP,
  98. .key_code[3] = KEY_VOLUMEDOWN,
  99. .key_code[4] = 0,
  100. .key_code[5] = 0,
  101. .key_code[6] = 0,
  102. .key_code[7] = 0,
  103. .linein_th = 5000,
  104. .moisture_en = false,
  105. .mbhc_micbias = MIC_BIAS_2,
  106. .anc_micbias = MIC_BIAS_2,
  107. .enable_anc_mic_detect = false,
  108. .moisture_duty_cycle_en = true,
  109. };
  110. /* set audio task affinity to core 1 & 2 */
  111. static const unsigned int audio_core_list[] = {1, 2};
  112. static cpumask_t audio_cpu_map = CPU_MASK_NONE;
  113. static struct dev_pm_qos_request *msm_audio_req = NULL;
  114. static void msm_audio_add_qos_request()
  115. {
  116. int i;
  117. int cpu = 0;
  118. msm_audio_req = kzalloc(sizeof(struct dev_pm_qos_request) * NR_CPUS,
  119. GFP_KERNEL);
  120. if (!msm_audio_req) {
  121. pr_err("%s failed to alloc mem for qos req.\n", __func__);
  122. return;
  123. }
  124. for (i = 0; i < ARRAY_SIZE(audio_core_list); i++) {
  125. if (audio_core_list[i] >= NR_CPUS)
  126. pr_err("%s incorrect cpu id: %d specified.\n", __func__, audio_core_list[i]);
  127. else
  128. cpumask_set_cpu(audio_core_list[i], &audio_cpu_map);
  129. }
  130. for_each_cpu(cpu, &audio_cpu_map) {
  131. dev_pm_qos_add_request(get_cpu_device(cpu),
  132. &msm_audio_req[cpu],
  133. DEV_PM_QOS_RESUME_LATENCY,
  134. PM_QOS_CPU_LATENCY_DEFAULT_VALUE);
  135. pr_debug("%s set cpu affinity to core %d.\n", __func__, cpu);
  136. }
  137. }
  138. static void msm_audio_remove_qos_request()
  139. {
  140. int cpu = 0;
  141. if (msm_audio_req) {
  142. for_each_cpu(cpu, &audio_cpu_map) {
  143. dev_pm_qos_remove_request(
  144. &msm_audio_req[cpu]);
  145. pr_debug("%s remove cpu affinity of core %d.\n", __func__, cpu);
  146. }
  147. kfree(msm_audio_req);
  148. }
  149. }
  150. static bool msm_usbc_swap_gnd_mic(struct snd_soc_component *component, bool active)
  151. {
  152. struct snd_soc_card *card = component->card;
  153. struct msm_asoc_mach_data *pdata =
  154. snd_soc_card_get_drvdata(card);
  155. if (!pdata->fsa_handle)
  156. return false;
  157. return fsa4480_switch_event(pdata->fsa_handle, FSA_MIC_GND_SWAP);
  158. }
  159. static void msm_parse_upd_configuration(struct platform_device *pdev,
  160. struct msm_asoc_mach_data *pdata)
  161. {
  162. int ret = 0;
  163. u32 dt_values[2];
  164. if (!pdev || !pdata)
  165. return;
  166. ret = of_property_read_string(pdev->dev.of_node,
  167. "qcom,upd_backends_used", &pdata->upd_config.backend_used);
  168. if (ret) {
  169. pr_debug("%s:could not find %s entry in dt\n",
  170. __func__, "qcom,upd_backends_used");
  171. return;
  172. }
  173. if (!strcmp(pdata->upd_config.backend_used, "wsa"))
  174. pdata->get_dev_num = wsa883x_codec_get_dev_num;
  175. else
  176. pdata->get_dev_num = wcd938x_codec_get_dev_num;
  177. ret = of_property_read_u32_array(pdev->dev.of_node,
  178. "qcom,upd_lpass_reg_addr", dt_values, MAX_EARPA_REG);
  179. if (ret) {
  180. pr_debug("%s: could not find %s entry in dt\n",
  181. __func__, "qcom,upd_lpass_reg_addr");
  182. return;
  183. } else {
  184. pdata->upd_config.ear_pa_hw_reg_cfg.lpass_cdc_rx0_rx_path_ctl_phy_addr =
  185. dt_values[0];
  186. pdata->upd_config.ear_pa_hw_reg_cfg.lpass_wr_fifo_reg_phy_addr =
  187. dt_values[1];
  188. }
  189. ret = of_property_read_u32(pdev->dev.of_node,
  190. "qcom,upd_ear_pa_reg_addr", &pdata->upd_config.ear_pa_pkd_reg_addr);
  191. if (ret) {
  192. pr_debug("%s: could not find %s entry in dt\n",
  193. __func__, "qcom,upd_ear_pa_reg_addr");
  194. }
  195. }
  196. static void msm_set_upd_config(struct snd_soc_pcm_runtime *rtd)
  197. {
  198. int val1 = 0, val2 = 0, ret = 0;
  199. u8 dev_num = 0;
  200. char cdc_name[DEV_NAME_STR_LEN];
  201. struct snd_soc_component *component = NULL;
  202. struct msm_asoc_mach_data *pdata = NULL;
  203. if (!rtd) {
  204. pr_err("%s: rtd is NULL\n", __func__);
  205. return;
  206. }
  207. pdata = snd_soc_card_get_drvdata(rtd->card);
  208. if (!pdata) {
  209. pr_err("%s: pdata is NULL\n", __func__);
  210. return;
  211. }
  212. if (!pdata->get_dev_num) {
  213. pr_err("%s: get_dev_num is NULL\n", __func__);
  214. return;
  215. }
  216. if (!pdata->upd_config.ear_pa_hw_reg_cfg.lpass_cdc_rx0_rx_path_ctl_phy_addr ||
  217. !pdata->upd_config.ear_pa_hw_reg_cfg.lpass_wr_fifo_reg_phy_addr ||
  218. !pdata->upd_config.ear_pa_pkd_reg_addr) {
  219. pr_err("%s: upd static configuration is not set\n", __func__);
  220. return;
  221. }
  222. memset(cdc_name, '\0', DEV_NAME_STR_LEN);
  223. if (!strcmp(pdata->upd_config.backend_used, "wsa")) {
  224. if (pdata->wsa_max_devs > 0)
  225. memcpy(cdc_name, "wsa-codec.1", strlen("wsa-codec.1"));
  226. }
  227. else
  228. memcpy(cdc_name, WCD938X_DRV_NAME, sizeof(WCD938X_DRV_NAME));
  229. component = snd_soc_rtdcom_lookup(rtd, cdc_name);
  230. if (!component) {
  231. pr_err("%s: %s component is NULL\n", __func__,
  232. cdc_name);
  233. return;
  234. }
  235. dev_num = pdata->get_dev_num(component);
  236. if (dev_num < 0 || dev_num > 6) {
  237. pr_err("%s: invalid slave dev num : %d\n", __func__,
  238. dev_num);
  239. return;
  240. }
  241. pdata->upd_config.ear_pa_pkd_cfg.ear_pa_enable_pkd_reg_addr =
  242. pdata->upd_config.ear_pa_pkd_reg_addr & 0xFFFF;
  243. pdata->upd_config.ear_pa_pkd_cfg.ear_pa_disable_pkd_reg_addr =
  244. pdata->upd_config.ear_pa_pkd_reg_addr & 0xFFFF;
  245. val1 = val2 = 0;
  246. /* bits 16:19 carry command id */
  247. val1 |= 1 << 16;
  248. /* bits 20:23 carry swr device number */
  249. val1 |= dev_num << 20;
  250. /*
  251. * bits 24:31 carry 8 bit data to disable or enable ear pa
  252. * for wcd 7bit is global enable bit - 1 -enable. 0 - disable
  253. * for wsa 0bit is global enable bit - 1 -enable, 0 - disable
  254. */
  255. val2 = val1;
  256. if (!strcmp(pdata->upd_config.backend_used, "wsa"))
  257. val1 |= 1 << 24;
  258. else
  259. val1 |= 1 << 31;
  260. pdata->upd_config.ear_pa_pkd_cfg.ear_pa_enable_pkd_reg_addr |= val1;
  261. pdata->upd_config.ear_pa_pkd_cfg.ear_pa_disable_pkd_reg_addr |= val2;
  262. ret = audio_prm_set_cdc_earpa_duty_cycling_req(&pdata->upd_config, 1);
  263. if (ret < 0)
  264. pr_err("%s: upd cdc duty cycling registration failed\n", __func__);
  265. }
  266. static struct snd_soc_ops msm_common_be_ops = {
  267. .hw_params = msm_common_snd_hw_params,
  268. .startup = msm_common_snd_startup,
  269. .shutdown = msm_common_snd_shutdown,
  270. };
  271. static int msm_dmic_event(struct snd_soc_dapm_widget *w,
  272. struct snd_kcontrol *kcontrol, int event)
  273. {
  274. struct msm_asoc_mach_data *pdata = NULL;
  275. struct snd_soc_component *component = snd_soc_dapm_to_component(w->dapm);
  276. int ret = 0;
  277. u32 dmic_idx;
  278. int *dmic_gpio_cnt;
  279. struct device_node *dmic_gpio;
  280. char *wname;
  281. wname = strpbrk(w->name, "012345");
  282. if (!wname) {
  283. dev_err(component->dev, "%s: widget not found\n", __func__);
  284. return -EINVAL;
  285. }
  286. ret = kstrtouint(wname, 10, &dmic_idx);
  287. if (ret < 0) {
  288. dev_err(component->dev, "%s: Invalid DMIC line on the codec\n",
  289. __func__);
  290. return -EINVAL;
  291. }
  292. pdata = snd_soc_card_get_drvdata(component->card);
  293. switch (dmic_idx) {
  294. case 0:
  295. case 1:
  296. dmic_gpio_cnt = &dmic_0_1_gpio_cnt;
  297. dmic_gpio = pdata->dmic01_gpio_p;
  298. break;
  299. case 2:
  300. case 3:
  301. dmic_gpio_cnt = &dmic_2_3_gpio_cnt;
  302. dmic_gpio = pdata->dmic23_gpio_p;
  303. break;
  304. case 4:
  305. case 5:
  306. dmic_gpio_cnt = &dmic_4_5_gpio_cnt;
  307. dmic_gpio = pdata->dmic45_gpio_p;
  308. break;
  309. default:
  310. dev_err(component->dev, "%s: Invalid DMIC Selection\n",
  311. __func__);
  312. return -EINVAL;
  313. }
  314. dev_dbg(component->dev, "%s: event %d DMIC%d dmic_gpio_cnt %d\n",
  315. __func__, event, dmic_idx, *dmic_gpio_cnt);
  316. switch (event) {
  317. case SND_SOC_DAPM_PRE_PMU:
  318. (*dmic_gpio_cnt)++;
  319. if (*dmic_gpio_cnt == 1) {
  320. ret = msm_cdc_pinctrl_select_active_state(
  321. dmic_gpio);
  322. if (ret < 0) {
  323. pr_err("%s: gpio set cannot be activated %sd",
  324. __func__, "dmic_gpio");
  325. return ret;
  326. }
  327. }
  328. break;
  329. case SND_SOC_DAPM_POST_PMD:
  330. (*dmic_gpio_cnt)--;
  331. if (*dmic_gpio_cnt == 0) {
  332. ret = msm_cdc_pinctrl_select_sleep_state(
  333. dmic_gpio);
  334. if (ret < 0) {
  335. pr_err("%s: gpio set cannot be de-activated %sd",
  336. __func__, "dmic_gpio");
  337. return ret;
  338. }
  339. }
  340. break;
  341. default:
  342. pr_err("%s: invalid DAPM event %d\n", __func__, event);
  343. return -EINVAL;
  344. }
  345. return 0;
  346. }
  347. static const struct snd_soc_dapm_widget msm_int_dapm_widgets[] = {
  348. SND_SOC_DAPM_MIC("Analog Mic1", NULL),
  349. SND_SOC_DAPM_MIC("Analog Mic2", NULL),
  350. SND_SOC_DAPM_MIC("Analog Mic3", NULL),
  351. SND_SOC_DAPM_MIC("Analog Mic4", NULL),
  352. SND_SOC_DAPM_MIC("Analog Mic5", NULL),
  353. SND_SOC_DAPM_MIC("Digital Mic0", msm_dmic_event),
  354. SND_SOC_DAPM_MIC("Digital Mic1", msm_dmic_event),
  355. SND_SOC_DAPM_MIC("Digital Mic2", msm_dmic_event),
  356. SND_SOC_DAPM_MIC("Digital Mic3", msm_dmic_event),
  357. SND_SOC_DAPM_MIC("Digital Mic4", msm_dmic_event),
  358. SND_SOC_DAPM_MIC("Digital Mic5", msm_dmic_event),
  359. SND_SOC_DAPM_MIC("Digital Mic6", NULL),
  360. SND_SOC_DAPM_MIC("Digital Mic7", NULL),
  361. };
  362. static int msm_wcn_init(struct snd_soc_pcm_runtime *rtd)
  363. {
  364. unsigned int rx_ch[WCN_CDC_SLIM_RX_CH_MAX] = {157, 158};
  365. unsigned int tx_ch[WCN_CDC_SLIM_TX_CH_MAX] = {159, 160};
  366. struct snd_soc_dai *codec_dai = asoc_rtd_to_codec(rtd, 0);
  367. int ret = 0;
  368. ret = snd_soc_dai_set_channel_map(codec_dai, ARRAY_SIZE(tx_ch),
  369. tx_ch, ARRAY_SIZE(rx_ch), rx_ch);
  370. if (ret)
  371. return ret;
  372. msm_common_dai_link_init(rtd);
  373. return ret;
  374. }
  375. static struct snd_info_entry *msm_snd_info_create_subdir(struct module *mod,
  376. const char *name,
  377. struct snd_info_entry *parent)
  378. {
  379. struct snd_info_entry *entry;
  380. entry = snd_info_create_module_entry(mod, name, parent);
  381. if (!entry)
  382. return NULL;
  383. entry->mode = S_IFDIR | 0555;
  384. if (snd_info_register(entry) < 0) {
  385. snd_info_free_entry(entry);
  386. return NULL;
  387. }
  388. return entry;
  389. }
  390. static void *def_wcd_mbhc_cal(void)
  391. {
  392. void *wcd_mbhc_cal;
  393. struct wcd_mbhc_btn_detect_cfg *btn_cfg;
  394. u16 *btn_high;
  395. wcd_mbhc_cal = kzalloc(WCD_MBHC_CAL_SIZE(WCD_MBHC_DEF_BUTTONS,
  396. WCD9XXX_MBHC_DEF_RLOADS), GFP_KERNEL);
  397. if (!wcd_mbhc_cal)
  398. return NULL;
  399. WCD_MBHC_CAL_PLUG_TYPE_PTR(wcd_mbhc_cal)->v_hs_max = WCD_MBHC_HS_V_MAX;
  400. WCD_MBHC_CAL_BTN_DET_PTR(wcd_mbhc_cal)->num_btn = WCD_MBHC_DEF_BUTTONS;
  401. btn_cfg = WCD_MBHC_CAL_BTN_DET_PTR(wcd_mbhc_cal);
  402. btn_high = ((void *)&btn_cfg->_v_btn_low) +
  403. (sizeof(btn_cfg->_v_btn_low[0]) * btn_cfg->num_btn);
  404. btn_high[0] = 75;
  405. btn_high[1] = 150;
  406. btn_high[2] = 237;
  407. btn_high[3] = 500;
  408. btn_high[4] = 500;
  409. btn_high[5] = 500;
  410. btn_high[6] = 500;
  411. btn_high[7] = 500;
  412. return wcd_mbhc_cal;
  413. }
  414. /* Digital audio interface glue - connects codec <---> CPU */
  415. static struct snd_soc_dai_link msm_common_be_dai_links[] = {
  416. /* Proxy Tx BACK END DAI Link */
  417. {
  418. .name = LPASS_BE_RT_PROXY_PCM_TX,
  419. .stream_name = LPASS_BE_RT_PROXY_PCM_TX,
  420. .capture_only = 1,
  421. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  422. SND_SOC_DPCM_TRIGGER_POST},
  423. .ignore_suspend = 1,
  424. .ops = &msm_common_be_ops,
  425. SND_SOC_DAILINK_REG(proxy_tx),
  426. },
  427. /* Proxy Rx BACK END DAI Link */
  428. {
  429. .name = LPASS_BE_RT_PROXY_PCM_RX,
  430. .stream_name = LPASS_BE_RT_PROXY_PCM_RX,
  431. .playback_only = 1,
  432. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  433. SND_SOC_DPCM_TRIGGER_POST},
  434. .ignore_pmdown_time = 1,
  435. .ignore_suspend = 1,
  436. .ops = &msm_common_be_ops,
  437. SND_SOC_DAILINK_REG(proxy_rx),
  438. },
  439. {
  440. .name = LPASS_BE_USB_AUDIO_RX,
  441. .stream_name = LPASS_BE_USB_AUDIO_RX,
  442. .playback_only = 1,
  443. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  444. SND_SOC_DPCM_TRIGGER_POST},
  445. .ignore_pmdown_time = 1,
  446. .ignore_suspend = 1,
  447. .ops = &msm_common_be_ops,
  448. SND_SOC_DAILINK_REG(usb_audio_rx),
  449. },
  450. {
  451. .name = LPASS_BE_USB_AUDIO_TX,
  452. .stream_name = LPASS_BE_USB_AUDIO_TX,
  453. .capture_only = 1,
  454. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  455. SND_SOC_DPCM_TRIGGER_POST},
  456. .ignore_suspend = 1,
  457. .ops = &msm_common_be_ops,
  458. SND_SOC_DAILINK_REG(usb_audio_tx),
  459. },
  460. };
  461. static struct snd_soc_dai_link msm_wcn_be_dai_links[] = {
  462. {
  463. .name = LPASS_BE_SLIMBUS_7_RX,
  464. .stream_name = LPASS_BE_SLIMBUS_7_RX,
  465. .playback_only = 1,
  466. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  467. SND_SOC_DPCM_TRIGGER_POST},
  468. .init = &msm_wcn_init,
  469. .ops = &msm_common_be_ops,
  470. /* dai link has playback support */
  471. .ignore_pmdown_time = 1,
  472. .ignore_suspend = 1,
  473. SND_SOC_DAILINK_REG(slimbus_7_rx),
  474. },
  475. {
  476. .name = LPASS_BE_SLIMBUS_7_TX,
  477. .stream_name = LPASS_BE_SLIMBUS_7_TX,
  478. .capture_only = 1,
  479. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  480. SND_SOC_DPCM_TRIGGER_POST},
  481. .ops = &msm_common_be_ops,
  482. .ignore_suspend = 1,
  483. SND_SOC_DAILINK_REG(slimbus_7_tx),
  484. },
  485. };
  486. static struct snd_soc_dai_link ext_disp_be_dai_link[] = {
  487. /* DISP PORT BACK END DAI Link */
  488. {
  489. .name = LPASS_BE_DISPLAY_PORT_RX,
  490. .stream_name = LPASS_BE_DISPLAY_PORT_RX,
  491. .playback_only = 1,
  492. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  493. SND_SOC_DPCM_TRIGGER_POST},
  494. .ignore_pmdown_time = 1,
  495. .ignore_suspend = 1,
  496. SND_SOC_DAILINK_REG(display_port),
  497. },
  498. };
  499. static struct snd_soc_dai_link msm_wsa_cdc_dma_be_dai_links[] = {
  500. /* WSA CDC DMA Backend DAI Links */
  501. {
  502. .name = LPASS_BE_WSA_CDC_DMA_RX_0,
  503. .stream_name = LPASS_BE_WSA_CDC_DMA_RX_0,
  504. .playback_only = 1,
  505. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  506. SND_SOC_DPCM_TRIGGER_POST},
  507. .ignore_pmdown_time = 1,
  508. .ignore_suspend = 1,
  509. .ops = &msm_common_be_ops,
  510. SND_SOC_DAILINK_REG(wsa_dma_rx0),
  511. .init = &msm_int_wsa_init,
  512. },
  513. {
  514. .name = LPASS_BE_WSA_CDC_DMA_RX_1,
  515. .stream_name = LPASS_BE_WSA_CDC_DMA_RX_1,
  516. .playback_only = 1,
  517. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  518. SND_SOC_DPCM_TRIGGER_POST},
  519. .ignore_pmdown_time = 1,
  520. .ignore_suspend = 1,
  521. .ops = &msm_common_be_ops,
  522. SND_SOC_DAILINK_REG(wsa_dma_rx1),
  523. },
  524. {
  525. .name = LPASS_BE_WSA_CDC_DMA_TX_1,
  526. .stream_name = LPASS_BE_WSA_CDC_DMA_TX_1,
  527. .capture_only = 1,
  528. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  529. SND_SOC_DPCM_TRIGGER_POST},
  530. .ignore_suspend = 1,
  531. .ops = &msm_common_be_ops,
  532. SND_SOC_DAILINK_REG(wsa_dma_tx1),
  533. },
  534. {
  535. .name = LPASS_BE_WSA_CDC_DMA_TX_0,
  536. .stream_name = LPASS_BE_WSA_CDC_DMA_TX_0,
  537. .capture_only = 1,
  538. .ignore_suspend = 1,
  539. .ops = &msm_common_be_ops,
  540. /* .no_host_mode = SND_SOC_DAI_LINK_NO_HOST, */
  541. SND_SOC_DAILINK_REG(vi_feedback),
  542. },
  543. {
  544. .name = LPASS_BE_WSA_CDC_DMA_RX_0_VIRT,
  545. .stream_name = LPASS_BE_WSA_CDC_DMA_RX_0_VIRT,
  546. .playback_only = 1,
  547. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  548. SND_SOC_DPCM_TRIGGER_POST},
  549. .ignore_pmdown_time = 1,
  550. .ignore_suspend = 1,
  551. .ops = &msm_common_be_ops,
  552. SND_SOC_DAILINK_REG(wsa_dma_rx0),
  553. .init = &msm_int_wsa_init,
  554. },
  555. };
  556. static struct snd_soc_dai_link msm_wsa2_cdc_dma_be_dai_links[] = {
  557. /* WSA2 CDC DMA Backend DAI Links */
  558. {
  559. .name = LPASS_BE_WSA2_CDC_DMA_RX_0,
  560. .stream_name = LPASS_BE_WSA2_CDC_DMA_RX_0,
  561. .playback_only = 1,
  562. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  563. SND_SOC_DPCM_TRIGGER_POST},
  564. .ignore_pmdown_time = 1,
  565. .ignore_suspend = 1,
  566. .ops = &msm_common_be_ops,
  567. SND_SOC_DAILINK_REG(wsa2_dma_rx0),
  568. },
  569. {
  570. .name = LPASS_BE_WSA2_CDC_DMA_RX_1,
  571. .stream_name = LPASS_BE_WSA2_CDC_DMA_RX_1,
  572. .playback_only = 1,
  573. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  574. SND_SOC_DPCM_TRIGGER_POST},
  575. .ignore_pmdown_time = 1,
  576. .ignore_suspend = 1,
  577. .ops = &msm_common_be_ops,
  578. SND_SOC_DAILINK_REG(wsa2_dma_rx1),
  579. },
  580. {
  581. .name = LPASS_BE_WSA2_CDC_DMA_TX_1,
  582. .stream_name = LPASS_BE_WSA2_CDC_DMA_TX_1,
  583. .capture_only = 1,
  584. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  585. SND_SOC_DPCM_TRIGGER_POST},
  586. .ignore_suspend = 1,
  587. .ops = &msm_common_be_ops,
  588. SND_SOC_DAILINK_REG(wsa2_dma_tx1),
  589. },
  590. {
  591. .name = LPASS_BE_WSA2_CDC_DMA_TX_0,
  592. .stream_name = LPASS_BE_WSA2_CDC_DMA_TX_0,
  593. .capture_only = 1,
  594. .ignore_suspend = 1,
  595. .ops = &msm_common_be_ops,
  596. /* .no_host_mode = SND_SOC_DAI_LINK_NO_HOST, */
  597. SND_SOC_DAILINK_REG(wsa2_vi_feedback),
  598. },
  599. };
  600. static struct snd_soc_dai_link msm_wsa_wsa2_cdc_dma_be_dai_links[] = {
  601. /* WSA CDC DMA Backend DAI Links */
  602. {
  603. .name = LPASS_BE_WSA_CDC_DMA_RX_0,
  604. .stream_name = LPASS_BE_WSA_CDC_DMA_RX_0,
  605. .playback_only = 1,
  606. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  607. SND_SOC_DPCM_TRIGGER_POST},
  608. .ignore_pmdown_time = 1,
  609. .ignore_suspend = 1,
  610. .ops = &msm_common_be_ops,
  611. SND_SOC_DAILINK_REG(wsa_wsa2_dma_rx0),
  612. .init = &msm_int_wsa_init,
  613. },
  614. {
  615. .name = LPASS_BE_WSA_CDC_DMA_RX_1,
  616. .stream_name = LPASS_BE_WSA_CDC_DMA_RX_1,
  617. .playback_only = 1,
  618. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  619. SND_SOC_DPCM_TRIGGER_POST},
  620. .ignore_pmdown_time = 1,
  621. .ignore_suspend = 1,
  622. .ops = &msm_common_be_ops,
  623. SND_SOC_DAILINK_REG(wsa_wsa2_dma_rx1),
  624. },
  625. {
  626. .name = LPASS_BE_WSA_CDC_DMA_TX_1,
  627. .stream_name = LPASS_BE_WSA_CDC_DMA_TX_1,
  628. .capture_only = 1,
  629. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  630. SND_SOC_DPCM_TRIGGER_POST},
  631. .ignore_suspend = 1,
  632. .ops = &msm_common_be_ops,
  633. SND_SOC_DAILINK_REG(wsa_wsa2_dma_tx1),
  634. },
  635. {
  636. .name = LPASS_BE_WSA_CDC_DMA_TX_0,
  637. .stream_name = LPASS_BE_WSA_CDC_DMA_TX_0,
  638. .capture_only = 1,
  639. .ignore_suspend = 1,
  640. .ops = &msm_common_be_ops,
  641. /* .no_host_mode = SND_SOC_DAI_LINK_NO_HOST, */
  642. SND_SOC_DAILINK_REG(wsa_wsa2_vi_feedback),
  643. },
  644. };
  645. static struct snd_soc_dai_link msm_rx_tx_cdc_dma_be_dai_links[] = {
  646. /* RX CDC DMA Backend DAI Links */
  647. {
  648. .name = LPASS_BE_RX_CDC_DMA_RX_0,
  649. .stream_name = LPASS_BE_RX_CDC_DMA_RX_0,
  650. .playback_only = 1,
  651. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  652. SND_SOC_DPCM_TRIGGER_POST},
  653. .ignore_pmdown_time = 1,
  654. .ignore_suspend = 1,
  655. .ops = &msm_common_be_ops,
  656. SND_SOC_DAILINK_REG(rx_dma_rx0),
  657. .init = &msm_rx_tx_codec_init,
  658. },
  659. {
  660. .name = LPASS_BE_RX_CDC_DMA_RX_1,
  661. .stream_name = LPASS_BE_RX_CDC_DMA_RX_1,
  662. .playback_only = 1,
  663. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  664. SND_SOC_DPCM_TRIGGER_POST},
  665. .ignore_pmdown_time = 1,
  666. .ignore_suspend = 1,
  667. .ops = &msm_common_be_ops,
  668. SND_SOC_DAILINK_REG(rx_dma_rx1),
  669. },
  670. {
  671. .name = LPASS_BE_RX_CDC_DMA_RX_2,
  672. .stream_name = LPASS_BE_RX_CDC_DMA_RX_2,
  673. .playback_only = 1,
  674. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  675. SND_SOC_DPCM_TRIGGER_POST},
  676. .ignore_pmdown_time = 1,
  677. .ignore_suspend = 1,
  678. .ops = &msm_common_be_ops,
  679. SND_SOC_DAILINK_REG(rx_dma_rx2),
  680. },
  681. {
  682. .name = LPASS_BE_RX_CDC_DMA_RX_3,
  683. .stream_name = LPASS_BE_RX_CDC_DMA_RX_3,
  684. .playback_only = 1,
  685. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  686. SND_SOC_DPCM_TRIGGER_POST},
  687. .ignore_pmdown_time = 1,
  688. .ignore_suspend = 1,
  689. .ops = &msm_common_be_ops,
  690. SND_SOC_DAILINK_REG(rx_dma_rx3),
  691. },
  692. {
  693. .name = LPASS_BE_RX_CDC_DMA_RX_5,
  694. .stream_name = LPASS_BE_RX_CDC_DMA_RX_5,
  695. .playback_only = 1,
  696. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  697. SND_SOC_DPCM_TRIGGER_POST},
  698. .ignore_pmdown_time = 1,
  699. .ignore_suspend = 1,
  700. .ops = &msm_common_be_ops,
  701. SND_SOC_DAILINK_REG(rx_dma_rx5),
  702. },
  703. {
  704. .name = LPASS_BE_RX_CDC_DMA_RX_6,
  705. .stream_name = LPASS_BE_RX_CDC_DMA_RX_6,
  706. .playback_only = 1,
  707. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  708. SND_SOC_DPCM_TRIGGER_POST},
  709. .ignore_pmdown_time = 1,
  710. .ignore_suspend = 1,
  711. .ops = &msm_common_be_ops,
  712. SND_SOC_DAILINK_REG(rx_dma_rx6),
  713. },
  714. /* TX CDC DMA Backend DAI Links */
  715. {
  716. .name = LPASS_BE_TX_CDC_DMA_TX_3,
  717. .stream_name = LPASS_BE_TX_CDC_DMA_TX_3,
  718. .capture_only = 1,
  719. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  720. SND_SOC_DPCM_TRIGGER_POST},
  721. .ignore_suspend = 1,
  722. .ops = &msm_common_be_ops,
  723. SND_SOC_DAILINK_REG(tx_dma_tx3),
  724. },
  725. {
  726. .name = LPASS_BE_TX_CDC_DMA_TX_4,
  727. .stream_name = LPASS_BE_TX_CDC_DMA_TX_4,
  728. .capture_only = 1,
  729. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  730. SND_SOC_DPCM_TRIGGER_POST},
  731. .ignore_suspend = 1,
  732. .ops = &msm_common_be_ops,
  733. SND_SOC_DAILINK_REG(tx_dma_tx4),
  734. },
  735. };
  736. static struct snd_soc_dai_link msm_va_cdc_dma_be_dai_links[] = {
  737. {
  738. .name = LPASS_BE_VA_CDC_DMA_TX_0,
  739. .stream_name = LPASS_BE_VA_CDC_DMA_TX_0,
  740. .capture_only = 1,
  741. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  742. SND_SOC_DPCM_TRIGGER_POST},
  743. .ignore_suspend = 1,
  744. .ops = &msm_common_be_ops,
  745. SND_SOC_DAILINK_REG(va_dma_tx0),
  746. },
  747. {
  748. .name = LPASS_BE_VA_CDC_DMA_TX_1,
  749. .stream_name = LPASS_BE_VA_CDC_DMA_TX_1,
  750. .capture_only = 1,
  751. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  752. SND_SOC_DPCM_TRIGGER_POST},
  753. .ignore_suspend = 1,
  754. .ops = &msm_common_be_ops,
  755. SND_SOC_DAILINK_REG(va_dma_tx1),
  756. },
  757. {
  758. .name = LPASS_BE_VA_CDC_DMA_TX_2,
  759. .stream_name = LPASS_BE_VA_CDC_DMA_TX_2,
  760. .capture_only = 1,
  761. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  762. SND_SOC_DPCM_TRIGGER_POST},
  763. .ignore_suspend = 1,
  764. .ops = &msm_common_be_ops,
  765. SND_SOC_DAILINK_REG(va_dma_tx2),
  766. },
  767. };
  768. /*
  769. * I2S interface pinctrl mapping
  770. * ------------------------------------
  771. * Primary - pri_mi2s
  772. * Secondary - lpi_i2s3
  773. * Tertiary - tert_mi2s
  774. * Quaternary - quat_mi2s (lpi_i2s0)
  775. * Quinary - lpi_i2s1
  776. * Senary - lpi_i2s2
  777. * ------------------------------------
  778. */
  779. static struct snd_soc_dai_link msm_mi2s_dai_links[] = {
  780. {
  781. .name = LPASS_BE_PRI_MI2S_RX,
  782. .stream_name = LPASS_BE_PRI_MI2S_RX,
  783. .playback_only = 1,
  784. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  785. SND_SOC_DPCM_TRIGGER_POST},
  786. .ops = &msm_common_be_ops,
  787. .ignore_suspend = 1,
  788. .ignore_pmdown_time = 1,
  789. SND_SOC_DAILINK_REG(pri_mi2s_rx),
  790. },
  791. {
  792. .name = LPASS_BE_PRI_MI2S_TX,
  793. .stream_name = LPASS_BE_PRI_MI2S_TX,
  794. .capture_only = 1,
  795. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  796. SND_SOC_DPCM_TRIGGER_POST},
  797. .ops = &msm_common_be_ops,
  798. .ignore_suspend = 1,
  799. SND_SOC_DAILINK_REG(pri_mi2s_tx),
  800. },
  801. {
  802. .name = LPASS_BE_SEC_MI2S_RX,
  803. .stream_name = LPASS_BE_SEC_MI2S_RX,
  804. .playback_only = 1,
  805. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  806. SND_SOC_DPCM_TRIGGER_POST},
  807. .ops = &msm_common_be_ops,
  808. .ignore_suspend = 1,
  809. .ignore_pmdown_time = 1,
  810. SND_SOC_DAILINK_REG(sec_mi2s_rx),
  811. },
  812. {
  813. .name = LPASS_BE_SEC_MI2S_TX,
  814. .stream_name = LPASS_BE_SEC_MI2S_TX,
  815. .capture_only = 1,
  816. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  817. SND_SOC_DPCM_TRIGGER_POST},
  818. .ops = &msm_common_be_ops,
  819. .ignore_suspend = 1,
  820. SND_SOC_DAILINK_REG(sec_mi2s_tx),
  821. },
  822. {
  823. .name = LPASS_BE_TERT_MI2S_RX,
  824. .stream_name = LPASS_BE_TERT_MI2S_RX,
  825. .playback_only = 1,
  826. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  827. SND_SOC_DPCM_TRIGGER_POST},
  828. .ops = &msm_common_be_ops,
  829. .ignore_suspend = 1,
  830. .ignore_pmdown_time = 1,
  831. SND_SOC_DAILINK_REG(tert_mi2s_rx),
  832. },
  833. {
  834. .name = LPASS_BE_TERT_MI2S_TX,
  835. .stream_name = LPASS_BE_TERT_MI2S_TX,
  836. .capture_only = 1,
  837. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  838. SND_SOC_DPCM_TRIGGER_POST},
  839. .ops = &msm_common_be_ops,
  840. .ignore_suspend = 1,
  841. SND_SOC_DAILINK_REG(tert_mi2s_tx),
  842. },
  843. {
  844. .name = LPASS_BE_QUAT_MI2S_RX,
  845. .stream_name = LPASS_BE_QUAT_MI2S_RX,
  846. .playback_only = 1,
  847. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  848. SND_SOC_DPCM_TRIGGER_POST},
  849. .ops = &msm_common_be_ops,
  850. .ignore_suspend = 1,
  851. .ignore_pmdown_time = 1,
  852. SND_SOC_DAILINK_REG(quat_mi2s_rx),
  853. },
  854. {
  855. .name = LPASS_BE_QUAT_MI2S_TX,
  856. .stream_name = LPASS_BE_QUAT_MI2S_TX,
  857. .capture_only = 1,
  858. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  859. SND_SOC_DPCM_TRIGGER_POST},
  860. .ops = &msm_common_be_ops,
  861. .ignore_suspend = 1,
  862. SND_SOC_DAILINK_REG(quat_mi2s_tx),
  863. },
  864. {
  865. .name = LPASS_BE_QUIN_MI2S_RX,
  866. .stream_name = LPASS_BE_QUIN_MI2S_RX,
  867. .playback_only = 1,
  868. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  869. SND_SOC_DPCM_TRIGGER_POST},
  870. .ops = &msm_common_be_ops,
  871. .ignore_suspend = 1,
  872. .ignore_pmdown_time = 1,
  873. SND_SOC_DAILINK_REG(quin_mi2s_rx),
  874. },
  875. {
  876. .name = LPASS_BE_QUIN_MI2S_TX,
  877. .stream_name = LPASS_BE_QUIN_MI2S_TX,
  878. .capture_only = 1,
  879. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  880. SND_SOC_DPCM_TRIGGER_POST},
  881. .ops = &msm_common_be_ops,
  882. .ignore_suspend = 1,
  883. SND_SOC_DAILINK_REG(quin_mi2s_tx),
  884. },
  885. {
  886. .name = LPASS_BE_SEN_MI2S_RX,
  887. .stream_name = LPASS_BE_SEN_MI2S_RX,
  888. .playback_only = 1,
  889. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  890. SND_SOC_DPCM_TRIGGER_POST},
  891. .ops = &msm_common_be_ops,
  892. .ignore_suspend = 1,
  893. .ignore_pmdown_time = 1,
  894. SND_SOC_DAILINK_REG(sen_mi2s_rx),
  895. },
  896. {
  897. .name = LPASS_BE_SEN_MI2S_TX,
  898. .stream_name = LPASS_BE_SEN_MI2S_TX,
  899. .capture_only = 1,
  900. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  901. SND_SOC_DPCM_TRIGGER_POST},
  902. .ops = &msm_common_be_ops,
  903. .ignore_suspend = 1,
  904. SND_SOC_DAILINK_REG(sen_mi2s_tx),
  905. },
  906. };
  907. static struct snd_soc_dai_link msm_tdm_dai_links[] = {
  908. {
  909. .name = LPASS_BE_PRI_TDM_RX_0,
  910. .stream_name = LPASS_BE_PRI_TDM_RX_0,
  911. .playback_only = 1,
  912. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  913. SND_SOC_DPCM_TRIGGER_POST},
  914. .ops = &msm_common_be_ops,
  915. .ignore_suspend = 1,
  916. .ignore_pmdown_time = 1,
  917. SND_SOC_DAILINK_REG(pri_tdm_rx_0),
  918. },
  919. {
  920. .name = LPASS_BE_PRI_TDM_TX_0,
  921. .stream_name = LPASS_BE_PRI_TDM_TX_0,
  922. .capture_only = 1,
  923. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  924. SND_SOC_DPCM_TRIGGER_POST},
  925. .ops = &msm_common_be_ops,
  926. .ignore_suspend = 1,
  927. SND_SOC_DAILINK_REG(pri_tdm_tx_0),
  928. },
  929. {
  930. .name = LPASS_BE_SEC_TDM_RX_0,
  931. .stream_name = LPASS_BE_SEC_TDM_RX_0,
  932. .playback_only = 1,
  933. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  934. SND_SOC_DPCM_TRIGGER_POST},
  935. .ops = &msm_common_be_ops,
  936. .ignore_suspend = 1,
  937. .ignore_pmdown_time = 1,
  938. SND_SOC_DAILINK_REG(sec_tdm_rx_0),
  939. },
  940. {
  941. .name = LPASS_BE_SEC_TDM_TX_0,
  942. .stream_name = LPASS_BE_SEC_TDM_TX_0,
  943. .capture_only = 1,
  944. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  945. SND_SOC_DPCM_TRIGGER_POST},
  946. .ops = &msm_common_be_ops,
  947. .ignore_suspend = 1,
  948. SND_SOC_DAILINK_REG(sec_tdm_tx_0),
  949. },
  950. {
  951. .name = LPASS_BE_TERT_TDM_RX_0,
  952. .stream_name = LPASS_BE_TERT_TDM_RX_0,
  953. .playback_only = 1,
  954. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  955. SND_SOC_DPCM_TRIGGER_POST},
  956. .ops = &msm_common_be_ops,
  957. .ignore_suspend = 1,
  958. .ignore_pmdown_time = 1,
  959. SND_SOC_DAILINK_REG(tert_tdm_rx_0),
  960. },
  961. {
  962. .name = LPASS_BE_TERT_TDM_TX_0,
  963. .stream_name = LPASS_BE_TERT_TDM_TX_0,
  964. .capture_only = 1,
  965. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  966. SND_SOC_DPCM_TRIGGER_POST},
  967. .ops = &msm_common_be_ops,
  968. .ignore_suspend = 1,
  969. SND_SOC_DAILINK_REG(tert_tdm_tx_0),
  970. },
  971. {
  972. .name = LPASS_BE_QUAT_TDM_RX_0,
  973. .stream_name = LPASS_BE_QUAT_TDM_RX_0,
  974. .playback_only = 1,
  975. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  976. SND_SOC_DPCM_TRIGGER_POST},
  977. .ops = &msm_common_be_ops,
  978. .ignore_suspend = 1,
  979. .ignore_pmdown_time = 1,
  980. SND_SOC_DAILINK_REG(quat_tdm_rx_0),
  981. },
  982. {
  983. .name = LPASS_BE_QUAT_TDM_TX_0,
  984. .stream_name = LPASS_BE_QUAT_TDM_TX_0,
  985. .capture_only = 1,
  986. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  987. SND_SOC_DPCM_TRIGGER_POST},
  988. .ops = &msm_common_be_ops,
  989. .ignore_suspend = 1,
  990. SND_SOC_DAILINK_REG(quat_tdm_tx_0),
  991. },
  992. {
  993. .name = LPASS_BE_QUIN_TDM_RX_0,
  994. .stream_name = LPASS_BE_QUIN_TDM_RX_0,
  995. .playback_only = 1,
  996. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  997. SND_SOC_DPCM_TRIGGER_POST},
  998. .ops = &msm_common_be_ops,
  999. .ignore_suspend = 1,
  1000. .ignore_pmdown_time = 1,
  1001. SND_SOC_DAILINK_REG(quin_tdm_rx_0),
  1002. },
  1003. {
  1004. .name = LPASS_BE_QUIN_TDM_TX_0,
  1005. .stream_name = LPASS_BE_QUIN_TDM_TX_0,
  1006. .capture_only = 1,
  1007. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  1008. SND_SOC_DPCM_TRIGGER_POST},
  1009. .ops = &msm_common_be_ops,
  1010. .ignore_suspend = 1,
  1011. SND_SOC_DAILINK_REG(quin_tdm_tx_0),
  1012. },
  1013. {
  1014. .name = LPASS_BE_SEN_TDM_RX_0,
  1015. .stream_name = LPASS_BE_SEN_TDM_RX_0,
  1016. .playback_only = 1,
  1017. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  1018. SND_SOC_DPCM_TRIGGER_POST},
  1019. .ops = &msm_common_be_ops,
  1020. .ignore_suspend = 1,
  1021. .ignore_pmdown_time = 1,
  1022. SND_SOC_DAILINK_REG(sen_tdm_rx_0),
  1023. },
  1024. {
  1025. .name = LPASS_BE_SEN_TDM_TX_0,
  1026. .stream_name = LPASS_BE_SEN_TDM_TX_0,
  1027. .capture_only = 1,
  1028. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  1029. SND_SOC_DPCM_TRIGGER_POST},
  1030. .ops = &msm_common_be_ops,
  1031. .ignore_suspend = 1,
  1032. SND_SOC_DAILINK_REG(sen_tdm_tx_0),
  1033. },
  1034. };
  1035. static struct snd_soc_dai_link msm_waipio_dai_links[
  1036. ARRAY_SIZE(msm_wsa_cdc_dma_be_dai_links) +
  1037. ARRAY_SIZE(msm_wsa2_cdc_dma_be_dai_links) +
  1038. ARRAY_SIZE(msm_wsa_wsa2_cdc_dma_be_dai_links) +
  1039. ARRAY_SIZE(msm_rx_tx_cdc_dma_be_dai_links) +
  1040. ARRAY_SIZE(msm_va_cdc_dma_be_dai_links) +
  1041. ARRAY_SIZE(ext_disp_be_dai_link) +
  1042. ARRAY_SIZE(msm_common_be_dai_links) +
  1043. ARRAY_SIZE(msm_wcn_be_dai_links) +
  1044. ARRAY_SIZE(msm_mi2s_dai_links) +
  1045. ARRAY_SIZE(msm_tdm_dai_links)];
  1046. static int msm_populate_dai_link_component_of_node(
  1047. struct snd_soc_card *card)
  1048. {
  1049. int i, j, index, ret = 0;
  1050. struct device *cdev = card->dev;
  1051. struct snd_soc_dai_link *dai_link = card->dai_link;
  1052. struct device_node *np = NULL;
  1053. int codecs_enabled = 0;
  1054. struct snd_soc_dai_link_component *codecs_comp = NULL;
  1055. if (!cdev) {
  1056. dev_err(cdev, "%s: Sound card device memory NULL\n", __func__);
  1057. return -ENODEV;
  1058. }
  1059. for (i = 0; i < card->num_links; i++) {
  1060. if (dai_link[i].init == NULL)
  1061. dai_link[i].init = &msm_common_dai_link_init;
  1062. /* populate codec_of_node for snd card dai links */
  1063. if (dai_link[i].num_codecs > 0) {
  1064. for (j = 0; j < dai_link[i].num_codecs; j++) {
  1065. if (dai_link[i].codecs[j].of_node ||
  1066. !dai_link[i].codecs[j].name)
  1067. continue;
  1068. index = of_property_match_string(cdev->of_node,
  1069. "asoc-codec-names",
  1070. dai_link[i].codecs[j].name);
  1071. if (index < 0)
  1072. continue;
  1073. np = of_parse_phandle(cdev->of_node,
  1074. "asoc-codec",
  1075. index);
  1076. if (!np) {
  1077. dev_err(cdev, "%s: retrieving phandle for codec %s failed\n",
  1078. __func__,
  1079. dai_link[i].codecs[j].name);
  1080. ret = -ENODEV;
  1081. goto err;
  1082. }
  1083. dai_link[i].codecs[j].of_node = np;
  1084. dai_link[i].codecs[j].name = NULL;
  1085. }
  1086. }
  1087. }
  1088. /* In multi-codec scenario, check if codecs are enabled for this platform */
  1089. for (i = 0; i < card->num_links; i++) {
  1090. codecs_enabled = 0;
  1091. if (dai_link[i].num_codecs > 1) {
  1092. for (j = 0; j < dai_link[i].num_codecs; j++) {
  1093. if (!dai_link[i].codecs[j].of_node)
  1094. continue;
  1095. np = dai_link[i].codecs[j].of_node;
  1096. if (!of_device_is_available(np)) {
  1097. dev_err(cdev, "%s: codec is disabled: %s\n",
  1098. __func__,
  1099. np->full_name);
  1100. dai_link[i].codecs[j].of_node = NULL;
  1101. continue;
  1102. }
  1103. codecs_enabled++;
  1104. }
  1105. if (codecs_enabled > 0 &&
  1106. codecs_enabled < dai_link[i].num_codecs) {
  1107. codecs_comp = devm_kzalloc(cdev,
  1108. sizeof(struct snd_soc_dai_link_component)
  1109. * codecs_enabled, GFP_KERNEL);
  1110. if (!codecs_comp) {
  1111. dev_err(cdev, "%s: %s dailink codec component alloc failed\n",
  1112. __func__, dai_link[i].name);
  1113. ret = -ENOMEM;
  1114. goto err;
  1115. }
  1116. index = 0;
  1117. for (j = 0; j < dai_link[i].num_codecs; j++) {
  1118. if(dai_link[i].codecs[j].of_node) {
  1119. codecs_comp[index].of_node =
  1120. dai_link[i].codecs[j].of_node;
  1121. codecs_comp[index].dai_name =
  1122. dai_link[i].codecs[j].dai_name;
  1123. codecs_comp[index].name = NULL;
  1124. index++;
  1125. }
  1126. }
  1127. dai_link[i].codecs = codecs_comp;
  1128. dai_link[i].num_codecs = codecs_enabled;
  1129. }
  1130. }
  1131. }
  1132. err:
  1133. return ret;
  1134. }
  1135. static int msm_audrx_stub_init(struct snd_soc_pcm_runtime *rtd)
  1136. {
  1137. return 0;
  1138. }
  1139. static int msm_snd_stub_hw_params(struct snd_pcm_substream *substream,
  1140. struct snd_pcm_hw_params *params)
  1141. {
  1142. return 0;
  1143. }
  1144. static struct snd_soc_ops msm_stub_be_ops = {
  1145. .hw_params = msm_snd_stub_hw_params,
  1146. };
  1147. struct snd_soc_card snd_soc_card_stub_msm = {
  1148. .name = "waipio-stub-snd-card",
  1149. };
  1150. static struct snd_soc_dai_link msm_stub_be_dai_links[] = {
  1151. /* Backend DAI Links */
  1152. {
  1153. .name = LPASS_BE_PRI_AUXPCM_RX,
  1154. .stream_name = LPASS_BE_PRI_AUXPCM_RX,
  1155. .playback_only = 1,
  1156. .init = &msm_audrx_stub_init,
  1157. .ignore_pmdown_time = 1,
  1158. .ignore_suspend = 1,
  1159. .ops = &msm_stub_be_ops,
  1160. SND_SOC_DAILINK_REG(auxpcm_rx),
  1161. },
  1162. {
  1163. .name = LPASS_BE_PRI_AUXPCM_TX,
  1164. .stream_name = LPASS_BE_PRI_AUXPCM_TX,
  1165. .capture_only = 1,
  1166. .ignore_suspend = 1,
  1167. .ops = &msm_stub_be_ops,
  1168. SND_SOC_DAILINK_REG(auxpcm_tx),
  1169. },
  1170. };
  1171. static struct snd_soc_dai_link msm_stub_dai_links[
  1172. ARRAY_SIZE(msm_stub_be_dai_links)];
  1173. static const struct of_device_id waipio_asoc_machine_of_match[] = {
  1174. { .compatible = "qcom,waipio-asoc-snd",
  1175. .data = "codec"},
  1176. { .compatible = "qcom,waipio-asoc-snd-stub",
  1177. .data = "stub_codec"},
  1178. {},
  1179. };
  1180. static int msm_snd_card_late_probe(struct snd_soc_card *card)
  1181. {
  1182. struct snd_soc_component *component = NULL;
  1183. struct snd_soc_pcm_runtime *rtd;
  1184. int ret = 0;
  1185. void *mbhc_calibration;
  1186. rtd = snd_soc_get_pcm_runtime(card, &card->dai_link[0]);
  1187. if (!rtd) {
  1188. dev_err(card->dev,
  1189. "%s: snd_soc_get_pcm_runtime for %s failed!\n",
  1190. __func__, card->dai_link[0]);
  1191. return -EINVAL;
  1192. }
  1193. component = snd_soc_rtdcom_lookup(rtd, WCD938X_DRV_NAME);
  1194. if (!component) {
  1195. pr_err("%s component is NULL\n", __func__);
  1196. return -EINVAL;
  1197. }
  1198. mbhc_calibration = def_wcd_mbhc_cal();
  1199. if (!mbhc_calibration)
  1200. return -ENOMEM;
  1201. wcd_mbhc_cfg.calibration = mbhc_calibration;
  1202. ret = wcd938x_mbhc_hs_detect(component, &wcd_mbhc_cfg);
  1203. if (ret) {
  1204. dev_err(component->dev, "%s: mbhc hs detect failed, err:%d\n",
  1205. __func__, ret);
  1206. goto err_hs_detect;
  1207. }
  1208. return 0;
  1209. err_hs_detect:
  1210. kfree(mbhc_calibration);
  1211. return ret;
  1212. }
  1213. static struct snd_soc_card *populate_snd_card_dailinks(struct device *dev, int wsa_max_devs)
  1214. {
  1215. struct snd_soc_card *card = NULL;
  1216. struct snd_soc_dai_link *dailink = NULL;
  1217. int total_links = 0;
  1218. int rc = 0;
  1219. u32 val = 0;
  1220. const struct of_device_id *match;
  1221. match = of_match_node(waipio_asoc_machine_of_match, dev->of_node);
  1222. if (!match) {
  1223. dev_err(dev, "%s: No DT match found for sound card\n",
  1224. __func__);
  1225. return NULL;
  1226. }
  1227. if (!strcmp(match->data, "codec")) {
  1228. card = &snd_soc_card_waipio_msm;
  1229. /* late probe uses dai link at index '0' to get wcd component */
  1230. memcpy(msm_waipio_dai_links + total_links,
  1231. msm_rx_tx_cdc_dma_be_dai_links,
  1232. sizeof(msm_rx_tx_cdc_dma_be_dai_links));
  1233. total_links +=
  1234. ARRAY_SIZE(msm_rx_tx_cdc_dma_be_dai_links);
  1235. switch (wsa_max_devs) {
  1236. case MONO_SPEAKER:
  1237. case STEREO_SPEAKER:
  1238. memcpy(msm_waipio_dai_links + total_links,
  1239. msm_wsa_cdc_dma_be_dai_links,
  1240. sizeof(msm_wsa_cdc_dma_be_dai_links));
  1241. total_links += ARRAY_SIZE(msm_wsa_cdc_dma_be_dai_links);
  1242. break;
  1243. case QUAD_SPEAKER:
  1244. memcpy(msm_waipio_dai_links + total_links,
  1245. msm_wsa2_cdc_dma_be_dai_links,
  1246. sizeof(msm_wsa2_cdc_dma_be_dai_links));
  1247. total_links += ARRAY_SIZE(msm_wsa2_cdc_dma_be_dai_links);
  1248. memcpy(msm_waipio_dai_links + total_links,
  1249. msm_wsa_wsa2_cdc_dma_be_dai_links,
  1250. sizeof(msm_wsa_wsa2_cdc_dma_be_dai_links));
  1251. total_links += ARRAY_SIZE(msm_wsa_wsa2_cdc_dma_be_dai_links);
  1252. break;
  1253. default:
  1254. dev_dbg(dev,
  1255. "%s: Unexpected number of WSAs, wsa_max_devs: %d\n",
  1256. __func__, wsa_max_devs);
  1257. break;
  1258. }
  1259. memcpy(msm_waipio_dai_links + total_links,
  1260. msm_va_cdc_dma_be_dai_links,
  1261. sizeof(msm_va_cdc_dma_be_dai_links));
  1262. total_links += ARRAY_SIZE(msm_va_cdc_dma_be_dai_links);
  1263. memcpy(msm_waipio_dai_links + total_links,
  1264. msm_common_be_dai_links,
  1265. sizeof(msm_common_be_dai_links));
  1266. total_links += ARRAY_SIZE(msm_common_be_dai_links);
  1267. rc = of_property_read_u32(dev->of_node,
  1268. "qcom,mi2s-audio-intf", &val);
  1269. if (!rc && val) {
  1270. memcpy(msm_waipio_dai_links + total_links,
  1271. msm_mi2s_dai_links,
  1272. sizeof(msm_mi2s_dai_links));
  1273. total_links += ARRAY_SIZE(msm_mi2s_dai_links);
  1274. }
  1275. rc = of_property_read_u32(dev->of_node,
  1276. "qcom,tdm-audio-intf", &val);
  1277. if (!rc && val) {
  1278. memcpy(msm_waipio_dai_links + total_links,
  1279. msm_tdm_dai_links,
  1280. sizeof(msm_tdm_dai_links));
  1281. total_links += ARRAY_SIZE(msm_tdm_dai_links);
  1282. }
  1283. rc = of_property_read_u32(dev->of_node,
  1284. "qcom,ext-disp-audio-rx", &val);
  1285. if (!rc && val) {
  1286. dev_dbg(dev, "%s(): ext disp audio support present\n",
  1287. __func__);
  1288. memcpy(msm_waipio_dai_links + total_links,
  1289. ext_disp_be_dai_link,
  1290. sizeof(ext_disp_be_dai_link));
  1291. total_links += ARRAY_SIZE(ext_disp_be_dai_link);
  1292. }
  1293. rc = of_property_read_u32(dev->of_node, "qcom,wcn-bt", &val);
  1294. if (!rc && val) {
  1295. dev_dbg(dev, "%s(): WCN BT support present\n",
  1296. __func__);
  1297. memcpy(msm_waipio_dai_links + total_links,
  1298. msm_wcn_be_dai_links,
  1299. sizeof(msm_wcn_be_dai_links));
  1300. total_links += ARRAY_SIZE(msm_wcn_be_dai_links);
  1301. }
  1302. dailink = msm_waipio_dai_links;
  1303. } else if(!strcmp(match->data, "stub_codec")) {
  1304. card = &snd_soc_card_stub_msm;
  1305. memcpy(msm_stub_dai_links,
  1306. msm_stub_be_dai_links,
  1307. sizeof(msm_stub_be_dai_links));
  1308. dailink = msm_stub_dai_links;
  1309. total_links = ARRAY_SIZE(msm_stub_be_dai_links);
  1310. }
  1311. if (card) {
  1312. card->dai_link = dailink;
  1313. card->num_links = total_links;
  1314. card->late_probe = msm_snd_card_late_probe;
  1315. }
  1316. return card;
  1317. }
  1318. static int msm_int_wsa_init(struct snd_soc_pcm_runtime *rtd)
  1319. {
  1320. u8 spkleft_ports[WSA883X_MAX_SWR_PORTS] = {0, 1, 2, 3};
  1321. u8 spkright_ports[WSA883X_MAX_SWR_PORTS] = {0, 1, 2, 3};
  1322. u8 spkleft_port_types[WSA883X_MAX_SWR_PORTS] = {SPKR_L, SPKR_L_COMP,
  1323. SPKR_L_BOOST, SPKR_L_VI};
  1324. u8 spkright_port_types[WSA883X_MAX_SWR_PORTS] = {SPKR_R, SPKR_R_COMP,
  1325. SPKR_R_BOOST, SPKR_R_VI};
  1326. unsigned int ch_rate[WSA883X_MAX_SWR_PORTS] = {SWR_CLK_RATE_2P4MHZ, SWR_CLK_RATE_0P6MHZ,
  1327. SWR_CLK_RATE_0P3MHZ, SWR_CLK_RATE_1P2MHZ};
  1328. unsigned int ch_mask[WSA883X_MAX_SWR_PORTS] = {0x1, 0xF, 0x3, 0x3};
  1329. struct snd_soc_component *component = NULL;
  1330. struct msm_asoc_mach_data *pdata =
  1331. snd_soc_card_get_drvdata(rtd->card);
  1332. if (pdata->wsa_max_devs > 0) {
  1333. component = snd_soc_rtdcom_lookup(rtd, "wsa-codec.1");
  1334. if (!component) {
  1335. pr_err("%s: wsa-codec.1 component is NULL\n", __func__);
  1336. return -EINVAL;
  1337. }
  1338. wsa883x_set_channel_map(component, &spkleft_ports[0],
  1339. WSA883X_MAX_SWR_PORTS, &ch_mask[0],
  1340. &ch_rate[0], &spkleft_port_types[0]);
  1341. wsa883x_codec_info_create_codec_entry(pdata->codec_root,
  1342. component);
  1343. }
  1344. /* If current platform has more than one WSA */
  1345. if (pdata->wsa_max_devs > 1) {
  1346. component = snd_soc_rtdcom_lookup(rtd, "wsa-codec.2");
  1347. if (!component) {
  1348. pr_err("%s: wsa-codec.2 component is NULL\n", __func__);
  1349. return -EINVAL;
  1350. }
  1351. wsa883x_set_channel_map(component, &spkright_ports[0],
  1352. WSA883X_MAX_SWR_PORTS, &ch_mask[0],
  1353. &ch_rate[0], &spkright_port_types[0]);
  1354. wsa883x_codec_info_create_codec_entry(pdata->codec_root,
  1355. component);
  1356. }
  1357. if (pdata->wsa_max_devs > 2) {
  1358. component = snd_soc_rtdcom_lookup(rtd, "wsa-codec.3");
  1359. if (!component) {
  1360. pr_err("%s: wsa-codec.3 component is NULL\n", __func__);
  1361. return -EINVAL;
  1362. }
  1363. wsa883x_set_channel_map(component, &spkleft_ports[0],
  1364. WSA883X_MAX_SWR_PORTS, &ch_mask[0],
  1365. &ch_rate[0], &spkleft_port_types[0]);
  1366. wsa883x_codec_info_create_codec_entry(pdata->codec_root,
  1367. component);
  1368. }
  1369. if (pdata->wsa_max_devs > 3) {
  1370. component = snd_soc_rtdcom_lookup(rtd, "wsa-codec.4");
  1371. if (!component) {
  1372. pr_err("%s: wsa-codec.4 component is NULL\n", __func__);
  1373. return -EINVAL;
  1374. }
  1375. wsa883x_set_channel_map(component, &spkright_ports[0],
  1376. WSA883X_MAX_SWR_PORTS, &ch_mask[0],
  1377. &ch_rate[0], &spkright_port_types[0]);
  1378. wsa883x_codec_info_create_codec_entry(pdata->codec_root,
  1379. component);
  1380. }
  1381. msm_common_dai_link_init(rtd);
  1382. return 0;
  1383. }
  1384. static int msm_rx_tx_codec_init(struct snd_soc_pcm_runtime *rtd)
  1385. {
  1386. int codec_variant = -1;
  1387. struct snd_soc_component *component = NULL;
  1388. struct snd_soc_component *lpass_cdc_component = NULL;
  1389. struct snd_soc_dapm_context *dapm = NULL;
  1390. struct snd_info_entry *entry = NULL;
  1391. struct snd_card *card = NULL;
  1392. struct msm_asoc_mach_data *pdata =
  1393. snd_soc_card_get_drvdata(rtd->card);
  1394. int ret = 0;
  1395. lpass_cdc_component = snd_soc_rtdcom_lookup(rtd, "lpass-cdc");
  1396. if (!lpass_cdc_component) {
  1397. pr_err("%s: could not find component for lpass-cdc\n",
  1398. __func__);
  1399. return ret;
  1400. }
  1401. dapm = snd_soc_component_get_dapm(lpass_cdc_component);
  1402. snd_soc_dapm_new_controls(dapm, msm_int_dapm_widgets,
  1403. ARRAY_SIZE(msm_int_dapm_widgets));
  1404. snd_soc_dapm_ignore_suspend(dapm, "Digital Mic0");
  1405. snd_soc_dapm_ignore_suspend(dapm, "Digital Mic1");
  1406. snd_soc_dapm_ignore_suspend(dapm, "Digital Mic2");
  1407. snd_soc_dapm_ignore_suspend(dapm, "Digital Mic3");
  1408. snd_soc_dapm_ignore_suspend(dapm, "Digital Mic4");
  1409. snd_soc_dapm_ignore_suspend(dapm, "Digital Mic5");
  1410. snd_soc_dapm_ignore_suspend(dapm, "Digital Mic6");
  1411. snd_soc_dapm_ignore_suspend(dapm, "Digital Mic7");
  1412. snd_soc_dapm_ignore_suspend(dapm, "Analog Mic1");
  1413. snd_soc_dapm_ignore_suspend(dapm, "Analog Mic2");
  1414. snd_soc_dapm_ignore_suspend(dapm, "Analog Mic3");
  1415. snd_soc_dapm_ignore_suspend(dapm, "Analog Mic4");
  1416. snd_soc_dapm_ignore_suspend(dapm, "Analog Mic5");
  1417. lpass_cdc_set_port_map(lpass_cdc_component, ARRAY_SIZE(sm_port_map), sm_port_map);
  1418. card = rtd->card->snd_card;
  1419. if (!pdata->codec_root) {
  1420. entry = msm_snd_info_create_subdir(card->module, "codecs",
  1421. card->proc_root);
  1422. if (!entry) {
  1423. pr_debug("%s: Cannot create codecs module entry\n",
  1424. __func__);
  1425. return ret;
  1426. }
  1427. pdata->codec_root = entry;
  1428. }
  1429. lpass_cdc_info_create_codec_entry(pdata->codec_root, lpass_cdc_component);
  1430. lpass_cdc_register_wake_irq(lpass_cdc_component, false);
  1431. if (pdata->wcd_disabled)
  1432. goto done;
  1433. component = snd_soc_rtdcom_lookup(rtd, WCD938X_DRV_NAME);
  1434. if (!component) {
  1435. pr_err("%s could not find component for %s\n",
  1436. __func__, WCD938X_DRV_NAME);
  1437. return -EINVAL;
  1438. }
  1439. dapm = snd_soc_component_get_dapm(component);
  1440. card = component->card->snd_card;
  1441. snd_soc_dapm_ignore_suspend(dapm, "EAR");
  1442. snd_soc_dapm_ignore_suspend(dapm, "AUX");
  1443. snd_soc_dapm_ignore_suspend(dapm, "HPHL");
  1444. snd_soc_dapm_ignore_suspend(dapm, "HPHR");
  1445. snd_soc_dapm_ignore_suspend(dapm, "AMIC1");
  1446. snd_soc_dapm_ignore_suspend(dapm, "AMIC2");
  1447. snd_soc_dapm_ignore_suspend(dapm, "AMIC3");
  1448. snd_soc_dapm_ignore_suspend(dapm, "AMIC4");
  1449. snd_soc_dapm_sync(dapm);
  1450. pdata = snd_soc_card_get_drvdata(component->card);
  1451. if (!pdata->codec_root) {
  1452. entry = msm_snd_info_create_subdir(card->module, "codecs",
  1453. card->proc_root);
  1454. if (!entry) {
  1455. dev_dbg(component->dev, "%s: Cannot create codecs module entry\n",
  1456. __func__);
  1457. return 0;
  1458. }
  1459. pdata->codec_root = entry;
  1460. }
  1461. wcd938x_info_create_codec_entry(pdata->codec_root, component);
  1462. codec_variant = wcd938x_get_codec_variant(component);
  1463. dev_dbg(component->dev, "%s: variant %d\n", __func__, codec_variant);
  1464. if (codec_variant == WCD9385)
  1465. ret = lpass_cdc_rx_set_fir_capability(lpass_cdc_component, true);
  1466. else
  1467. ret = lpass_cdc_rx_set_fir_capability(lpass_cdc_component, false);
  1468. if (ret < 0) {
  1469. dev_err(component->dev, "%s: set fir capability failed: %d\n",
  1470. __func__, ret);
  1471. return ret;
  1472. }
  1473. done:
  1474. codec_reg_done = true;
  1475. msm_common_dai_link_init(rtd);
  1476. return ret;
  1477. }
  1478. static int waipio_ssr_enable(struct device *dev, void *data)
  1479. {
  1480. struct platform_device *pdev = to_platform_device(dev);
  1481. struct snd_soc_card *card = platform_get_drvdata(pdev);
  1482. struct snd_soc_pcm_runtime *rtd = NULL, *rtd_wcd = NULL, *rtd_wsa = NULL;
  1483. struct msm_asoc_mach_data *pdata = NULL;
  1484. int ret = 0;
  1485. if (!card) {
  1486. dev_err(dev, "%s: card is NULL\n", __func__);
  1487. ret = -EINVAL;
  1488. goto err;
  1489. }
  1490. if (!strcmp(card->name, "waipio-stub-snd-card")) {
  1491. /* TODO */
  1492. dev_dbg(dev, "%s: TODO \n", __func__);
  1493. }
  1494. snd_card_notify_user(SND_CARD_STATUS_ONLINE);
  1495. dev_dbg(dev, "%s: setting snd_card to ONLINE\n", __func__);
  1496. pdata = snd_soc_card_get_drvdata(card);
  1497. if (!pdata) {
  1498. dev_dbg(dev, "%s: pdata is NULL \n", __func__);
  1499. goto err;
  1500. }
  1501. rtd_wcd = snd_soc_get_pcm_runtime(card, &card->dai_link[0]);
  1502. if (!rtd_wcd) {
  1503. dev_dbg(dev,
  1504. "%s: snd_soc_get_pcm_runtime for %s failed!\n",
  1505. __func__, card->dai_link[0]);
  1506. }
  1507. if (pdata->wsa_max_devs > 0) {
  1508. rtd_wsa = snd_soc_get_pcm_runtime(card,
  1509. &card->dai_link[ARRAY_SIZE(msm_rx_tx_cdc_dma_be_dai_links)]);
  1510. if (!rtd_wsa) {
  1511. dev_dbg(dev,
  1512. "%s: snd_soc_get_pcm_runtime for %s failed!\n",
  1513. __func__, card->dai_link[ARRAY_SIZE(msm_rx_tx_cdc_dma_be_dai_links)]);
  1514. }
  1515. }
  1516. /* set UPD configuration */
  1517. if(!pdata->upd_config.backend_used) {
  1518. dev_dbg(dev,
  1519. "%s: upd- backend_used is NULL\n", __func__);
  1520. goto err;
  1521. }
  1522. if (!strcmp(pdata->upd_config.backend_used, "wsa")) {
  1523. if (!rtd_wsa)
  1524. goto err;
  1525. else
  1526. rtd = rtd_wsa;
  1527. } else if(!strcmp(pdata->upd_config.backend_used, "wcd")) {
  1528. if (!rtd_wcd &&!pdata->wcd_disabled)
  1529. goto err;
  1530. else
  1531. rtd = rtd_wcd;
  1532. } else {
  1533. dev_err(card->dev, "%s: Invalid backend to set UPD config\n",
  1534. __func__);
  1535. goto err;
  1536. }
  1537. msm_set_upd_config(rtd);
  1538. err:
  1539. return ret;
  1540. }
  1541. static void waipio_ssr_disable(struct device *dev, void *data)
  1542. {
  1543. struct platform_device *pdev = to_platform_device(dev);
  1544. struct snd_soc_card *card = platform_get_drvdata(pdev);
  1545. if (!card) {
  1546. dev_err(dev, "%s: card is NULL\n", __func__);
  1547. return;
  1548. }
  1549. dev_dbg(dev, "%s: setting snd_card to OFFLINE\n", __func__);
  1550. snd_card_notify_user(SND_CARD_STATUS_OFFLINE);
  1551. if (!strcmp(card->name, "waipio-stub-snd-card")) {
  1552. /* TODO */
  1553. dev_dbg(dev, "%s: TODO \n", __func__);
  1554. }
  1555. }
  1556. static const struct snd_event_ops waipio_ssr_ops = {
  1557. .enable = waipio_ssr_enable,
  1558. .disable = waipio_ssr_disable,
  1559. };
  1560. static int msm_audio_ssr_compare(struct device *dev, void *data)
  1561. {
  1562. struct device_node *node = data;
  1563. dev_dbg(dev, "%s: dev->of_node = 0x%p, node = 0x%p\n",
  1564. __func__, dev->of_node, node);
  1565. return (dev->of_node && dev->of_node == node);
  1566. }
  1567. static int msm_audio_ssr_register(struct device *dev)
  1568. {
  1569. struct device_node *np = dev->of_node;
  1570. struct snd_event_clients *ssr_clients = NULL;
  1571. struct device_node *node = NULL;
  1572. int ret = 0;
  1573. int i = 0;
  1574. for (i = 0; ; i++) {
  1575. node = of_parse_phandle(np, "qcom,msm_audio_ssr_devs", i);
  1576. if (!node)
  1577. break;
  1578. snd_event_mstr_add_client(&ssr_clients,
  1579. msm_audio_ssr_compare, node);
  1580. }
  1581. ret = snd_event_master_register(dev, &waipio_ssr_ops,
  1582. ssr_clients, NULL);
  1583. if (!ret)
  1584. snd_event_notify(dev, SND_EVENT_UP);
  1585. return ret;
  1586. }
  1587. struct msm_common_pdata *msm_common_get_pdata(struct snd_soc_card *card)
  1588. {
  1589. struct msm_asoc_mach_data *pdata = snd_soc_card_get_drvdata(card);
  1590. if (!pdata)
  1591. return NULL;
  1592. return pdata->common_pdata;
  1593. }
  1594. void msm_common_set_pdata(struct snd_soc_card *card,
  1595. struct msm_common_pdata *common_pdata)
  1596. {
  1597. struct msm_asoc_mach_data *pdata = snd_soc_card_get_drvdata(card);
  1598. if (!pdata)
  1599. return;
  1600. pdata->common_pdata = common_pdata;
  1601. }
  1602. static int msm_asoc_machine_probe(struct platform_device *pdev)
  1603. {
  1604. struct snd_soc_card *card = NULL;
  1605. struct msm_asoc_mach_data *pdata = NULL;
  1606. int ret = 0;
  1607. struct clk *lpass_audio_hw_vote = NULL;
  1608. if (!pdev->dev.of_node) {
  1609. dev_err(&pdev->dev, "%s: No platform supplied from device tree\n", __func__);
  1610. return -EINVAL;
  1611. }
  1612. pdata = devm_kzalloc(&pdev->dev,
  1613. sizeof(struct msm_asoc_mach_data), GFP_KERNEL);
  1614. if (!pdata)
  1615. return -ENOMEM;
  1616. of_property_read_u32(pdev->dev.of_node,
  1617. "qcom,wcd-disabled",
  1618. &pdata->wcd_disabled);
  1619. /* Get maximum WSA device count for this platform */
  1620. ret = of_property_read_u32(pdev->dev.of_node,
  1621. "qcom,wsa-max-devs", &pdata->wsa_max_devs);
  1622. if (ret) {
  1623. dev_info(&pdev->dev,
  1624. "%s: wsa-max-devs property missing in DT %s, ret = %d\n",
  1625. __func__, pdev->dev.of_node->full_name, ret);
  1626. pdata->wsa_max_devs = 0;
  1627. }
  1628. card = populate_snd_card_dailinks(&pdev->dev, pdata->wsa_max_devs);
  1629. if (!card) {
  1630. dev_err(&pdev->dev, "%s: Card uninitialized\n", __func__);
  1631. ret = -EINVAL;
  1632. goto err;
  1633. }
  1634. card->dev = &pdev->dev;
  1635. platform_set_drvdata(pdev, card);
  1636. snd_soc_card_set_drvdata(card, pdata);
  1637. ret = snd_soc_of_parse_card_name(card, "qcom,model");
  1638. if (ret) {
  1639. dev_err(&pdev->dev, "%s: parse card name failed, err:%d\n",
  1640. __func__, ret);
  1641. goto err;
  1642. }
  1643. ret = snd_soc_of_parse_audio_routing(card, "qcom,audio-routing");
  1644. if (ret) {
  1645. dev_err(&pdev->dev, "%s: parse audio routing failed, err:%d\n",
  1646. __func__, ret);
  1647. goto err;
  1648. }
  1649. ret = msm_populate_dai_link_component_of_node(card);
  1650. if (ret) {
  1651. ret = -EPROBE_DEFER;
  1652. goto err;
  1653. }
  1654. /* parse upd configuration */
  1655. msm_parse_upd_configuration(pdev, pdata);
  1656. ret = devm_snd_soc_register_card(&pdev->dev, card);
  1657. if (ret == -EPROBE_DEFER) {
  1658. if (codec_reg_done)
  1659. ret = -EINVAL;
  1660. goto err;
  1661. } else if (ret) {
  1662. dev_err(&pdev->dev, "%s: snd_soc_register_card failed (%d)\n",
  1663. __func__, ret);
  1664. goto err;
  1665. }
  1666. dev_info(&pdev->dev, "%s: Sound card %s registered\n",
  1667. __func__, card->name);
  1668. if (wcd_mbhc_cfg.enable_usbc_analog)
  1669. wcd_mbhc_cfg.swap_gnd_mic = msm_usbc_swap_gnd_mic;
  1670. pdata->fsa_handle = of_parse_phandle(pdev->dev.of_node,
  1671. "fsa4480-i2c-handle", 0);
  1672. if (!pdata->fsa_handle)
  1673. dev_dbg(&pdev->dev, "property %s not detected in node %s\n",
  1674. "fsa4480-i2c-handle", pdev->dev.of_node->full_name);
  1675. pdata->dmic01_gpio_p = of_parse_phandle(pdev->dev.of_node,
  1676. "qcom,cdc-dmic01-gpios",
  1677. 0);
  1678. pdata->dmic23_gpio_p = of_parse_phandle(pdev->dev.of_node,
  1679. "qcom,cdc-dmic23-gpios",
  1680. 0);
  1681. pdata->dmic45_gpio_p = of_parse_phandle(pdev->dev.of_node,
  1682. "qcom,cdc-dmic45-gpios",
  1683. 0);
  1684. if (pdata->dmic01_gpio_p)
  1685. msm_cdc_pinctrl_set_wakeup_capable(pdata->dmic01_gpio_p, false);
  1686. if (pdata->dmic23_gpio_p)
  1687. msm_cdc_pinctrl_set_wakeup_capable(pdata->dmic23_gpio_p, false);
  1688. if (pdata->dmic45_gpio_p)
  1689. msm_cdc_pinctrl_set_wakeup_capable(pdata->dmic45_gpio_p, false);
  1690. msm_common_snd_init(pdev, card);
  1691. /* Register LPASS audio hw vote */
  1692. lpass_audio_hw_vote = devm_clk_get(&pdev->dev, "lpass_audio_hw_vote");
  1693. if (IS_ERR(lpass_audio_hw_vote)) {
  1694. ret = PTR_ERR(lpass_audio_hw_vote);
  1695. dev_dbg(&pdev->dev, "%s: clk get %s failed %d\n",
  1696. __func__, "lpass_audio_hw_vote", ret);
  1697. lpass_audio_hw_vote = NULL;
  1698. ret = 0;
  1699. }
  1700. pdata->lpass_audio_hw_vote = lpass_audio_hw_vote;
  1701. pdata->core_audio_vote_count = 0;
  1702. ret = msm_audio_ssr_register(&pdev->dev);
  1703. if (ret)
  1704. pr_err("%s: Registration with SND event FWK failed ret = %d\n",
  1705. __func__, ret);
  1706. is_initial_boot = true;
  1707. /* Add QoS request for audio tasks */
  1708. msm_audio_add_qos_request();
  1709. /* change card status to ONLINE */
  1710. dev_dbg(&pdev->dev, "%s: setting snd_card to ONLINE\n", __func__);
  1711. snd_card_set_card_status(SND_CARD_STATUS_ONLINE);
  1712. return 0;
  1713. err:
  1714. devm_kfree(&pdev->dev, pdata);
  1715. return ret;
  1716. }
  1717. static int msm_asoc_machine_remove(struct platform_device *pdev)
  1718. {
  1719. struct snd_soc_card *card = platform_get_drvdata(pdev);
  1720. struct msm_asoc_mach_data *pdata = NULL;
  1721. struct msm_common_pdata *common_pdata = NULL;
  1722. if (card)
  1723. pdata = snd_soc_card_get_drvdata(card);
  1724. if (pdata)
  1725. common_pdata = pdata->common_pdata;
  1726. msm_common_snd_deinit(common_pdata);
  1727. snd_event_master_deregister(&pdev->dev);
  1728. snd_soc_unregister_card(card);
  1729. msm_audio_remove_qos_request();
  1730. return 0;
  1731. }
  1732. static struct platform_driver waipio_asoc_machine_driver = {
  1733. .driver = {
  1734. .name = DRV_NAME,
  1735. .owner = THIS_MODULE,
  1736. .pm = &snd_soc_pm_ops,
  1737. .of_match_table = waipio_asoc_machine_of_match,
  1738. .suppress_bind_attrs = true,
  1739. },
  1740. .probe = msm_asoc_machine_probe,
  1741. .remove = msm_asoc_machine_remove,
  1742. };
  1743. static int __init msm_asoc_machine_init(void)
  1744. {
  1745. snd_card_sysfs_init();
  1746. return platform_driver_register(&waipio_asoc_machine_driver);
  1747. }
  1748. module_init(msm_asoc_machine_init);
  1749. static void __exit msm_asoc_machine_exit(void)
  1750. {
  1751. platform_driver_unregister(&waipio_asoc_machine_driver);
  1752. }
  1753. module_exit(msm_asoc_machine_exit);
  1754. MODULE_SOFTDEP("pre: bt_fm_slim");
  1755. MODULE_DESCRIPTION("ALSA SoC msm");
  1756. MODULE_LICENSE("GPL v2");
  1757. MODULE_ALIAS("platform:" DRV_NAME);
  1758. MODULE_DEVICE_TABLE(of, waipio_asoc_machine_of_match);