msm_common.c 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683
  1. /* Copyright (c) 2020-2021, The Linux Foundation. All rights reserved.
  2. *
  3. * This program is free software; you can redistribute it and/or modify
  4. * it under the terms of the GNU General Public License version 2 and
  5. * only version 2 as published by the Free Software Foundation.
  6. *
  7. * This program is distributed in the hope that it will be useful,
  8. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  10. * GNU General Public License for more details.
  11. */
  12. #include <linux/gpio.h>
  13. #include <linux/of_gpio.h>
  14. #include <linux/platform_device.h>
  15. #include <linux/slab.h>
  16. #include <linux/of_device.h>
  17. #include <sound/control.h>
  18. #include <sound/core.h>
  19. #include <sound/soc.h>
  20. #include <sound/pcm_params.h>
  21. #include <asoc/msm-cdc-pinctrl.h>
  22. #include <dsp/spf-core.h>
  23. #include <dsp/msm_audio_ion.h>
  24. #include <sound/info.h>
  25. #include <dsp/apr_audio-v2.h>
  26. #include <dsp/audio_prm.h>
  27. #include "msm_common.h"
  28. struct snd_card_pdata {
  29. struct kobject snd_card_kobj;
  30. int card_status;
  31. }*snd_card_pdata;
  32. #define to_asoc_mach_common_pdata(kobj) \
  33. container_of((kobj), struct msm_common_pdata, aud_dev_kobj)
  34. #define DEVICE_ENABLE 1
  35. #define DEVICE_DISABLE 0
  36. #define ARRAY_SZ 21
  37. #define BUF_SZ 32
  38. #define DIR_SZ 10
  39. #define MAX_CODEC_DAI 8
  40. #define TDM_SLOT_WIDTH_BITS 32
  41. #define TDM_MAX_SLOTS 4
  42. static struct attribute device_state_attr = {
  43. .name = "state",
  44. .mode = 0660,
  45. };
  46. static struct attribute card_state_attr = {
  47. .name = "card_state",
  48. .mode = 0660,
  49. };
  50. #define MAX_PORT 20
  51. #define CODEC_CHMAP "Channel Map"
  52. enum backend_id {
  53. SLIM = 1,
  54. CODEC_DMA,
  55. };
  56. struct chmap_pdata {
  57. int id;
  58. uint32_t num_codec_dai;
  59. struct snd_soc_dai *dai[MAX_CODEC_DAI];
  60. };
  61. #define MAX_USR_INPUT 10
  62. static ssize_t aud_dev_sysfs_store(struct kobject *kobj,
  63. struct attribute *attr,
  64. const char *buf, size_t count)
  65. {
  66. ssize_t ret = -EINVAL;
  67. struct msm_common_pdata *pdata = to_asoc_mach_common_pdata(kobj);
  68. uint32_t pcm_id, state = 0;
  69. if (count > MAX_USR_INPUT) {
  70. pr_err("%s: invalid string written", __func__);
  71. goto done;
  72. }
  73. sscanf(buf, "%d %d", &pcm_id, &state);
  74. if ((pcm_id > pdata->num_aud_devs) || (pcm_id < 0)) {
  75. pr_err("%s: invalid pcm id %d \n", __func__, pcm_id);
  76. goto done;
  77. }
  78. if ((state > DEVICE_ENABLE) || (state < DEVICE_DISABLE)) {
  79. pr_err("%s: invalid state %d \n", __func__, state);
  80. goto done;
  81. }
  82. pr_debug("%s: pcm_id %d state %d \n", __func__, pcm_id, state);
  83. pdata->aud_dev_state[pcm_id] = state;
  84. if ( state == DEVICE_ENABLE && (pdata->dsp_sessions_closed != 0))
  85. pdata->dsp_sessions_closed = 0;
  86. ret = count;
  87. done:
  88. return ret;
  89. }
  90. static const struct sysfs_ops aud_dev_sysfs_ops = {
  91. .store = aud_dev_sysfs_store,
  92. };
  93. static struct kobj_type aud_dev_ktype = {
  94. .sysfs_ops = &aud_dev_sysfs_ops,
  95. };
  96. static int aud_dev_sysfs_init(struct msm_common_pdata *pdata)
  97. {
  98. int ret = 0;
  99. char dir[10] = "aud_dev";
  100. ret = kobject_init_and_add(&pdata->aud_dev_kobj, &aud_dev_ktype,
  101. kernel_kobj, dir);
  102. if (ret < 0) {
  103. pr_err("%s: Failed to add kobject %s, err = %d\n",
  104. __func__, dir, ret);
  105. goto done;
  106. }
  107. ret = sysfs_create_file(&pdata->aud_dev_kobj, &device_state_attr);
  108. if (ret < 0) {
  109. pr_err("%s: Failed to add wdsp_boot sysfs entry to %s\n",
  110. __func__, dir);
  111. goto fail_create_file;
  112. }
  113. return ret;
  114. fail_create_file:
  115. kobject_put(&pdata->aud_dev_kobj);
  116. done:
  117. return ret;
  118. }
  119. int snd_card_notify_user(int card_status)
  120. {
  121. snd_card_pdata->card_status = card_status;
  122. sysfs_notify(&snd_card_pdata->snd_card_kobj, NULL, "card_state");
  123. return 0;
  124. }
  125. static ssize_t snd_card_sysfs_show(struct kobject *kobj,
  126. struct attribute *attr, char *buf)
  127. {
  128. return snprintf(buf, BUF_SZ, "%d", snd_card_pdata->card_status);
  129. }
  130. static ssize_t snd_card_sysfs_store(struct kobject *kobj,
  131. struct attribute *attr, const char *buf, size_t count)
  132. {
  133. sscanf(buf, "%d", &snd_card_pdata->card_status);
  134. sysfs_notify(&snd_card_pdata->snd_card_kobj, NULL, "card_state");
  135. return 0;
  136. }
  137. static const struct sysfs_ops snd_card_sysfs_ops = {
  138. .show = snd_card_sysfs_show,
  139. .store = snd_card_sysfs_store,
  140. };
  141. static struct kobj_type snd_card_ktype = {
  142. .sysfs_ops = &snd_card_sysfs_ops,
  143. };
  144. int snd_card_sysfs_init(void)
  145. {
  146. int ret = 0;
  147. char dir[DIR_SZ] = "snd_card";
  148. snd_card_pdata = kcalloc(1, sizeof(struct snd_card_pdata), GFP_KERNEL);
  149. ret = kobject_init_and_add(&snd_card_pdata->snd_card_kobj, &snd_card_ktype,
  150. kernel_kobj, dir);
  151. if (ret < 0) {
  152. pr_err("%s: Failed to add kobject %s, err = %d\n",
  153. __func__, dir, ret);
  154. goto done;
  155. }
  156. ret = sysfs_create_file(&snd_card_pdata->snd_card_kobj, &card_state_attr);
  157. if (ret < 0) {
  158. pr_err("%s: Failed to add snd_card sysfs entry to %s\n",
  159. __func__, dir);
  160. goto fail_create_file;
  161. }
  162. return ret;
  163. fail_create_file:
  164. kobject_put(&snd_card_pdata->snd_card_kobj);
  165. done:
  166. return ret;
  167. }
  168. static void check_userspace_service_state(struct snd_soc_pcm_runtime *rtd,
  169. struct msm_common_pdata *pdata)
  170. {
  171. dev_info(rtd->card->dev,"%s: pcm_id %d state %d\n", __func__,
  172. rtd->num, pdata->aud_dev_state[rtd->num]);
  173. if (pdata->aud_dev_state[rtd->num] == DEVICE_ENABLE) {
  174. dev_info(rtd->card->dev, "%s userspace service crashed\n",
  175. __func__);
  176. if (pdata->dsp_sessions_closed == 0) {
  177. /*Issue close all graph cmd to DSP*/
  178. spf_core_apm_close_all();
  179. /*unmap all dma mapped buffers*/
  180. msm_audio_ion_crash_handler();
  181. pdata->dsp_sessions_closed = 1;
  182. }
  183. /*Reset the state as sysfs node wont be triggred*/
  184. pdata->aud_dev_state[rtd->num] = 0;
  185. }
  186. }
  187. static int get_intf_index(const char *stream_name)
  188. {
  189. if (strnstr(stream_name, "LPAIF_RXTX", strlen(stream_name)))
  190. return QUAT_MI2S_TDM_AUXPCM;
  191. else if (strnstr(stream_name, "LPAIF_WSA", strlen(stream_name)))
  192. return SEN_MI2S_TDM_AUXPCM;
  193. else if (strnstr(stream_name, "LPAIF_VA", strlen(stream_name)))
  194. return QUIN_MI2S_TDM_AUXPCM;
  195. else if (strnstr(stream_name, "LPAIF_AUD", strlen(stream_name)))
  196. return SEC_MI2S_TDM_AUXPCM;
  197. else if (strnstr(stream_name, "LPAIF", strlen(stream_name))) {
  198. if (strnstr(stream_name, "PRIMARY", strlen(stream_name)))
  199. return PRI_MI2S_TDM_AUXPCM;
  200. else if (strnstr(stream_name, "TERTIARY", strlen(stream_name)))
  201. return TER_MI2S_TDM_AUXPCM;
  202. }
  203. pr_debug("%s: stream name %s does not match\n", __func__, stream_name);
  204. return -EINVAL;
  205. }
  206. static int get_intf_clk_id(int index)
  207. {
  208. int clk_id;
  209. switch(index) {
  210. case PRI_MI2S_TDM_AUXPCM:
  211. clk_id = Q6AFE_LPASS_CLK_ID_PRI_TDM_IBIT;
  212. break;
  213. case SEC_MI2S_TDM_AUXPCM:
  214. clk_id = Q6AFE_LPASS_CLK_ID_SEC_TDM_IBIT;
  215. break;
  216. case TER_MI2S_TDM_AUXPCM:
  217. clk_id = Q6AFE_LPASS_CLK_ID_TER_TDM_IBIT;
  218. break;
  219. case QUAT_MI2S_TDM_AUXPCM:
  220. clk_id = Q6AFE_LPASS_CLK_ID_QUAD_TDM_IBIT;
  221. break;
  222. case QUIN_MI2S_TDM_AUXPCM:
  223. clk_id = Q6AFE_LPASS_CLK_ID_QUIN_TDM_IBIT;
  224. break;
  225. case SEN_MI2S_TDM_AUXPCM:
  226. clk_id = Q6AFE_LPASS_CLK_ID_SEN_TDM_IBIT;
  227. break;
  228. default:
  229. pr_err("%s: Invalid interface index: %d\n", __func__, index);
  230. clk_id = -EINVAL;
  231. }
  232. pr_debug("%s: clk id: %d\n", __func__, clk_id);
  233. return clk_id;
  234. }
  235. int msm_common_snd_hw_params(struct snd_pcm_substream *substream,
  236. struct snd_pcm_hw_params *params)
  237. {
  238. int ret = 0;
  239. int slot_width = TDM_SLOT_WIDTH_BITS;
  240. int slots = TDM_MAX_SLOTS;
  241. unsigned int rate;
  242. struct snd_soc_pcm_runtime *rtd = substream->private_data;
  243. const char *stream_name = rtd->dai_link->stream_name;
  244. struct snd_soc_card *card = rtd->card;
  245. struct msm_common_pdata *pdata = msm_common_get_pdata(card);
  246. int index = get_intf_index(stream_name);
  247. struct clk_cfg tdm_clk_cfg;
  248. if (index >= 0) {
  249. mutex_lock(&pdata->lock[index]);
  250. if (pdata->mi2s_gpio_p[index]) {
  251. if ((strnstr(stream_name, "TDM", strlen(stream_name)))) {
  252. rate = params_rate(params);
  253. tdm_clk_cfg.clk_id = get_intf_clk_id(index);
  254. if (tdm_clk_cfg.clk_id < 0) {
  255. ret = -EINVAL;
  256. pr_err("%s: Invalid clk id %d", __func__,
  257. tdm_clk_cfg.clk_id);
  258. goto done;
  259. }
  260. tdm_clk_cfg.clk_freq_in_hz = rate * slot_width * slots;
  261. tdm_clk_cfg.clk_attri = Q6AFE_LPASS_CLK_ATTRIBUTE_COUPLE_NO;
  262. tdm_clk_cfg.clk_root = 0;
  263. pr_debug("%s: clk_id :%d clk freq %d\n", __func__,
  264. tdm_clk_cfg.clk_id, tdm_clk_cfg.clk_freq_in_hz);
  265. ret = audio_prm_set_lpass_clk_cfg(&tdm_clk_cfg, 1);
  266. if (ret < 0) {
  267. pr_err("%s: prm lpass clk cfg set failed ret %d\n",
  268. __func__, ret);
  269. goto done;
  270. }
  271. }
  272. }
  273. done:
  274. mutex_unlock(&pdata->lock[index]);
  275. }
  276. return ret;
  277. }
  278. int msm_common_snd_startup(struct snd_pcm_substream *substream)
  279. {
  280. int ret = 0;
  281. struct snd_soc_pcm_runtime *rtd = substream->private_data;
  282. struct snd_soc_card *card = rtd->card;
  283. struct msm_common_pdata *pdata = msm_common_get_pdata(card);
  284. const char *stream_name = rtd->dai_link->stream_name;
  285. int index = get_intf_index(stream_name);
  286. dev_dbg(rtd->card->dev,
  287. "%s: substream = %s stream = %d\n",
  288. __func__, substream->name, substream->stream);
  289. if (!pdata) {
  290. dev_err(rtd->card->dev, "%s: pdata is NULL\n", __func__);
  291. return -EINVAL;
  292. }
  293. if (index >= 0) {
  294. mutex_lock(&pdata->lock[index]);
  295. if (pdata->mi2s_gpio_p[index]) {
  296. if (atomic_read(&(pdata->mi2s_gpio_ref_cnt[index])) == 0) {
  297. ret = msm_cdc_pinctrl_select_active_state(
  298. pdata->mi2s_gpio_p[index]);
  299. if (ret) {
  300. pr_err("%s:pinctrl set actve fail with %d\n",
  301. __func__, ret);
  302. goto done;
  303. }
  304. }
  305. atomic_inc(&(pdata->mi2s_gpio_ref_cnt[index]));
  306. }
  307. done:
  308. mutex_unlock(&pdata->lock[index]);
  309. }
  310. return ret;
  311. }
  312. void msm_common_snd_shutdown(struct snd_pcm_substream *substream)
  313. {
  314. int ret;
  315. struct snd_soc_pcm_runtime *rtd = substream->private_data;
  316. struct snd_soc_card *card = rtd->card;
  317. struct msm_common_pdata *pdata = msm_common_get_pdata(card);
  318. const char *stream_name = rtd->dai_link->stream_name;
  319. int index = get_intf_index(stream_name);
  320. struct clk_cfg tdm_clk_cfg;
  321. memset(&tdm_clk_cfg, 0, sizeof(struct clk_cfg));
  322. pr_debug("%s(): substream = %s stream = %d\n", __func__,
  323. substream->name, substream->stream);
  324. if (!pdata) {
  325. dev_err(card->dev, "%s: pdata is NULL\n", __func__);
  326. return;
  327. }
  328. check_userspace_service_state(rtd, pdata);
  329. if (index >= 0) {
  330. mutex_lock(&pdata->lock[index]);
  331. if (pdata->mi2s_gpio_p[index]) {
  332. atomic_dec(&pdata->mi2s_gpio_ref_cnt[index]);
  333. if (atomic_read(&pdata->mi2s_gpio_ref_cnt[index]) == 0) {
  334. if ((strnstr(stream_name, "TDM", strlen(stream_name)))) {
  335. tdm_clk_cfg.clk_id = get_intf_clk_id(index);
  336. ret = audio_prm_set_lpass_clk_cfg(&tdm_clk_cfg, 0);
  337. if (ret < 0)
  338. pr_err("%s: prm clk cfg set failed ret %d\n",
  339. __func__, ret);
  340. }
  341. ret = msm_cdc_pinctrl_select_sleep_state(
  342. pdata->mi2s_gpio_p[index]);
  343. if (ret)
  344. dev_err(card->dev,
  345. "%s: pinctrl set actv fail %d\n",
  346. __func__, ret);
  347. }
  348. }
  349. mutex_unlock(&pdata->lock[index]);
  350. }
  351. }
  352. int msm_common_snd_init(struct platform_device *pdev, struct snd_soc_card *card)
  353. {
  354. struct msm_common_pdata *common_pdata = NULL;
  355. int count;
  356. common_pdata = kcalloc(1, sizeof(struct msm_common_pdata), GFP_KERNEL);
  357. if (!common_pdata)
  358. return -ENOMEM;
  359. for (count = 0; count < MI2S_TDM_AUXPCM_MAX; count++) {
  360. mutex_init(&common_pdata->lock[count]);
  361. atomic_set(&common_pdata->mi2s_gpio_ref_cnt[count], 0);
  362. }
  363. common_pdata->mi2s_gpio_p[PRI_MI2S_TDM_AUXPCM] = of_parse_phandle(pdev->dev.of_node,
  364. "qcom,pri-mi2s-gpios", 0);
  365. common_pdata->mi2s_gpio_p[SEC_MI2S_TDM_AUXPCM] = of_parse_phandle(pdev->dev.of_node,
  366. "qcom,sec-mi2s-gpios", 0);
  367. common_pdata->mi2s_gpio_p[TER_MI2S_TDM_AUXPCM] = of_parse_phandle(pdev->dev.of_node,
  368. "qcom,tert-mi2s-gpios", 0);
  369. common_pdata->mi2s_gpio_p[QUAT_MI2S_TDM_AUXPCM] = of_parse_phandle(pdev->dev.of_node,
  370. "qcom,quat-mi2s-gpios", 0);
  371. common_pdata->mi2s_gpio_p[QUIN_MI2S_TDM_AUXPCM] = of_parse_phandle(pdev->dev.of_node,
  372. "qcom,quin-mi2s-gpios", 0);
  373. common_pdata->mi2s_gpio_p[SEN_MI2S_TDM_AUXPCM] = of_parse_phandle(pdev->dev.of_node,
  374. "qcom,sen-mi2s-gpios", 0);
  375. common_pdata->aud_dev_state = devm_kcalloc(&pdev->dev, card->num_links,
  376. sizeof(uint8_t), GFP_KERNEL);
  377. dev_info(&pdev->dev, "num_links %d \n", card->num_links);
  378. common_pdata->num_aud_devs = card->num_links;
  379. aud_dev_sysfs_init(common_pdata);
  380. msm_common_set_pdata(card, common_pdata);
  381. return 0;
  382. };
  383. void msm_common_snd_deinit(struct msm_common_pdata *common_pdata)
  384. {
  385. int count;
  386. if (!common_pdata)
  387. return;
  388. for (count = 0; count < MI2S_TDM_AUXPCM_MAX; count++) {
  389. mutex_destroy(&common_pdata->lock[count]);
  390. }
  391. }
  392. int msm_channel_map_info(struct snd_kcontrol *kcontrol,
  393. struct snd_ctl_elem_info *uinfo)
  394. {
  395. uinfo->type = SNDRV_CTL_ELEM_TYPE_BYTES;
  396. uinfo->count = sizeof(uint32_t) * MAX_PORT;
  397. return 0;
  398. }
  399. int msm_channel_map_get(struct snd_kcontrol *kcontrol,
  400. struct snd_ctl_elem_value *ucontrol)
  401. {
  402. struct chmap_pdata *kctl_pdata =
  403. (struct chmap_pdata *)kcontrol->private_data;
  404. struct snd_soc_dai *codec_dai = NULL;
  405. int backend_id = 0;
  406. uint32_t rx_ch[MAX_PORT] = {0}, tx_ch[MAX_PORT] = {0};
  407. uint32_t rx_ch_cnt = 0, tx_ch_cnt = 0;
  408. uint32_t *chmap_data = NULL;
  409. int ret = 0, len = 0, i = 0;
  410. if (kctl_pdata == NULL) {
  411. pr_debug("%s: chmap_pdata is not initialized\n", __func__);
  412. return -EINVAL;
  413. }
  414. codec_dai = kctl_pdata->dai[0];
  415. backend_id = kctl_pdata->id;
  416. switch (backend_id) {
  417. case SLIM: {
  418. uint32_t *chmap;
  419. uint32_t ch_cnt;
  420. ret = snd_soc_dai_get_channel_map(codec_dai,
  421. &tx_ch_cnt, tx_ch, &rx_ch_cnt, rx_ch);
  422. if (ret || (tx_ch_cnt == 0 && rx_ch_cnt == 0)) {
  423. pr_debug("%s: got incorrect channel map for backend_id:%d\n",
  424. __func__, backend_id);
  425. return ret;
  426. }
  427. if (rx_ch_cnt) {
  428. chmap = rx_ch;
  429. ch_cnt = rx_ch_cnt;
  430. } else {
  431. chmap = tx_ch;
  432. ch_cnt = tx_ch_cnt;
  433. }
  434. len = sizeof(uint32_t) * (ch_cnt + 1);
  435. chmap_data = kzalloc(len, GFP_KERNEL);
  436. if (!chmap_data)
  437. return -ENOMEM;
  438. chmap_data[0] = ch_cnt;
  439. for (i = 0; i < ch_cnt; i++)
  440. chmap_data[i+1] = chmap[i];
  441. memcpy(ucontrol->value.bytes.data, chmap_data, len);
  442. break;
  443. }
  444. case CODEC_DMA: {
  445. uint32_t cur_rx_ch = 0, cur_tx_ch = 0;
  446. uint32_t cur_rx_ch_cnt = 0, cur_tx_ch_cnt = 0;
  447. for (i = 0; i < kctl_pdata->num_codec_dai; ++i) {
  448. codec_dai = kctl_pdata->dai[i];
  449. if(!codec_dai) {
  450. continue;
  451. }
  452. cur_rx_ch_cnt = 0;
  453. cur_tx_ch_cnt = 0;
  454. cur_tx_ch = 0;
  455. cur_rx_ch = 0;
  456. ret = snd_soc_dai_get_channel_map(codec_dai,
  457. &cur_tx_ch_cnt, &cur_tx_ch,
  458. &cur_rx_ch_cnt, &cur_rx_ch);
  459. /* DAIs that not supports get_channel_map should pass */
  460. if (ret && (ret != -ENOTSUPP)) {
  461. pr_err("%s: get channel map failed for backend_id:%d,"
  462. " ret:%d\n",
  463. __func__, backend_id, ret);
  464. return ret;
  465. }
  466. rx_ch_cnt += cur_rx_ch_cnt;
  467. tx_ch_cnt += cur_tx_ch_cnt;
  468. rx_ch[0] |= cur_rx_ch;
  469. tx_ch[0] |= cur_tx_ch;
  470. }
  471. /* reset return value from the loop above */
  472. ret = 0;
  473. if (rx_ch_cnt == 0 && tx_ch_cnt == 0) {
  474. pr_debug("%s: got incorrect channel map for backend_id:%d, ",
  475. "RX Channel Count:%d,"
  476. "TX Channel Count:%d\n",
  477. __func__, backend_id, rx_ch_cnt, tx_ch_cnt);
  478. return ret;
  479. }
  480. chmap_data = kzalloc(sizeof(uint32_t) * 2, GFP_KERNEL);
  481. if (!chmap_data)
  482. return -ENOMEM;
  483. if (rx_ch_cnt) {
  484. chmap_data[0] = rx_ch_cnt;
  485. chmap_data[1] = rx_ch[0];
  486. } else {
  487. chmap_data[0] = tx_ch_cnt;
  488. chmap_data[1] = tx_ch[0];
  489. }
  490. memcpy(ucontrol->value.bytes.data, chmap_data,
  491. sizeof(uint32_t) * 2);
  492. break;
  493. }
  494. default:
  495. pr_err("%s, Invalid backend %d\n", __func__, backend_id);
  496. ret = -EINVAL;
  497. break;
  498. }
  499. kfree(chmap_data);
  500. return ret;
  501. }
  502. void msm_common_get_backend_name(const char *stream_name, char **backend_name)
  503. {
  504. char arg[ARRAY_SZ] = {0};
  505. char value[61] = {0};
  506. sscanf(stream_name, "%20[^-]-%60s", arg, value);
  507. *backend_name = kzalloc(ARRAY_SZ, GFP_KERNEL);
  508. if (!(*backend_name))
  509. return;
  510. strlcpy(*backend_name, arg, ARRAY_SZ);
  511. }
  512. int msm_common_dai_link_init(struct snd_soc_pcm_runtime *rtd)
  513. {
  514. struct snd_soc_dai *codec_dai = asoc_rtd_to_codec(rtd, 0);
  515. struct snd_soc_component *component = NULL;
  516. struct snd_soc_dai_link *dai_link = rtd->dai_link;
  517. struct device *dev = rtd->card->dev;
  518. int ret = 0;
  519. int index = 0;
  520. const char *mixer_ctl_name = CODEC_CHMAP;
  521. char *mixer_str = NULL;
  522. char *backend_name = NULL;
  523. uint32_t ctl_len = 0;
  524. struct chmap_pdata *pdata;
  525. struct snd_kcontrol *kctl;
  526. struct snd_kcontrol_new msm_common_channel_map[1] = {
  527. {
  528. .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
  529. .name = "?",
  530. .access = SNDRV_CTL_ELEM_ACCESS_READWRITE,
  531. .info = msm_channel_map_info,
  532. .get = msm_channel_map_get,
  533. .private_value = 0,
  534. }
  535. };
  536. if (!codec_dai) {
  537. pr_err("%s: failed to get codec dai", __func__);
  538. return -EINVAL;
  539. }
  540. component = codec_dai->component;
  541. msm_common_get_backend_name(dai_link->stream_name, &backend_name);
  542. if (!backend_name) {
  543. pr_err("%s: failed to get backend name", __func__);
  544. return -EINVAL;
  545. }
  546. pdata = devm_kzalloc(dev, sizeof(struct chmap_pdata), GFP_KERNEL);
  547. if (!pdata)
  548. return -ENOMEM;
  549. if ((!strncmp(backend_name, "SLIM", strlen("SLIM"))) ||
  550. (!strncmp(backend_name, "CODEC_DMA", strlen("CODEC_DMA")))) {
  551. ctl_len = strlen(dai_link->stream_name) + 1 +
  552. strlen(mixer_ctl_name) + 1;
  553. mixer_str = kzalloc(ctl_len, GFP_KERNEL);
  554. if (!mixer_str)
  555. return -ENOMEM;
  556. snprintf(mixer_str, ctl_len, "%s %s", dai_link->stream_name,
  557. mixer_ctl_name);
  558. msm_common_channel_map[0].name = mixer_str;
  559. msm_common_channel_map[0].private_value = 0;
  560. pr_debug("Registering new mixer ctl %s\n", mixer_str);
  561. ret = snd_soc_add_component_controls(component,
  562. msm_common_channel_map,
  563. ARRAY_SIZE(msm_common_channel_map));
  564. kctl = snd_soc_card_get_kcontrol(rtd->card, mixer_str);
  565. if (!kctl) {
  566. pr_err("failed to get kctl %s\n", mixer_str);
  567. ret = -EINVAL;
  568. goto free_mixer_str;
  569. }
  570. pdata->dai[0] = codec_dai;
  571. pdata->num_codec_dai = 1;
  572. if (!strncmp(backend_name, "SLIM", strlen("SLIM"))) {
  573. pdata->id = SLIM;
  574. } else {
  575. pdata->id = CODEC_DMA;
  576. if (rtd->num_codecs <= MAX_CODEC_DAI) {
  577. pdata->num_codec_dai = rtd->num_codecs;
  578. for_each_rtd_codec_dais(rtd, index, codec_dai) {
  579. pdata->dai[index] = codec_dai;
  580. }
  581. }
  582. }
  583. kctl->private_data = pdata;
  584. free_mixer_str:
  585. kfree(backend_name);
  586. kfree(mixer_str);
  587. }
  588. return ret;
  589. }