q6afe-dai.c 33 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094
  1. // SPDX-License-Identifier: GPL-2.0
  2. // Copyright (c) 2011-2017, The Linux Foundation. All rights reserved.
  3. // Copyright (c) 2018, Linaro Limited
  4. #include <linux/err.h>
  5. #include <linux/init.h>
  6. #include <linux/module.h>
  7. #include <linux/device.h>
  8. #include <linux/platform_device.h>
  9. #include <linux/slab.h>
  10. #include <sound/pcm.h>
  11. #include <sound/soc.h>
  12. #include <sound/pcm_params.h>
  13. #include "q6dsp-lpass-ports.h"
  14. #include "q6afe.h"
  15. struct q6afe_dai_priv_data {
  16. uint32_t sd_line_mask;
  17. uint32_t sync_mode;
  18. uint32_t sync_src;
  19. uint32_t data_out_enable;
  20. uint32_t invert_sync;
  21. uint32_t data_delay;
  22. uint32_t data_align;
  23. };
  24. struct q6afe_dai_data {
  25. struct q6afe_port *port[AFE_PORT_MAX];
  26. struct q6afe_port_config port_config[AFE_PORT_MAX];
  27. bool is_port_started[AFE_PORT_MAX];
  28. struct q6afe_dai_priv_data priv[AFE_PORT_MAX];
  29. };
  30. static int q6slim_hw_params(struct snd_pcm_substream *substream,
  31. struct snd_pcm_hw_params *params,
  32. struct snd_soc_dai *dai)
  33. {
  34. struct q6afe_dai_data *dai_data = dev_get_drvdata(dai->dev);
  35. struct q6afe_slim_cfg *slim = &dai_data->port_config[dai->id].slim;
  36. slim->sample_rate = params_rate(params);
  37. switch (params_format(params)) {
  38. case SNDRV_PCM_FORMAT_S16_LE:
  39. case SNDRV_PCM_FORMAT_SPECIAL:
  40. slim->bit_width = 16;
  41. break;
  42. case SNDRV_PCM_FORMAT_S24_LE:
  43. slim->bit_width = 24;
  44. break;
  45. case SNDRV_PCM_FORMAT_S32_LE:
  46. slim->bit_width = 32;
  47. break;
  48. default:
  49. pr_err("%s: format %d\n",
  50. __func__, params_format(params));
  51. return -EINVAL;
  52. }
  53. return 0;
  54. }
  55. static int q6hdmi_hw_params(struct snd_pcm_substream *substream,
  56. struct snd_pcm_hw_params *params,
  57. struct snd_soc_dai *dai)
  58. {
  59. struct q6afe_dai_data *dai_data = dev_get_drvdata(dai->dev);
  60. int channels = params_channels(params);
  61. struct q6afe_hdmi_cfg *hdmi = &dai_data->port_config[dai->id].hdmi;
  62. hdmi->sample_rate = params_rate(params);
  63. switch (params_format(params)) {
  64. case SNDRV_PCM_FORMAT_S16_LE:
  65. hdmi->bit_width = 16;
  66. break;
  67. case SNDRV_PCM_FORMAT_S24_LE:
  68. hdmi->bit_width = 24;
  69. break;
  70. }
  71. /* HDMI spec CEA-861-E: Table 28 Audio InfoFrame Data Byte 4 */
  72. switch (channels) {
  73. case 2:
  74. hdmi->channel_allocation = 0;
  75. break;
  76. case 3:
  77. hdmi->channel_allocation = 0x02;
  78. break;
  79. case 4:
  80. hdmi->channel_allocation = 0x06;
  81. break;
  82. case 5:
  83. hdmi->channel_allocation = 0x0A;
  84. break;
  85. case 6:
  86. hdmi->channel_allocation = 0x0B;
  87. break;
  88. case 7:
  89. hdmi->channel_allocation = 0x12;
  90. break;
  91. case 8:
  92. hdmi->channel_allocation = 0x13;
  93. break;
  94. default:
  95. dev_err(dai->dev, "invalid Channels = %u\n", channels);
  96. return -EINVAL;
  97. }
  98. return 0;
  99. }
  100. static int q6i2s_hw_params(struct snd_pcm_substream *substream,
  101. struct snd_pcm_hw_params *params,
  102. struct snd_soc_dai *dai)
  103. {
  104. struct q6afe_dai_data *dai_data = dev_get_drvdata(dai->dev);
  105. struct q6afe_i2s_cfg *i2s = &dai_data->port_config[dai->id].i2s_cfg;
  106. i2s->sample_rate = params_rate(params);
  107. i2s->bit_width = params_width(params);
  108. i2s->num_channels = params_channels(params);
  109. i2s->sd_line_mask = dai_data->priv[dai->id].sd_line_mask;
  110. return 0;
  111. }
  112. static int q6i2s_set_fmt(struct snd_soc_dai *dai, unsigned int fmt)
  113. {
  114. struct q6afe_dai_data *dai_data = dev_get_drvdata(dai->dev);
  115. struct q6afe_i2s_cfg *i2s = &dai_data->port_config[dai->id].i2s_cfg;
  116. i2s->fmt = fmt;
  117. return 0;
  118. }
  119. static int q6tdm_set_tdm_slot(struct snd_soc_dai *dai,
  120. unsigned int tx_mask,
  121. unsigned int rx_mask,
  122. int slots, int slot_width)
  123. {
  124. struct q6afe_dai_data *dai_data = dev_get_drvdata(dai->dev);
  125. struct q6afe_tdm_cfg *tdm = &dai_data->port_config[dai->id].tdm;
  126. unsigned int cap_mask;
  127. int rc = 0;
  128. /* HW only supports 16 and 32 bit slot width configuration */
  129. if ((slot_width != 16) && (slot_width != 32)) {
  130. dev_err(dai->dev, "%s: invalid slot_width %d\n",
  131. __func__, slot_width);
  132. return -EINVAL;
  133. }
  134. /* HW supports 1-32 slots configuration. Typical: 1, 2, 4, 8, 16, 32 */
  135. switch (slots) {
  136. case 2:
  137. cap_mask = 0x03;
  138. break;
  139. case 4:
  140. cap_mask = 0x0F;
  141. break;
  142. case 8:
  143. cap_mask = 0xFF;
  144. break;
  145. case 16:
  146. cap_mask = 0xFFFF;
  147. break;
  148. default:
  149. dev_err(dai->dev, "%s: invalid slots %d\n",
  150. __func__, slots);
  151. return -EINVAL;
  152. }
  153. switch (dai->id) {
  154. case PRIMARY_TDM_RX_0 ... QUINARY_TDM_TX_7:
  155. tdm->nslots_per_frame = slots;
  156. tdm->slot_width = slot_width;
  157. /* TDM RX dais ids are even and tx are odd */
  158. tdm->slot_mask = ((dai->id & 0x1) ? tx_mask : rx_mask) & cap_mask;
  159. break;
  160. default:
  161. dev_err(dai->dev, "%s: invalid dai id 0x%x\n",
  162. __func__, dai->id);
  163. return -EINVAL;
  164. }
  165. return rc;
  166. }
  167. static int q6tdm_set_channel_map(struct snd_soc_dai *dai,
  168. unsigned int tx_num, unsigned int *tx_slot,
  169. unsigned int rx_num, unsigned int *rx_slot)
  170. {
  171. struct q6afe_dai_data *dai_data = dev_get_drvdata(dai->dev);
  172. struct q6afe_tdm_cfg *tdm = &dai_data->port_config[dai->id].tdm;
  173. int rc = 0;
  174. int i = 0;
  175. switch (dai->id) {
  176. case PRIMARY_TDM_RX_0 ... QUINARY_TDM_TX_7:
  177. if (dai->id & 0x1) {
  178. if (!tx_slot) {
  179. dev_err(dai->dev, "tx slot not found\n");
  180. return -EINVAL;
  181. }
  182. if (tx_num > AFE_PORT_MAX_AUDIO_CHAN_CNT) {
  183. dev_err(dai->dev, "invalid tx num %d\n",
  184. tx_num);
  185. return -EINVAL;
  186. }
  187. for (i = 0; i < tx_num; i++)
  188. tdm->ch_mapping[i] = tx_slot[i];
  189. for (i = tx_num; i < AFE_PORT_MAX_AUDIO_CHAN_CNT; i++)
  190. tdm->ch_mapping[i] = Q6AFE_CMAP_INVALID;
  191. tdm->num_channels = tx_num;
  192. } else {
  193. /* rx */
  194. if (!rx_slot) {
  195. dev_err(dai->dev, "rx slot not found\n");
  196. return -EINVAL;
  197. }
  198. if (rx_num > AFE_PORT_MAX_AUDIO_CHAN_CNT) {
  199. dev_err(dai->dev, "invalid rx num %d\n",
  200. rx_num);
  201. return -EINVAL;
  202. }
  203. for (i = 0; i < rx_num; i++)
  204. tdm->ch_mapping[i] = rx_slot[i];
  205. for (i = rx_num; i < AFE_PORT_MAX_AUDIO_CHAN_CNT; i++)
  206. tdm->ch_mapping[i] = Q6AFE_CMAP_INVALID;
  207. tdm->num_channels = rx_num;
  208. }
  209. break;
  210. default:
  211. dev_err(dai->dev, "%s: invalid dai id 0x%x\n",
  212. __func__, dai->id);
  213. return -EINVAL;
  214. }
  215. return rc;
  216. }
  217. static int q6tdm_hw_params(struct snd_pcm_substream *substream,
  218. struct snd_pcm_hw_params *params,
  219. struct snd_soc_dai *dai)
  220. {
  221. struct q6afe_dai_data *dai_data = dev_get_drvdata(dai->dev);
  222. struct q6afe_tdm_cfg *tdm = &dai_data->port_config[dai->id].tdm;
  223. tdm->bit_width = params_width(params);
  224. tdm->sample_rate = params_rate(params);
  225. tdm->num_channels = params_channels(params);
  226. tdm->data_align_type = dai_data->priv[dai->id].data_align;
  227. tdm->sync_src = dai_data->priv[dai->id].sync_src;
  228. tdm->sync_mode = dai_data->priv[dai->id].sync_mode;
  229. return 0;
  230. }
  231. static int q6dma_set_channel_map(struct snd_soc_dai *dai,
  232. unsigned int tx_num, unsigned int *tx_ch_mask,
  233. unsigned int rx_num, unsigned int *rx_ch_mask)
  234. {
  235. struct q6afe_dai_data *dai_data = dev_get_drvdata(dai->dev);
  236. struct q6afe_cdc_dma_cfg *cfg = &dai_data->port_config[dai->id].dma_cfg;
  237. int ch_mask;
  238. int rc = 0;
  239. switch (dai->id) {
  240. case WSA_CODEC_DMA_TX_0:
  241. case WSA_CODEC_DMA_TX_1:
  242. case WSA_CODEC_DMA_TX_2:
  243. case VA_CODEC_DMA_TX_0:
  244. case VA_CODEC_DMA_TX_1:
  245. case VA_CODEC_DMA_TX_2:
  246. case TX_CODEC_DMA_TX_0:
  247. case TX_CODEC_DMA_TX_1:
  248. case TX_CODEC_DMA_TX_2:
  249. case TX_CODEC_DMA_TX_3:
  250. case TX_CODEC_DMA_TX_4:
  251. case TX_CODEC_DMA_TX_5:
  252. if (!tx_ch_mask) {
  253. dev_err(dai->dev, "tx slot not found\n");
  254. return -EINVAL;
  255. }
  256. if (tx_num > AFE_PORT_MAX_AUDIO_CHAN_CNT) {
  257. dev_err(dai->dev, "invalid tx num %d\n",
  258. tx_num);
  259. return -EINVAL;
  260. }
  261. ch_mask = *tx_ch_mask;
  262. break;
  263. case WSA_CODEC_DMA_RX_0:
  264. case WSA_CODEC_DMA_RX_1:
  265. case RX_CODEC_DMA_RX_0:
  266. case RX_CODEC_DMA_RX_1:
  267. case RX_CODEC_DMA_RX_2:
  268. case RX_CODEC_DMA_RX_3:
  269. case RX_CODEC_DMA_RX_4:
  270. case RX_CODEC_DMA_RX_5:
  271. case RX_CODEC_DMA_RX_6:
  272. case RX_CODEC_DMA_RX_7:
  273. /* rx */
  274. if (!rx_ch_mask) {
  275. dev_err(dai->dev, "rx slot not found\n");
  276. return -EINVAL;
  277. }
  278. if (rx_num > AFE_PORT_MAX_AUDIO_CHAN_CNT) {
  279. dev_err(dai->dev, "invalid rx num %d\n",
  280. rx_num);
  281. return -EINVAL;
  282. }
  283. ch_mask = *rx_ch_mask;
  284. break;
  285. default:
  286. dev_err(dai->dev, "%s: invalid dai id 0x%x\n",
  287. __func__, dai->id);
  288. return -EINVAL;
  289. }
  290. cfg->active_channels_mask = ch_mask;
  291. return rc;
  292. }
  293. static int q6dma_hw_params(struct snd_pcm_substream *substream,
  294. struct snd_pcm_hw_params *params,
  295. struct snd_soc_dai *dai)
  296. {
  297. struct q6afe_dai_data *dai_data = dev_get_drvdata(dai->dev);
  298. struct q6afe_cdc_dma_cfg *cfg = &dai_data->port_config[dai->id].dma_cfg;
  299. cfg->bit_width = params_width(params);
  300. cfg->sample_rate = params_rate(params);
  301. cfg->num_channels = params_channels(params);
  302. return 0;
  303. }
  304. static void q6afe_dai_shutdown(struct snd_pcm_substream *substream,
  305. struct snd_soc_dai *dai)
  306. {
  307. struct q6afe_dai_data *dai_data = dev_get_drvdata(dai->dev);
  308. int rc;
  309. if (!dai_data->is_port_started[dai->id])
  310. return;
  311. rc = q6afe_port_stop(dai_data->port[dai->id]);
  312. if (rc < 0)
  313. dev_err(dai->dev, "fail to close AFE port (%d)\n", rc);
  314. dai_data->is_port_started[dai->id] = false;
  315. }
  316. static int q6afe_dai_prepare(struct snd_pcm_substream *substream,
  317. struct snd_soc_dai *dai)
  318. {
  319. struct q6afe_dai_data *dai_data = dev_get_drvdata(dai->dev);
  320. int rc;
  321. if (dai_data->is_port_started[dai->id]) {
  322. /* stop the port and restart with new port config */
  323. rc = q6afe_port_stop(dai_data->port[dai->id]);
  324. if (rc < 0) {
  325. dev_err(dai->dev, "fail to close AFE port (%d)\n", rc);
  326. return rc;
  327. }
  328. }
  329. switch (dai->id) {
  330. case HDMI_RX:
  331. case DISPLAY_PORT_RX:
  332. q6afe_hdmi_port_prepare(dai_data->port[dai->id],
  333. &dai_data->port_config[dai->id].hdmi);
  334. break;
  335. case SLIMBUS_0_RX ... SLIMBUS_6_TX:
  336. q6afe_slim_port_prepare(dai_data->port[dai->id],
  337. &dai_data->port_config[dai->id].slim);
  338. break;
  339. case QUINARY_MI2S_RX ... QUINARY_MI2S_TX:
  340. case PRIMARY_MI2S_RX ... QUATERNARY_MI2S_TX:
  341. rc = q6afe_i2s_port_prepare(dai_data->port[dai->id],
  342. &dai_data->port_config[dai->id].i2s_cfg);
  343. if (rc < 0) {
  344. dev_err(dai->dev, "fail to prepare AFE port %x\n",
  345. dai->id);
  346. return rc;
  347. }
  348. break;
  349. case PRIMARY_TDM_RX_0 ... QUINARY_TDM_TX_7:
  350. q6afe_tdm_port_prepare(dai_data->port[dai->id],
  351. &dai_data->port_config[dai->id].tdm);
  352. break;
  353. case WSA_CODEC_DMA_RX_0 ... RX_CODEC_DMA_RX_7:
  354. q6afe_cdc_dma_port_prepare(dai_data->port[dai->id],
  355. &dai_data->port_config[dai->id].dma_cfg);
  356. break;
  357. default:
  358. return -EINVAL;
  359. }
  360. rc = q6afe_port_start(dai_data->port[dai->id]);
  361. if (rc < 0) {
  362. dev_err(dai->dev, "fail to start AFE port %x\n", dai->id);
  363. return rc;
  364. }
  365. dai_data->is_port_started[dai->id] = true;
  366. return 0;
  367. }
  368. static int q6slim_set_channel_map(struct snd_soc_dai *dai,
  369. unsigned int tx_num, unsigned int *tx_slot,
  370. unsigned int rx_num, unsigned int *rx_slot)
  371. {
  372. struct q6afe_dai_data *dai_data = dev_get_drvdata(dai->dev);
  373. struct q6afe_port_config *pcfg = &dai_data->port_config[dai->id];
  374. int i;
  375. if (dai->id & 0x1) {
  376. /* TX */
  377. if (!tx_slot) {
  378. pr_err("%s: tx slot not found\n", __func__);
  379. return -EINVAL;
  380. }
  381. for (i = 0; i < tx_num; i++)
  382. pcfg->slim.ch_mapping[i] = tx_slot[i];
  383. pcfg->slim.num_channels = tx_num;
  384. } else {
  385. if (!rx_slot) {
  386. pr_err("%s: rx slot not found\n", __func__);
  387. return -EINVAL;
  388. }
  389. for (i = 0; i < rx_num; i++)
  390. pcfg->slim.ch_mapping[i] = rx_slot[i];
  391. pcfg->slim.num_channels = rx_num;
  392. }
  393. return 0;
  394. }
  395. static int q6afe_mi2s_set_sysclk(struct snd_soc_dai *dai,
  396. int clk_id, unsigned int freq, int dir)
  397. {
  398. struct q6afe_dai_data *dai_data = dev_get_drvdata(dai->dev);
  399. struct q6afe_port *port = dai_data->port[dai->id];
  400. switch (clk_id) {
  401. case LPAIF_DIG_CLK:
  402. return q6afe_port_set_sysclk(port, clk_id, 0, 5, freq, dir);
  403. case LPAIF_BIT_CLK:
  404. case LPAIF_OSR_CLK:
  405. return q6afe_port_set_sysclk(port, clk_id,
  406. Q6AFE_LPASS_CLK_SRC_INTERNAL,
  407. Q6AFE_LPASS_CLK_ROOT_DEFAULT,
  408. freq, dir);
  409. case Q6AFE_LPASS_CLK_ID_PRI_MI2S_IBIT ... Q6AFE_LPASS_CLK_ID_QUI_MI2S_OSR:
  410. case Q6AFE_LPASS_CLK_ID_MCLK_1 ... Q6AFE_LPASS_CLK_ID_INT_MCLK_1:
  411. case Q6AFE_LPASS_CLK_ID_WSA_CORE_MCLK ... Q6AFE_LPASS_CLK_ID_VA_CORE_2X_MCLK:
  412. return q6afe_port_set_sysclk(port, clk_id,
  413. Q6AFE_LPASS_CLK_ATTRIBUTE_COUPLE_NO,
  414. Q6AFE_LPASS_CLK_ROOT_DEFAULT,
  415. freq, dir);
  416. case Q6AFE_LPASS_CLK_ID_PRI_TDM_IBIT ... Q6AFE_LPASS_CLK_ID_QUIN_TDM_EBIT:
  417. return q6afe_port_set_sysclk(port, clk_id,
  418. Q6AFE_LPASS_CLK_ATTRIBUTE_INVERT_COUPLE_NO,
  419. Q6AFE_LPASS_CLK_ROOT_DEFAULT,
  420. freq, dir);
  421. }
  422. return 0;
  423. }
  424. static const struct snd_soc_dapm_route q6afe_dapm_routes[] = {
  425. {"HDMI Playback", NULL, "HDMI_RX"},
  426. {"Display Port Playback", NULL, "DISPLAY_PORT_RX"},
  427. {"Slimbus Playback", NULL, "SLIMBUS_0_RX"},
  428. {"Slimbus1 Playback", NULL, "SLIMBUS_1_RX"},
  429. {"Slimbus2 Playback", NULL, "SLIMBUS_2_RX"},
  430. {"Slimbus3 Playback", NULL, "SLIMBUS_3_RX"},
  431. {"Slimbus4 Playback", NULL, "SLIMBUS_4_RX"},
  432. {"Slimbus5 Playback", NULL, "SLIMBUS_5_RX"},
  433. {"Slimbus6 Playback", NULL, "SLIMBUS_6_RX"},
  434. {"SLIMBUS_0_TX", NULL, "Slimbus Capture"},
  435. {"SLIMBUS_1_TX", NULL, "Slimbus1 Capture"},
  436. {"SLIMBUS_2_TX", NULL, "Slimbus2 Capture"},
  437. {"SLIMBUS_3_TX", NULL, "Slimbus3 Capture"},
  438. {"SLIMBUS_4_TX", NULL, "Slimbus4 Capture"},
  439. {"SLIMBUS_5_TX", NULL, "Slimbus5 Capture"},
  440. {"SLIMBUS_6_TX", NULL, "Slimbus6 Capture"},
  441. {"Primary MI2S Playback", NULL, "PRI_MI2S_RX"},
  442. {"Secondary MI2S Playback", NULL, "SEC_MI2S_RX"},
  443. {"Tertiary MI2S Playback", NULL, "TERT_MI2S_RX"},
  444. {"Quaternary MI2S Playback", NULL, "QUAT_MI2S_RX"},
  445. {"Quinary MI2S Playback", NULL, "QUIN_MI2S_RX"},
  446. {"Primary TDM0 Playback", NULL, "PRIMARY_TDM_RX_0"},
  447. {"Primary TDM1 Playback", NULL, "PRIMARY_TDM_RX_1"},
  448. {"Primary TDM2 Playback", NULL, "PRIMARY_TDM_RX_2"},
  449. {"Primary TDM3 Playback", NULL, "PRIMARY_TDM_RX_3"},
  450. {"Primary TDM4 Playback", NULL, "PRIMARY_TDM_RX_4"},
  451. {"Primary TDM5 Playback", NULL, "PRIMARY_TDM_RX_5"},
  452. {"Primary TDM6 Playback", NULL, "PRIMARY_TDM_RX_6"},
  453. {"Primary TDM7 Playback", NULL, "PRIMARY_TDM_RX_7"},
  454. {"Secondary TDM0 Playback", NULL, "SEC_TDM_RX_0"},
  455. {"Secondary TDM1 Playback", NULL, "SEC_TDM_RX_1"},
  456. {"Secondary TDM2 Playback", NULL, "SEC_TDM_RX_2"},
  457. {"Secondary TDM3 Playback", NULL, "SEC_TDM_RX_3"},
  458. {"Secondary TDM4 Playback", NULL, "SEC_TDM_RX_4"},
  459. {"Secondary TDM5 Playback", NULL, "SEC_TDM_RX_5"},
  460. {"Secondary TDM6 Playback", NULL, "SEC_TDM_RX_6"},
  461. {"Secondary TDM7 Playback", NULL, "SEC_TDM_RX_7"},
  462. {"Tertiary TDM0 Playback", NULL, "TERT_TDM_RX_0"},
  463. {"Tertiary TDM1 Playback", NULL, "TERT_TDM_RX_1"},
  464. {"Tertiary TDM2 Playback", NULL, "TERT_TDM_RX_2"},
  465. {"Tertiary TDM3 Playback", NULL, "TERT_TDM_RX_3"},
  466. {"Tertiary TDM4 Playback", NULL, "TERT_TDM_RX_4"},
  467. {"Tertiary TDM5 Playback", NULL, "TERT_TDM_RX_5"},
  468. {"Tertiary TDM6 Playback", NULL, "TERT_TDM_RX_6"},
  469. {"Tertiary TDM7 Playback", NULL, "TERT_TDM_RX_7"},
  470. {"Quaternary TDM0 Playback", NULL, "QUAT_TDM_RX_0"},
  471. {"Quaternary TDM1 Playback", NULL, "QUAT_TDM_RX_1"},
  472. {"Quaternary TDM2 Playback", NULL, "QUAT_TDM_RX_2"},
  473. {"Quaternary TDM3 Playback", NULL, "QUAT_TDM_RX_3"},
  474. {"Quaternary TDM4 Playback", NULL, "QUAT_TDM_RX_4"},
  475. {"Quaternary TDM5 Playback", NULL, "QUAT_TDM_RX_5"},
  476. {"Quaternary TDM6 Playback", NULL, "QUAT_TDM_RX_6"},
  477. {"Quaternary TDM7 Playback", NULL, "QUAT_TDM_RX_7"},
  478. {"Quinary TDM0 Playback", NULL, "QUIN_TDM_RX_0"},
  479. {"Quinary TDM1 Playback", NULL, "QUIN_TDM_RX_1"},
  480. {"Quinary TDM2 Playback", NULL, "QUIN_TDM_RX_2"},
  481. {"Quinary TDM3 Playback", NULL, "QUIN_TDM_RX_3"},
  482. {"Quinary TDM4 Playback", NULL, "QUIN_TDM_RX_4"},
  483. {"Quinary TDM5 Playback", NULL, "QUIN_TDM_RX_5"},
  484. {"Quinary TDM6 Playback", NULL, "QUIN_TDM_RX_6"},
  485. {"Quinary TDM7 Playback", NULL, "QUIN_TDM_RX_7"},
  486. {"PRIMARY_TDM_TX_0", NULL, "Primary TDM0 Capture"},
  487. {"PRIMARY_TDM_TX_1", NULL, "Primary TDM1 Capture"},
  488. {"PRIMARY_TDM_TX_2", NULL, "Primary TDM2 Capture"},
  489. {"PRIMARY_TDM_TX_3", NULL, "Primary TDM3 Capture"},
  490. {"PRIMARY_TDM_TX_4", NULL, "Primary TDM4 Capture"},
  491. {"PRIMARY_TDM_TX_5", NULL, "Primary TDM5 Capture"},
  492. {"PRIMARY_TDM_TX_6", NULL, "Primary TDM6 Capture"},
  493. {"PRIMARY_TDM_TX_7", NULL, "Primary TDM7 Capture"},
  494. {"SEC_TDM_TX_0", NULL, "Secondary TDM0 Capture"},
  495. {"SEC_TDM_TX_1", NULL, "Secondary TDM1 Capture"},
  496. {"SEC_TDM_TX_2", NULL, "Secondary TDM2 Capture"},
  497. {"SEC_TDM_TX_3", NULL, "Secondary TDM3 Capture"},
  498. {"SEC_TDM_TX_4", NULL, "Secondary TDM4 Capture"},
  499. {"SEC_TDM_TX_5", NULL, "Secondary TDM5 Capture"},
  500. {"SEC_TDM_TX_6", NULL, "Secondary TDM6 Capture"},
  501. {"SEC_TDM_TX_7", NULL, "Secondary TDM7 Capture"},
  502. {"TERT_TDM_TX_0", NULL, "Tertiary TDM0 Capture"},
  503. {"TERT_TDM_TX_1", NULL, "Tertiary TDM1 Capture"},
  504. {"TERT_TDM_TX_2", NULL, "Tertiary TDM2 Capture"},
  505. {"TERT_TDM_TX_3", NULL, "Tertiary TDM3 Capture"},
  506. {"TERT_TDM_TX_4", NULL, "Tertiary TDM4 Capture"},
  507. {"TERT_TDM_TX_5", NULL, "Tertiary TDM5 Capture"},
  508. {"TERT_TDM_TX_6", NULL, "Tertiary TDM6 Capture"},
  509. {"TERT_TDM_TX_7", NULL, "Tertiary TDM7 Capture"},
  510. {"QUAT_TDM_TX_0", NULL, "Quaternary TDM0 Capture"},
  511. {"QUAT_TDM_TX_1", NULL, "Quaternary TDM1 Capture"},
  512. {"QUAT_TDM_TX_2", NULL, "Quaternary TDM2 Capture"},
  513. {"QUAT_TDM_TX_3", NULL, "Quaternary TDM3 Capture"},
  514. {"QUAT_TDM_TX_4", NULL, "Quaternary TDM4 Capture"},
  515. {"QUAT_TDM_TX_5", NULL, "Quaternary TDM5 Capture"},
  516. {"QUAT_TDM_TX_6", NULL, "Quaternary TDM6 Capture"},
  517. {"QUAT_TDM_TX_7", NULL, "Quaternary TDM7 Capture"},
  518. {"QUIN_TDM_TX_0", NULL, "Quinary TDM0 Capture"},
  519. {"QUIN_TDM_TX_1", NULL, "Quinary TDM1 Capture"},
  520. {"QUIN_TDM_TX_2", NULL, "Quinary TDM2 Capture"},
  521. {"QUIN_TDM_TX_3", NULL, "Quinary TDM3 Capture"},
  522. {"QUIN_TDM_TX_4", NULL, "Quinary TDM4 Capture"},
  523. {"QUIN_TDM_TX_5", NULL, "Quinary TDM5 Capture"},
  524. {"QUIN_TDM_TX_6", NULL, "Quinary TDM6 Capture"},
  525. {"QUIN_TDM_TX_7", NULL, "Quinary TDM7 Capture"},
  526. {"TERT_MI2S_TX", NULL, "Tertiary MI2S Capture"},
  527. {"PRI_MI2S_TX", NULL, "Primary MI2S Capture"},
  528. {"SEC_MI2S_TX", NULL, "Secondary MI2S Capture"},
  529. {"QUAT_MI2S_TX", NULL, "Quaternary MI2S Capture"},
  530. {"QUIN_MI2S_TX", NULL, "Quinary MI2S Capture"},
  531. {"WSA_CODEC_DMA_RX_0 Playback", NULL, "WSA_CODEC_DMA_RX_0"},
  532. {"WSA_CODEC_DMA_TX_0", NULL, "WSA_CODEC_DMA_TX_0 Capture"},
  533. {"WSA_CODEC_DMA_RX_1 Playback", NULL, "WSA_CODEC_DMA_RX_1"},
  534. {"WSA_CODEC_DMA_TX_1", NULL, "WSA_CODEC_DMA_TX_1 Capture"},
  535. {"WSA_CODEC_DMA_TX_2", NULL, "WSA_CODEC_DMA_TX_2 Capture"},
  536. {"VA_CODEC_DMA_TX_0", NULL, "VA_CODEC_DMA_TX_0 Capture"},
  537. {"VA_CODEC_DMA_TX_1", NULL, "VA_CODEC_DMA_TX_1 Capture"},
  538. {"VA_CODEC_DMA_TX_2", NULL, "VA_CODEC_DMA_TX_2 Capture"},
  539. {"RX_CODEC_DMA_RX_0 Playback", NULL, "RX_CODEC_DMA_RX_0"},
  540. {"TX_CODEC_DMA_TX_0", NULL, "TX_CODEC_DMA_TX_0 Capture"},
  541. {"RX_CODEC_DMA_RX_1 Playback", NULL, "RX_CODEC_DMA_RX_1"},
  542. {"TX_CODEC_DMA_TX_1", NULL, "TX_CODEC_DMA_TX_1 Capture"},
  543. {"RX_CODEC_DMA_RX_2 Playback", NULL, "RX_CODEC_DMA_RX_2"},
  544. {"TX_CODEC_DMA_TX_2", NULL, "TX_CODEC_DMA_TX_2 Capture"},
  545. {"RX_CODEC_DMA_RX_3 Playback", NULL, "RX_CODEC_DMA_RX_3"},
  546. {"TX_CODEC_DMA_TX_3", NULL, "TX_CODEC_DMA_TX_3 Capture"},
  547. {"RX_CODEC_DMA_RX_4 Playback", NULL, "RX_CODEC_DMA_RX_4"},
  548. {"TX_CODEC_DMA_TX_4", NULL, "TX_CODEC_DMA_TX_4 Capture"},
  549. {"RX_CODEC_DMA_RX_5 Playback", NULL, "RX_CODEC_DMA_RX_5"},
  550. {"TX_CODEC_DMA_TX_5", NULL, "TX_CODEC_DMA_TX_5 Capture"},
  551. {"RX_CODEC_DMA_RX_6 Playback", NULL, "RX_CODEC_DMA_RX_6"},
  552. {"RX_CODEC_DMA_RX_7 Playback", NULL, "RX_CODEC_DMA_RX_7"},
  553. };
  554. static const struct snd_soc_dai_ops q6hdmi_ops = {
  555. .prepare = q6afe_dai_prepare,
  556. .hw_params = q6hdmi_hw_params,
  557. .shutdown = q6afe_dai_shutdown,
  558. };
  559. static const struct snd_soc_dai_ops q6i2s_ops = {
  560. .prepare = q6afe_dai_prepare,
  561. .hw_params = q6i2s_hw_params,
  562. .set_fmt = q6i2s_set_fmt,
  563. .shutdown = q6afe_dai_shutdown,
  564. .set_sysclk = q6afe_mi2s_set_sysclk,
  565. };
  566. static const struct snd_soc_dai_ops q6slim_ops = {
  567. .prepare = q6afe_dai_prepare,
  568. .hw_params = q6slim_hw_params,
  569. .shutdown = q6afe_dai_shutdown,
  570. .set_channel_map = q6slim_set_channel_map,
  571. };
  572. static const struct snd_soc_dai_ops q6tdm_ops = {
  573. .prepare = q6afe_dai_prepare,
  574. .shutdown = q6afe_dai_shutdown,
  575. .set_sysclk = q6afe_mi2s_set_sysclk,
  576. .set_tdm_slot = q6tdm_set_tdm_slot,
  577. .set_channel_map = q6tdm_set_channel_map,
  578. .hw_params = q6tdm_hw_params,
  579. };
  580. static const struct snd_soc_dai_ops q6dma_ops = {
  581. .prepare = q6afe_dai_prepare,
  582. .shutdown = q6afe_dai_shutdown,
  583. .set_sysclk = q6afe_mi2s_set_sysclk,
  584. .set_channel_map = q6dma_set_channel_map,
  585. .hw_params = q6dma_hw_params,
  586. };
  587. static int msm_dai_q6_dai_probe(struct snd_soc_dai *dai)
  588. {
  589. struct q6afe_dai_data *dai_data = dev_get_drvdata(dai->dev);
  590. struct q6afe_port *port;
  591. port = q6afe_port_get_from_id(dai->dev, dai->id);
  592. if (IS_ERR(port)) {
  593. dev_err(dai->dev, "Unable to get afe port\n");
  594. return -EINVAL;
  595. }
  596. dai_data->port[dai->id] = port;
  597. return 0;
  598. }
  599. static int msm_dai_q6_dai_remove(struct snd_soc_dai *dai)
  600. {
  601. struct q6afe_dai_data *dai_data = dev_get_drvdata(dai->dev);
  602. q6afe_port_put(dai_data->port[dai->id]);
  603. dai_data->port[dai->id] = NULL;
  604. return 0;
  605. }
  606. static const struct snd_soc_dapm_widget q6afe_dai_widgets[] = {
  607. SND_SOC_DAPM_AIF_IN("HDMI_RX", NULL, 0, SND_SOC_NOPM, 0, 0),
  608. SND_SOC_DAPM_AIF_IN("SLIMBUS_0_RX", NULL, 0, SND_SOC_NOPM, 0, 0),
  609. SND_SOC_DAPM_AIF_IN("SLIMBUS_1_RX", NULL, 0, SND_SOC_NOPM, 0, 0),
  610. SND_SOC_DAPM_AIF_IN("SLIMBUS_2_RX", NULL, 0, SND_SOC_NOPM, 0, 0),
  611. SND_SOC_DAPM_AIF_IN("SLIMBUS_3_RX", NULL, 0, SND_SOC_NOPM, 0, 0),
  612. SND_SOC_DAPM_AIF_IN("SLIMBUS_4_RX", NULL, 0, SND_SOC_NOPM, 0, 0),
  613. SND_SOC_DAPM_AIF_IN("SLIMBUS_5_RX", NULL, 0, SND_SOC_NOPM, 0, 0),
  614. SND_SOC_DAPM_AIF_IN("SLIMBUS_6_RX", NULL, 0, SND_SOC_NOPM, 0, 0),
  615. SND_SOC_DAPM_AIF_OUT("SLIMBUS_0_TX", NULL, 0, SND_SOC_NOPM, 0, 0),
  616. SND_SOC_DAPM_AIF_OUT("SLIMBUS_1_TX", NULL, 0, SND_SOC_NOPM, 0, 0),
  617. SND_SOC_DAPM_AIF_OUT("SLIMBUS_2_TX", NULL, 0, SND_SOC_NOPM, 0, 0),
  618. SND_SOC_DAPM_AIF_OUT("SLIMBUS_3_TX", NULL, 0, SND_SOC_NOPM, 0, 0),
  619. SND_SOC_DAPM_AIF_OUT("SLIMBUS_4_TX", NULL, 0, SND_SOC_NOPM, 0, 0),
  620. SND_SOC_DAPM_AIF_OUT("SLIMBUS_5_TX", NULL, 0, SND_SOC_NOPM, 0, 0),
  621. SND_SOC_DAPM_AIF_OUT("SLIMBUS_6_TX", NULL, 0, SND_SOC_NOPM, 0, 0),
  622. SND_SOC_DAPM_AIF_IN("QUIN_MI2S_RX", NULL,
  623. 0, SND_SOC_NOPM, 0, 0),
  624. SND_SOC_DAPM_AIF_OUT("QUIN_MI2S_TX", NULL,
  625. 0, SND_SOC_NOPM, 0, 0),
  626. SND_SOC_DAPM_AIF_IN("QUAT_MI2S_RX", NULL,
  627. 0, SND_SOC_NOPM, 0, 0),
  628. SND_SOC_DAPM_AIF_OUT("QUAT_MI2S_TX", NULL,
  629. 0, SND_SOC_NOPM, 0, 0),
  630. SND_SOC_DAPM_AIF_IN("TERT_MI2S_RX", NULL,
  631. 0, SND_SOC_NOPM, 0, 0),
  632. SND_SOC_DAPM_AIF_OUT("TERT_MI2S_TX", NULL,
  633. 0, SND_SOC_NOPM, 0, 0),
  634. SND_SOC_DAPM_AIF_IN("SEC_MI2S_RX", NULL,
  635. 0, SND_SOC_NOPM, 0, 0),
  636. SND_SOC_DAPM_AIF_OUT("SEC_MI2S_TX", NULL,
  637. 0, SND_SOC_NOPM, 0, 0),
  638. SND_SOC_DAPM_AIF_IN("SEC_MI2S_RX_SD1",
  639. "Secondary MI2S Playback SD1",
  640. 0, SND_SOC_NOPM, 0, 0),
  641. SND_SOC_DAPM_AIF_IN("PRI_MI2S_RX", NULL,
  642. 0, SND_SOC_NOPM, 0, 0),
  643. SND_SOC_DAPM_AIF_OUT("PRI_MI2S_TX", NULL,
  644. 0, SND_SOC_NOPM, 0, 0),
  645. SND_SOC_DAPM_AIF_IN("PRIMARY_TDM_RX_0", NULL,
  646. 0, SND_SOC_NOPM, 0, 0),
  647. SND_SOC_DAPM_AIF_IN("PRIMARY_TDM_RX_1", NULL,
  648. 0, SND_SOC_NOPM, 0, 0),
  649. SND_SOC_DAPM_AIF_IN("PRIMARY_TDM_RX_2", NULL,
  650. 0, SND_SOC_NOPM, 0, 0),
  651. SND_SOC_DAPM_AIF_IN("PRIMARY_TDM_RX_3", NULL,
  652. 0, SND_SOC_NOPM, 0, 0),
  653. SND_SOC_DAPM_AIF_IN("PRIMARY_TDM_RX_4", NULL,
  654. 0, SND_SOC_NOPM, 0, 0),
  655. SND_SOC_DAPM_AIF_IN("PRIMARY_TDM_RX_5", NULL,
  656. 0, SND_SOC_NOPM, 0, 0),
  657. SND_SOC_DAPM_AIF_IN("PRIMARY_TDM_RX_6", NULL,
  658. 0, SND_SOC_NOPM, 0, 0),
  659. SND_SOC_DAPM_AIF_IN("PRIMARY_TDM_RX_7", NULL,
  660. 0, SND_SOC_NOPM, 0, 0),
  661. SND_SOC_DAPM_AIF_OUT("PRIMARY_TDM_TX_0", NULL,
  662. 0, SND_SOC_NOPM, 0, 0),
  663. SND_SOC_DAPM_AIF_OUT("PRIMARY_TDM_TX_1", NULL,
  664. 0, SND_SOC_NOPM, 0, 0),
  665. SND_SOC_DAPM_AIF_OUT("PRIMARY_TDM_TX_2", NULL,
  666. 0, SND_SOC_NOPM, 0, 0),
  667. SND_SOC_DAPM_AIF_OUT("PRIMARY_TDM_TX_3", NULL,
  668. 0, SND_SOC_NOPM, 0, 0),
  669. SND_SOC_DAPM_AIF_OUT("PRIMARY_TDM_TX_4", NULL,
  670. 0, SND_SOC_NOPM, 0, 0),
  671. SND_SOC_DAPM_AIF_OUT("PRIMARY_TDM_TX_5", NULL,
  672. 0, SND_SOC_NOPM, 0, 0),
  673. SND_SOC_DAPM_AIF_OUT("PRIMARY_TDM_TX_6", NULL,
  674. 0, SND_SOC_NOPM, 0, 0),
  675. SND_SOC_DAPM_AIF_OUT("PRIMARY_TDM_TX_7", NULL,
  676. 0, SND_SOC_NOPM, 0, 0),
  677. SND_SOC_DAPM_AIF_IN("SEC_TDM_RX_0", NULL,
  678. 0, SND_SOC_NOPM, 0, 0),
  679. SND_SOC_DAPM_AIF_IN("SEC_TDM_RX_1", NULL,
  680. 0, SND_SOC_NOPM, 0, 0),
  681. SND_SOC_DAPM_AIF_IN("SEC_TDM_RX_2", NULL,
  682. 0, SND_SOC_NOPM, 0, 0),
  683. SND_SOC_DAPM_AIF_IN("SEC_TDM_RX_3", NULL,
  684. 0, SND_SOC_NOPM, 0, 0),
  685. SND_SOC_DAPM_AIF_IN("SEC_TDM_RX_4", NULL,
  686. 0, SND_SOC_NOPM, 0, 0),
  687. SND_SOC_DAPM_AIF_IN("SEC_TDM_RX_5", NULL,
  688. 0, SND_SOC_NOPM, 0, 0),
  689. SND_SOC_DAPM_AIF_IN("SEC_TDM_RX_6", NULL,
  690. 0, SND_SOC_NOPM, 0, 0),
  691. SND_SOC_DAPM_AIF_IN("SEC_TDM_RX_7", NULL,
  692. 0, SND_SOC_NOPM, 0, 0),
  693. SND_SOC_DAPM_AIF_OUT("SEC_TDM_TX_0", NULL,
  694. 0, SND_SOC_NOPM, 0, 0),
  695. SND_SOC_DAPM_AIF_OUT("SEC_TDM_TX_1", NULL,
  696. 0, SND_SOC_NOPM, 0, 0),
  697. SND_SOC_DAPM_AIF_OUT("SEC_TDM_TX_2", NULL,
  698. 0, SND_SOC_NOPM, 0, 0),
  699. SND_SOC_DAPM_AIF_OUT("SEC_TDM_TX_3", NULL,
  700. 0, SND_SOC_NOPM, 0, 0),
  701. SND_SOC_DAPM_AIF_OUT("SEC_TDM_TX_4", NULL,
  702. 0, SND_SOC_NOPM, 0, 0),
  703. SND_SOC_DAPM_AIF_OUT("SEC_TDM_TX_5", NULL,
  704. 0, SND_SOC_NOPM, 0, 0),
  705. SND_SOC_DAPM_AIF_OUT("SEC_TDM_TX_6", NULL,
  706. 0, SND_SOC_NOPM, 0, 0),
  707. SND_SOC_DAPM_AIF_OUT("SEC_TDM_TX_7", NULL,
  708. 0, SND_SOC_NOPM, 0, 0),
  709. SND_SOC_DAPM_AIF_IN("TERT_TDM_RX_0", NULL,
  710. 0, SND_SOC_NOPM, 0, 0),
  711. SND_SOC_DAPM_AIF_IN("TERT_TDM_RX_1", NULL,
  712. 0, SND_SOC_NOPM, 0, 0),
  713. SND_SOC_DAPM_AIF_IN("TERT_TDM_RX_2", NULL,
  714. 0, SND_SOC_NOPM, 0, 0),
  715. SND_SOC_DAPM_AIF_IN("TERT_TDM_RX_3", NULL,
  716. 0, SND_SOC_NOPM, 0, 0),
  717. SND_SOC_DAPM_AIF_IN("TERT_TDM_RX_4", NULL,
  718. 0, SND_SOC_NOPM, 0, 0),
  719. SND_SOC_DAPM_AIF_IN("TERT_TDM_RX_5", NULL,
  720. 0, SND_SOC_NOPM, 0, 0),
  721. SND_SOC_DAPM_AIF_IN("TERT_TDM_RX_6", NULL,
  722. 0, SND_SOC_NOPM, 0, 0),
  723. SND_SOC_DAPM_AIF_IN("TERT_TDM_RX_7", NULL,
  724. 0, SND_SOC_NOPM, 0, 0),
  725. SND_SOC_DAPM_AIF_OUT("TERT_TDM_TX_0", NULL,
  726. 0, SND_SOC_NOPM, 0, 0),
  727. SND_SOC_DAPM_AIF_OUT("TERT_TDM_TX_1", NULL,
  728. 0, SND_SOC_NOPM, 0, 0),
  729. SND_SOC_DAPM_AIF_OUT("TERT_TDM_TX_2", NULL,
  730. 0, SND_SOC_NOPM, 0, 0),
  731. SND_SOC_DAPM_AIF_OUT("TERT_TDM_TX_3", NULL,
  732. 0, SND_SOC_NOPM, 0, 0),
  733. SND_SOC_DAPM_AIF_OUT("TERT_TDM_TX_4", NULL,
  734. 0, SND_SOC_NOPM, 0, 0),
  735. SND_SOC_DAPM_AIF_OUT("TERT_TDM_TX_5", NULL,
  736. 0, SND_SOC_NOPM, 0, 0),
  737. SND_SOC_DAPM_AIF_OUT("TERT_TDM_TX_6", NULL,
  738. 0, SND_SOC_NOPM, 0, 0),
  739. SND_SOC_DAPM_AIF_OUT("TERT_TDM_TX_7", NULL,
  740. 0, SND_SOC_NOPM, 0, 0),
  741. SND_SOC_DAPM_AIF_IN("QUAT_TDM_RX_0", NULL,
  742. 0, SND_SOC_NOPM, 0, 0),
  743. SND_SOC_DAPM_AIF_IN("QUAT_TDM_RX_1", NULL,
  744. 0, SND_SOC_NOPM, 0, 0),
  745. SND_SOC_DAPM_AIF_IN("QUAT_TDM_RX_2", NULL,
  746. 0, SND_SOC_NOPM, 0, 0),
  747. SND_SOC_DAPM_AIF_IN("QUAT_TDM_RX_3", NULL,
  748. 0, SND_SOC_NOPM, 0, 0),
  749. SND_SOC_DAPM_AIF_IN("QUAT_TDM_RX_4", NULL,
  750. 0, SND_SOC_NOPM, 0, 0),
  751. SND_SOC_DAPM_AIF_IN("QUAT_TDM_RX_5", NULL,
  752. 0, SND_SOC_NOPM, 0, 0),
  753. SND_SOC_DAPM_AIF_IN("QUAT_TDM_RX_6", NULL,
  754. 0, SND_SOC_NOPM, 0, 0),
  755. SND_SOC_DAPM_AIF_IN("QUAT_TDM_RX_7", NULL,
  756. 0, SND_SOC_NOPM, 0, 0),
  757. SND_SOC_DAPM_AIF_OUT("QUAT_TDM_TX_0", NULL,
  758. 0, SND_SOC_NOPM, 0, 0),
  759. SND_SOC_DAPM_AIF_OUT("QUAT_TDM_TX_1", NULL,
  760. 0, SND_SOC_NOPM, 0, 0),
  761. SND_SOC_DAPM_AIF_OUT("QUAT_TDM_TX_2", NULL,
  762. 0, SND_SOC_NOPM, 0, 0),
  763. SND_SOC_DAPM_AIF_OUT("QUAT_TDM_TX_3", NULL,
  764. 0, SND_SOC_NOPM, 0, 0),
  765. SND_SOC_DAPM_AIF_OUT("QUAT_TDM_TX_4", NULL,
  766. 0, SND_SOC_NOPM, 0, 0),
  767. SND_SOC_DAPM_AIF_OUT("QUAT_TDM_TX_5", NULL,
  768. 0, SND_SOC_NOPM, 0, 0),
  769. SND_SOC_DAPM_AIF_OUT("QUAT_TDM_TX_6", NULL,
  770. 0, SND_SOC_NOPM, 0, 0),
  771. SND_SOC_DAPM_AIF_OUT("QUAT_TDM_TX_7", NULL,
  772. 0, SND_SOC_NOPM, 0, 0),
  773. SND_SOC_DAPM_AIF_IN("QUIN_TDM_RX_0", NULL,
  774. 0, SND_SOC_NOPM, 0, 0),
  775. SND_SOC_DAPM_AIF_IN("QUIN_TDM_RX_1", NULL,
  776. 0, SND_SOC_NOPM, 0, 0),
  777. SND_SOC_DAPM_AIF_IN("QUIN_TDM_RX_2", NULL,
  778. 0, SND_SOC_NOPM, 0, 0),
  779. SND_SOC_DAPM_AIF_IN("QUIN_TDM_RX_3", NULL,
  780. 0, SND_SOC_NOPM, 0, 0),
  781. SND_SOC_DAPM_AIF_IN("QUIN_TDM_RX_4", NULL,
  782. 0, SND_SOC_NOPM, 0, 0),
  783. SND_SOC_DAPM_AIF_IN("QUIN_TDM_RX_5", NULL,
  784. 0, SND_SOC_NOPM, 0, 0),
  785. SND_SOC_DAPM_AIF_IN("QUIN_TDM_RX_6", NULL,
  786. 0, SND_SOC_NOPM, 0, 0),
  787. SND_SOC_DAPM_AIF_IN("QUIN_TDM_RX_7", NULL,
  788. 0, SND_SOC_NOPM, 0, 0),
  789. SND_SOC_DAPM_AIF_OUT("QUIN_TDM_TX_0", NULL,
  790. 0, SND_SOC_NOPM, 0, 0),
  791. SND_SOC_DAPM_AIF_OUT("QUIN_TDM_TX_1", NULL,
  792. 0, SND_SOC_NOPM, 0, 0),
  793. SND_SOC_DAPM_AIF_OUT("QUIN_TDM_TX_2", NULL,
  794. 0, SND_SOC_NOPM, 0, 0),
  795. SND_SOC_DAPM_AIF_OUT("QUIN_TDM_TX_3", NULL,
  796. 0, SND_SOC_NOPM, 0, 0),
  797. SND_SOC_DAPM_AIF_OUT("QUIN_TDM_TX_4", NULL,
  798. 0, SND_SOC_NOPM, 0, 0),
  799. SND_SOC_DAPM_AIF_OUT("QUIN_TDM_TX_5", NULL,
  800. 0, SND_SOC_NOPM, 0, 0),
  801. SND_SOC_DAPM_AIF_OUT("QUIN_TDM_TX_6", NULL,
  802. 0, SND_SOC_NOPM, 0, 0),
  803. SND_SOC_DAPM_AIF_OUT("QUIN_TDM_TX_7", NULL,
  804. 0, SND_SOC_NOPM, 0, 0),
  805. SND_SOC_DAPM_AIF_OUT("DISPLAY_PORT_RX", "NULL", 0, SND_SOC_NOPM, 0, 0),
  806. SND_SOC_DAPM_AIF_IN("WSA_CODEC_DMA_RX_0", "NULL",
  807. 0, SND_SOC_NOPM, 0, 0),
  808. SND_SOC_DAPM_AIF_OUT("WSA_CODEC_DMA_TX_0", "NULL",
  809. 0, SND_SOC_NOPM, 0, 0),
  810. SND_SOC_DAPM_AIF_IN("WSA_CODEC_DMA_RX_1", "NULL",
  811. 0, SND_SOC_NOPM, 0, 0),
  812. SND_SOC_DAPM_AIF_OUT("WSA_CODEC_DMA_TX_1", "NULL",
  813. 0, SND_SOC_NOPM, 0, 0),
  814. SND_SOC_DAPM_AIF_OUT("WSA_CODEC_DMA_TX_2", "NULL",
  815. 0, SND_SOC_NOPM, 0, 0),
  816. SND_SOC_DAPM_AIF_OUT("VA_CODEC_DMA_TX_0", "NULL",
  817. 0, SND_SOC_NOPM, 0, 0),
  818. SND_SOC_DAPM_AIF_OUT("VA_CODEC_DMA_TX_1", "NULL",
  819. 0, SND_SOC_NOPM, 0, 0),
  820. SND_SOC_DAPM_AIF_OUT("VA_CODEC_DMA_TX_2", "NULL",
  821. 0, SND_SOC_NOPM, 0, 0),
  822. SND_SOC_DAPM_AIF_IN("RX_CODEC_DMA_RX_0", "NULL",
  823. 0, SND_SOC_NOPM, 0, 0),
  824. SND_SOC_DAPM_AIF_OUT("TX_CODEC_DMA_TX_0", "NULL",
  825. 0, SND_SOC_NOPM, 0, 0),
  826. SND_SOC_DAPM_AIF_IN("RX_CODEC_DMA_RX_1", "NULL",
  827. 0, SND_SOC_NOPM, 0, 0),
  828. SND_SOC_DAPM_AIF_OUT("TX_CODEC_DMA_TX_1", "NULL",
  829. 0, SND_SOC_NOPM, 0, 0),
  830. SND_SOC_DAPM_AIF_IN("RX_CODEC_DMA_RX_2", "NULL",
  831. 0, SND_SOC_NOPM, 0, 0),
  832. SND_SOC_DAPM_AIF_OUT("TX_CODEC_DMA_TX_2", "NULL",
  833. 0, SND_SOC_NOPM, 0, 0),
  834. SND_SOC_DAPM_AIF_IN("RX_CODEC_DMA_RX_3", "NULL",
  835. 0, SND_SOC_NOPM, 0, 0),
  836. SND_SOC_DAPM_AIF_OUT("TX_CODEC_DMA_TX_3", "NULL",
  837. 0, SND_SOC_NOPM, 0, 0),
  838. SND_SOC_DAPM_AIF_IN("RX_CODEC_DMA_RX_4", "NULL",
  839. 0, SND_SOC_NOPM, 0, 0),
  840. SND_SOC_DAPM_AIF_OUT("TX_CODEC_DMA_TX_4", "NULL",
  841. 0, SND_SOC_NOPM, 0, 0),
  842. SND_SOC_DAPM_AIF_IN("RX_CODEC_DMA_RX_5", "NULL",
  843. 0, SND_SOC_NOPM, 0, 0),
  844. SND_SOC_DAPM_AIF_OUT("TX_CODEC_DMA_TX_5", "NULL",
  845. 0, SND_SOC_NOPM, 0, 0),
  846. SND_SOC_DAPM_AIF_IN("RX_CODEC_DMA_RX_6", "NULL",
  847. 0, SND_SOC_NOPM, 0, 0),
  848. SND_SOC_DAPM_AIF_IN("RX_CODEC_DMA_RX_7", "NULL",
  849. 0, SND_SOC_NOPM, 0, 0),
  850. };
  851. static const struct snd_soc_component_driver q6afe_dai_component = {
  852. .name = "q6afe-dai-component",
  853. .dapm_widgets = q6afe_dai_widgets,
  854. .num_dapm_widgets = ARRAY_SIZE(q6afe_dai_widgets),
  855. .dapm_routes = q6afe_dapm_routes,
  856. .num_dapm_routes = ARRAY_SIZE(q6afe_dapm_routes),
  857. .of_xlate_dai_name = q6dsp_audio_ports_of_xlate_dai_name,
  858. };
  859. static void of_q6afe_parse_dai_data(struct device *dev,
  860. struct q6afe_dai_data *data)
  861. {
  862. struct device_node *node;
  863. int ret;
  864. for_each_child_of_node(dev->of_node, node) {
  865. unsigned int lines[Q6AFE_MAX_MI2S_LINES];
  866. struct q6afe_dai_priv_data *priv;
  867. int id, i, num_lines;
  868. ret = of_property_read_u32(node, "reg", &id);
  869. if (ret || id < 0 || id >= AFE_PORT_MAX) {
  870. dev_err(dev, "valid dai id not found:%d\n", ret);
  871. continue;
  872. }
  873. switch (id) {
  874. /* MI2S specific properties */
  875. case QUINARY_MI2S_RX ... QUINARY_MI2S_TX:
  876. case PRIMARY_MI2S_RX ... QUATERNARY_MI2S_TX:
  877. priv = &data->priv[id];
  878. ret = of_property_read_variable_u32_array(node,
  879. "qcom,sd-lines",
  880. lines, 0,
  881. Q6AFE_MAX_MI2S_LINES);
  882. if (ret < 0)
  883. num_lines = 0;
  884. else
  885. num_lines = ret;
  886. priv->sd_line_mask = 0;
  887. for (i = 0; i < num_lines; i++)
  888. priv->sd_line_mask |= BIT(lines[i]);
  889. break;
  890. case PRIMARY_TDM_RX_0 ... QUINARY_TDM_TX_7:
  891. priv = &data->priv[id];
  892. ret = of_property_read_u32(node, "qcom,tdm-sync-mode",
  893. &priv->sync_mode);
  894. if (ret) {
  895. dev_err(dev, "No Sync mode from DT\n");
  896. break;
  897. }
  898. ret = of_property_read_u32(node, "qcom,tdm-sync-src",
  899. &priv->sync_src);
  900. if (ret) {
  901. dev_err(dev, "No Sync Src from DT\n");
  902. break;
  903. }
  904. ret = of_property_read_u32(node, "qcom,tdm-data-out",
  905. &priv->data_out_enable);
  906. if (ret) {
  907. dev_err(dev, "No Data out enable from DT\n");
  908. break;
  909. }
  910. ret = of_property_read_u32(node, "qcom,tdm-invert-sync",
  911. &priv->invert_sync);
  912. if (ret) {
  913. dev_err(dev, "No Invert sync from DT\n");
  914. break;
  915. }
  916. ret = of_property_read_u32(node, "qcom,tdm-data-delay",
  917. &priv->data_delay);
  918. if (ret) {
  919. dev_err(dev, "No Data Delay from DT\n");
  920. break;
  921. }
  922. ret = of_property_read_u32(node, "qcom,tdm-data-align",
  923. &priv->data_align);
  924. if (ret) {
  925. dev_err(dev, "No Data align from DT\n");
  926. break;
  927. }
  928. break;
  929. default:
  930. break;
  931. }
  932. }
  933. }
  934. static int q6afe_dai_dev_probe(struct platform_device *pdev)
  935. {
  936. struct q6dsp_audio_port_dai_driver_config cfg;
  937. struct snd_soc_dai_driver *dais;
  938. struct q6afe_dai_data *dai_data;
  939. struct device *dev = &pdev->dev;
  940. int num_dais;
  941. dai_data = devm_kzalloc(dev, sizeof(*dai_data), GFP_KERNEL);
  942. if (!dai_data)
  943. return -ENOMEM;
  944. dev_set_drvdata(dev, dai_data);
  945. of_q6afe_parse_dai_data(dev, dai_data);
  946. cfg.probe = msm_dai_q6_dai_probe;
  947. cfg.remove = msm_dai_q6_dai_remove;
  948. cfg.q6hdmi_ops = &q6hdmi_ops;
  949. cfg.q6slim_ops = &q6slim_ops;
  950. cfg.q6i2s_ops = &q6i2s_ops;
  951. cfg.q6tdm_ops = &q6tdm_ops;
  952. cfg.q6dma_ops = &q6dma_ops;
  953. dais = q6dsp_audio_ports_set_config(dev, &cfg, &num_dais);
  954. return devm_snd_soc_register_component(dev, &q6afe_dai_component, dais, num_dais);
  955. }
  956. #ifdef CONFIG_OF
  957. static const struct of_device_id q6afe_dai_device_id[] = {
  958. { .compatible = "qcom,q6afe-dais" },
  959. {},
  960. };
  961. MODULE_DEVICE_TABLE(of, q6afe_dai_device_id);
  962. #endif
  963. static struct platform_driver q6afe_dai_platform_driver = {
  964. .driver = {
  965. .name = "q6afe-dai",
  966. .of_match_table = of_match_ptr(q6afe_dai_device_id),
  967. },
  968. .probe = q6afe_dai_dev_probe,
  969. };
  970. module_platform_driver(q6afe_dai_platform_driver);
  971. MODULE_DESCRIPTION("Q6 Audio Frontend dai driver");
  972. MODULE_LICENSE("GPL v2");