msm-dai-stub-v2.c 9.9 KB

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