msm-dai-stub-v2.c 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /* Copyright (c) 2013-2014, 2017, 2019 The Linux Foundation. All rights reserved.
  3. */
  4. #include <linux/init.h>
  5. #include <linux/module.h>
  6. #include <linux/device.h>
  7. #include <linux/of_device.h>
  8. #include <linux/platform_device.h>
  9. #include <sound/core.h>
  10. #include <sound/pcm.h>
  11. #include <sound/soc.h>
  12. enum {
  13. STUB_RX,
  14. STUB_TX,
  15. STUB_1_RX,
  16. STUB_1_TX,
  17. STUB_DTMF_TX,
  18. STUB_HOST_RX_CAPTURE_TX,
  19. STUB_HOST_RX_PLAYBACK_RX,
  20. STUB_HOST_TX_CAPTURE_TX,
  21. STUB_HOST_TX_PLAYBACK_RX,
  22. };
  23. static int msm_dai_stub_set_channel_map(struct snd_soc_dai *dai,
  24. unsigned int tx_num, unsigned int *tx_slot,
  25. unsigned int rx_num, unsigned int *rx_slot)
  26. {
  27. pr_debug("%s:\n", __func__);
  28. return 0;
  29. }
  30. static struct snd_soc_dai_ops msm_dai_stub_ops = {
  31. .set_channel_map = msm_dai_stub_set_channel_map,
  32. };
  33. static int msm_dai_stub_add_route(struct snd_soc_dai *dai)
  34. {
  35. struct snd_soc_dapm_route intercon;
  36. struct snd_soc_dapm_context *dapm;
  37. if (!dai || !dai->driver) {
  38. pr_err("%s Invalid params\n", __func__);
  39. return -EINVAL;
  40. }
  41. dapm = snd_soc_component_get_dapm(dai->component);
  42. memset(&intercon, 0, sizeof(intercon));
  43. if (dai->driver->playback.stream_name &&
  44. dai->driver->playback.aif_name) {
  45. dev_dbg(dai->dev, "%s add route for widget %s",
  46. __func__, dai->driver->playback.stream_name);
  47. intercon.source = dai->driver->playback.aif_name;
  48. intercon.sink = dai->driver->playback.stream_name;
  49. dev_dbg(dai->dev, "%s src %s sink %s\n",
  50. __func__, intercon.source, intercon.sink);
  51. snd_soc_dapm_add_routes(dapm, &intercon, 1);
  52. }
  53. if (dai->driver->capture.stream_name &&
  54. dai->driver->capture.aif_name) {
  55. dev_dbg(dai->dev, "%s add route for widget %s",
  56. __func__, dai->driver->capture.stream_name);
  57. intercon.sink = dai->driver->capture.aif_name;
  58. intercon.source = dai->driver->capture.stream_name;
  59. dev_dbg(dai->dev, "%s src %s sink %s\n",
  60. __func__, intercon.source, intercon.sink);
  61. snd_soc_dapm_add_routes(dapm, &intercon, 1);
  62. }
  63. return 0;
  64. }
  65. static int msm_dai_stub_dai_probe(struct snd_soc_dai *dai)
  66. {
  67. return msm_dai_stub_add_route(dai);
  68. }
  69. static int msm_dai_stub_dai_remove(struct snd_soc_dai *dai)
  70. {
  71. pr_debug("%s:\n", __func__);
  72. return 0;
  73. }
  74. static struct snd_soc_dai_driver msm_dai_stub_dai_rx = {
  75. .playback = {
  76. .stream_name = "Stub Playback",
  77. .aif_name = "STUB_RX",
  78. .rates = SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_8000 |
  79. SNDRV_PCM_RATE_16000,
  80. .formats = SNDRV_PCM_FMTBIT_S16_LE,
  81. .channels_min = 1,
  82. .channels_max = 2,
  83. .rate_min = 8000,
  84. .rate_max = 48000,
  85. },
  86. .ops = &msm_dai_stub_ops,
  87. .probe = &msm_dai_stub_dai_probe,
  88. .remove = &msm_dai_stub_dai_remove,
  89. };
  90. static struct snd_soc_dai_driver msm_dai_stub_dai_tx[] = {
  91. {
  92. .capture = {
  93. .stream_name = "Stub Capture",
  94. .aif_name = "STUB_TX",
  95. .rates = SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_8000 |
  96. SNDRV_PCM_RATE_16000,
  97. .formats = SNDRV_PCM_FMTBIT_S16_LE,
  98. .channels_min = 1,
  99. .channels_max = 2,
  100. .rate_min = 8000,
  101. .rate_max = 48000,
  102. },
  103. .ops = &msm_dai_stub_ops,
  104. .probe = &msm_dai_stub_dai_probe,
  105. .remove = &msm_dai_stub_dai_remove,
  106. },
  107. {
  108. .capture = {
  109. .stream_name = "Stub1 Capture",
  110. .aif_name = "STUB_1_TX",
  111. .rates = SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_8000 |
  112. SNDRV_PCM_RATE_16000,
  113. .formats = SNDRV_PCM_FMTBIT_S16_LE,
  114. .channels_min = 1,
  115. .channels_max = 2,
  116. .rate_min = 8000,
  117. .rate_max = 48000,
  118. },
  119. .ops = &msm_dai_stub_ops,
  120. .probe = &msm_dai_stub_dai_probe,
  121. .remove = &msm_dai_stub_dai_remove,
  122. }
  123. };
  124. static struct snd_soc_dai_driver msm_dai_stub_dtmf_tx_dai = {
  125. .capture = {
  126. .stream_name = "DTMF TX",
  127. .aif_name = "STUB_DTMF_TX",
  128. .rates = SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_8000 |
  129. SNDRV_PCM_RATE_16000,
  130. .formats = SNDRV_PCM_FMTBIT_S16_LE,
  131. .channels_min = 1,
  132. .channels_max = 2,
  133. .rate_min = 8000,
  134. .rate_max = 48000,
  135. },
  136. .ops = &msm_dai_stub_ops,
  137. .probe = &msm_dai_stub_dai_probe,
  138. .remove = &msm_dai_stub_dai_remove,
  139. };
  140. static struct snd_soc_dai_driver msm_dai_stub_host_capture_tx_dai[] = {
  141. {
  142. .capture = {
  143. .stream_name = "CS-VOICE HOST RX CAPTURE",
  144. .aif_name = "STUB_HOST_RX_CAPTURE_TX",
  145. .rates = SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_8000 |
  146. SNDRV_PCM_RATE_16000,
  147. .formats = SNDRV_PCM_FMTBIT_S16_LE,
  148. .channels_min = 1,
  149. .channels_max = 2,
  150. .rate_min = 8000,
  151. .rate_max = 48000,
  152. },
  153. .ops = &msm_dai_stub_ops,
  154. .probe = &msm_dai_stub_dai_probe,
  155. .remove = &msm_dai_stub_dai_remove,
  156. },
  157. {
  158. .capture = {
  159. .stream_name = "CS-VOICE HOST TX CAPTURE",
  160. .aif_name = "STUB_HOST_TX_CAPTURE_TX",
  161. .rates = SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_8000 |
  162. SNDRV_PCM_RATE_16000,
  163. .formats = SNDRV_PCM_FMTBIT_S16_LE,
  164. .channels_min = 1,
  165. .channels_max = 2,
  166. .rate_min = 8000,
  167. .rate_max = 48000,
  168. },
  169. .ops = &msm_dai_stub_ops,
  170. .probe = &msm_dai_stub_dai_probe,
  171. .remove = &msm_dai_stub_dai_remove,
  172. },
  173. };
  174. static struct snd_soc_dai_driver msm_dai_stub_host_playback_rx_dai[] = {
  175. {
  176. .playback = {
  177. .stream_name = "CS-VOICE HOST RX PLAYBACK",
  178. .aif_name = "STUB_HOST_RX_PLAYBACK_RX",
  179. .rates = SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_8000 |
  180. SNDRV_PCM_RATE_16000,
  181. .formats = SNDRV_PCM_FMTBIT_S16_LE,
  182. .channels_min = 1,
  183. .channels_max = 2,
  184. .rate_min = 8000,
  185. .rate_max = 48000,
  186. },
  187. .ops = &msm_dai_stub_ops,
  188. .probe = &msm_dai_stub_dai_probe,
  189. .remove = &msm_dai_stub_dai_remove,
  190. },
  191. {
  192. .playback = {
  193. .stream_name = "CS-VOICE HOST TX PLAYBACK",
  194. .aif_name = "STUB_HOST_TX_PLAYBACK_RX",
  195. .rates = SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_8000 |
  196. SNDRV_PCM_RATE_16000,
  197. .formats = SNDRV_PCM_FMTBIT_S16_LE,
  198. .channels_min = 1,
  199. .channels_max = 2,
  200. .rate_min = 8000,
  201. .rate_max = 48000,
  202. },
  203. .ops = &msm_dai_stub_ops,
  204. .probe = &msm_dai_stub_dai_probe,
  205. .remove = &msm_dai_stub_dai_remove,
  206. },
  207. };
  208. static const struct snd_soc_component_driver msm_dai_stub_component = {
  209. .name = "msm-dai-stub-dev",
  210. };
  211. static int msm_dai_stub_dev_probe(struct platform_device *pdev)
  212. {
  213. int rc, id = -1;
  214. const char *stub_dev_id = "qcom,msm-dai-stub-dev-id";
  215. rc = of_property_read_u32(pdev->dev.of_node, stub_dev_id, &id);
  216. if (rc) {
  217. dev_err(&pdev->dev,
  218. "%s: missing %s in dt node\n", __func__, stub_dev_id);
  219. return rc;
  220. }
  221. pdev->id = id;
  222. pr_debug("%s: dev name %s, id:%d\n", __func__,
  223. dev_name(&pdev->dev), pdev->id);
  224. switch (id) {
  225. case STUB_RX:
  226. rc = snd_soc_register_component(&pdev->dev,
  227. &msm_dai_stub_component, &msm_dai_stub_dai_rx, 1);
  228. break;
  229. case STUB_TX:
  230. rc = snd_soc_register_component(&pdev->dev,
  231. &msm_dai_stub_component, &msm_dai_stub_dai_tx[0], 1);
  232. break;
  233. case STUB_1_TX:
  234. rc = snd_soc_register_component(&pdev->dev,
  235. &msm_dai_stub_component, &msm_dai_stub_dai_tx[1], 1);
  236. break;
  237. case STUB_DTMF_TX:
  238. rc = snd_soc_register_component(&pdev->dev,
  239. &msm_dai_stub_component,
  240. &msm_dai_stub_dtmf_tx_dai, 1);
  241. break;
  242. case STUB_HOST_RX_CAPTURE_TX:
  243. rc = snd_soc_register_component(&pdev->dev,
  244. &msm_dai_stub_component,
  245. &msm_dai_stub_host_capture_tx_dai[0], 1);
  246. break;
  247. case STUB_HOST_TX_CAPTURE_TX:
  248. rc = snd_soc_register_component(&pdev->dev,
  249. &msm_dai_stub_component,
  250. &msm_dai_stub_host_capture_tx_dai[1], 1);
  251. break;
  252. case STUB_HOST_RX_PLAYBACK_RX:
  253. rc = snd_soc_register_component(&pdev->dev,
  254. &msm_dai_stub_component,
  255. &msm_dai_stub_host_playback_rx_dai[0], 1);
  256. break;
  257. case STUB_HOST_TX_PLAYBACK_RX:
  258. rc = snd_soc_register_component(&pdev->dev,
  259. &msm_dai_stub_component,
  260. &msm_dai_stub_host_playback_rx_dai[1], 1);
  261. break;
  262. }
  263. return rc;
  264. }
  265. static int msm_dai_stub_dev_remove(struct platform_device *pdev)
  266. {
  267. snd_soc_unregister_component(&pdev->dev);
  268. return 0;
  269. }
  270. static const struct of_device_id msm_dai_stub_dev_dt_match[] = {
  271. { .compatible = "qcom,msm-dai-stub-dev", },
  272. { }
  273. };
  274. MODULE_DEVICE_TABLE(of, msm_dai_stub_dev_dt_match);
  275. static struct platform_driver msm_dai_stub_dev = {
  276. .probe = msm_dai_stub_dev_probe,
  277. .remove = msm_dai_stub_dev_remove,
  278. .driver = {
  279. .name = "msm-dai-stub-dev",
  280. .owner = THIS_MODULE,
  281. .of_match_table = msm_dai_stub_dev_dt_match,
  282. .suppress_bind_attrs = true,
  283. },
  284. };
  285. static int msm_dai_stub_probe(struct platform_device *pdev)
  286. {
  287. int rc = 0;
  288. dev_dbg(&pdev->dev, "dev name %s\n", dev_name(&pdev->dev));
  289. rc = of_platform_populate(pdev->dev.of_node, NULL, NULL, &pdev->dev);
  290. if (rc) {
  291. dev_err(&pdev->dev, "%s: failed to add child nodes, rc=%d\n",
  292. __func__, rc);
  293. } else
  294. dev_dbg(&pdev->dev, "%s: added child node\n", __func__);
  295. return rc;
  296. }
  297. static int msm_dai_stub_remove(struct platform_device *pdev)
  298. {
  299. pr_debug("%s:\n", __func__);
  300. return 0;
  301. }
  302. static const struct of_device_id msm_dai_stub_dt_match[] = {
  303. {.compatible = "qcom,msm-dai-stub"},
  304. {}
  305. };
  306. MODULE_DEVICE_TABLE(of, msm_dai_stub_dt_match);
  307. static struct platform_driver msm_dai_stub_driver = {
  308. .probe = msm_dai_stub_probe,
  309. .remove = msm_dai_stub_remove,
  310. .driver = {
  311. .name = "msm-dai-stub",
  312. .owner = THIS_MODULE,
  313. .of_match_table = msm_dai_stub_dt_match,
  314. .suppress_bind_attrs = true,
  315. },
  316. };
  317. int __init msm_dai_stub_init(void)
  318. {
  319. int rc = 0;
  320. pr_debug("%s:\n", __func__);
  321. rc = platform_driver_register(&msm_dai_stub_driver);
  322. if (rc) {
  323. pr_err("%s: fail to register dai q6 driver", __func__);
  324. goto fail;
  325. }
  326. rc = platform_driver_register(&msm_dai_stub_dev);
  327. if (rc) {
  328. pr_err("%s: fail to register dai q6 dev driver", __func__);
  329. goto dai_stub_dev_fail;
  330. }
  331. return rc;
  332. dai_stub_dev_fail:
  333. platform_driver_unregister(&msm_dai_stub_driver);
  334. fail:
  335. return rc;
  336. }
  337. void msm_dai_stub_exit(void)
  338. {
  339. pr_debug("%s:\n", __func__);
  340. platform_driver_unregister(&msm_dai_stub_dev);
  341. platform_driver_unregister(&msm_dai_stub_driver);
  342. }
  343. /* Module information */
  344. MODULE_DESCRIPTION("MSM Stub DSP DAI driver");
  345. MODULE_LICENSE("GPL v2");