acp5x-i2s.c 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416
  1. // SPDX-License-Identifier: GPL-2.0+
  2. //
  3. // AMD ALSA SoC PCM Driver
  4. //
  5. // Copyright (C) 2021 Advanced Micro Devices, Inc. All rights reserved.
  6. #include <linux/platform_device.h>
  7. #include <linux/module.h>
  8. #include <linux/err.h>
  9. #include <linux/io.h>
  10. #include <sound/pcm_params.h>
  11. #include <sound/soc.h>
  12. #include <sound/soc-dai.h>
  13. #include <linux/dma-mapping.h>
  14. #include "acp5x.h"
  15. #define DRV_NAME "acp5x_i2s_playcap"
  16. static int acp5x_i2s_set_fmt(struct snd_soc_dai *cpu_dai,
  17. unsigned int fmt)
  18. {
  19. struct i2s_dev_data *adata;
  20. int mode;
  21. adata = snd_soc_dai_get_drvdata(cpu_dai);
  22. mode = fmt & SND_SOC_DAIFMT_FORMAT_MASK;
  23. switch (mode) {
  24. case SND_SOC_DAIFMT_I2S:
  25. adata->tdm_mode = TDM_DISABLE;
  26. break;
  27. case SND_SOC_DAIFMT_DSP_A:
  28. adata->tdm_mode = TDM_ENABLE;
  29. break;
  30. default:
  31. return -EINVAL;
  32. }
  33. mode = fmt & SND_SOC_DAIFMT_CLOCK_PROVIDER_MASK;
  34. switch (mode) {
  35. case SND_SOC_DAIFMT_BP_FP:
  36. adata->master_mode = I2S_MASTER_MODE_ENABLE;
  37. break;
  38. case SND_SOC_DAIFMT_BC_FC:
  39. adata->master_mode = I2S_MASTER_MODE_DISABLE;
  40. break;
  41. }
  42. return 0;
  43. }
  44. static int acp5x_i2s_set_tdm_slot(struct snd_soc_dai *cpu_dai,
  45. u32 tx_mask, u32 rx_mask,
  46. int slots, int slot_width)
  47. {
  48. struct i2s_dev_data *adata;
  49. u32 frm_len;
  50. u16 slot_len;
  51. adata = snd_soc_dai_get_drvdata(cpu_dai);
  52. /* These values are as per Hardware Spec */
  53. switch (slot_width) {
  54. case SLOT_WIDTH_8:
  55. slot_len = 8;
  56. break;
  57. case SLOT_WIDTH_16:
  58. slot_len = 16;
  59. break;
  60. case SLOT_WIDTH_24:
  61. slot_len = 24;
  62. break;
  63. case SLOT_WIDTH_32:
  64. slot_len = 0;
  65. break;
  66. default:
  67. return -EINVAL;
  68. }
  69. frm_len = FRM_LEN | (slots << 15) | (slot_len << 18);
  70. adata->tdm_fmt = frm_len;
  71. return 0;
  72. }
  73. static int acp5x_i2s_hwparams(struct snd_pcm_substream *substream,
  74. struct snd_pcm_hw_params *params,
  75. struct snd_soc_dai *dai)
  76. {
  77. struct i2s_stream_instance *rtd;
  78. struct snd_soc_pcm_runtime *prtd;
  79. struct snd_soc_card *card;
  80. struct acp5x_platform_info *pinfo;
  81. struct i2s_dev_data *adata;
  82. u32 val;
  83. u32 reg_val, frmt_reg;
  84. u32 lrclk_div_val, bclk_div_val;
  85. lrclk_div_val = 0;
  86. bclk_div_val = 0;
  87. prtd = asoc_substream_to_rtd(substream);
  88. rtd = substream->runtime->private_data;
  89. card = prtd->card;
  90. adata = snd_soc_dai_get_drvdata(dai);
  91. pinfo = snd_soc_card_get_drvdata(card);
  92. if (pinfo) {
  93. if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK)
  94. rtd->i2s_instance = pinfo->play_i2s_instance;
  95. else
  96. rtd->i2s_instance = pinfo->cap_i2s_instance;
  97. }
  98. /* These values are as per Hardware Spec */
  99. switch (params_format(params)) {
  100. case SNDRV_PCM_FORMAT_U8:
  101. case SNDRV_PCM_FORMAT_S8:
  102. rtd->xfer_resolution = 0x0;
  103. break;
  104. case SNDRV_PCM_FORMAT_S16_LE:
  105. rtd->xfer_resolution = 0x02;
  106. break;
  107. case SNDRV_PCM_FORMAT_S24_LE:
  108. rtd->xfer_resolution = 0x04;
  109. break;
  110. case SNDRV_PCM_FORMAT_S32_LE:
  111. rtd->xfer_resolution = 0x05;
  112. break;
  113. default:
  114. return -EINVAL;
  115. }
  116. if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) {
  117. switch (rtd->i2s_instance) {
  118. case I2S_HS_INSTANCE:
  119. reg_val = ACP_HSTDM_ITER;
  120. frmt_reg = ACP_HSTDM_TXFRMT;
  121. break;
  122. case I2S_SP_INSTANCE:
  123. default:
  124. reg_val = ACP_I2STDM_ITER;
  125. frmt_reg = ACP_I2STDM_TXFRMT;
  126. }
  127. } else {
  128. switch (rtd->i2s_instance) {
  129. case I2S_HS_INSTANCE:
  130. reg_val = ACP_HSTDM_IRER;
  131. frmt_reg = ACP_HSTDM_RXFRMT;
  132. break;
  133. case I2S_SP_INSTANCE:
  134. default:
  135. reg_val = ACP_I2STDM_IRER;
  136. frmt_reg = ACP_I2STDM_RXFRMT;
  137. }
  138. }
  139. if (adata->tdm_mode) {
  140. val = acp_readl(rtd->acp5x_base + reg_val);
  141. acp_writel(val | 0x2, rtd->acp5x_base + reg_val);
  142. acp_writel(adata->tdm_fmt, rtd->acp5x_base + frmt_reg);
  143. }
  144. val = acp_readl(rtd->acp5x_base + reg_val);
  145. val &= ~ACP5x_ITER_IRER_SAMP_LEN_MASK;
  146. val = val | (rtd->xfer_resolution << 3);
  147. acp_writel(val, rtd->acp5x_base + reg_val);
  148. if (adata->master_mode) {
  149. switch (params_format(params)) {
  150. case SNDRV_PCM_FORMAT_S16_LE:
  151. switch (params_rate(params)) {
  152. case 8000:
  153. bclk_div_val = 768;
  154. break;
  155. case 16000:
  156. bclk_div_val = 384;
  157. break;
  158. case 24000:
  159. bclk_div_val = 256;
  160. break;
  161. case 32000:
  162. bclk_div_val = 192;
  163. break;
  164. case 44100:
  165. case 48000:
  166. bclk_div_val = 128;
  167. break;
  168. case 88200:
  169. case 96000:
  170. bclk_div_val = 64;
  171. break;
  172. case 192000:
  173. bclk_div_val = 32;
  174. break;
  175. default:
  176. return -EINVAL;
  177. }
  178. lrclk_div_val = 32;
  179. break;
  180. case SNDRV_PCM_FORMAT_S32_LE:
  181. switch (params_rate(params)) {
  182. case 8000:
  183. bclk_div_val = 384;
  184. break;
  185. case 16000:
  186. bclk_div_val = 192;
  187. break;
  188. case 24000:
  189. bclk_div_val = 128;
  190. break;
  191. case 32000:
  192. bclk_div_val = 96;
  193. break;
  194. case 44100:
  195. case 48000:
  196. bclk_div_val = 64;
  197. break;
  198. case 88200:
  199. case 96000:
  200. bclk_div_val = 32;
  201. break;
  202. case 192000:
  203. bclk_div_val = 16;
  204. break;
  205. default:
  206. return -EINVAL;
  207. }
  208. lrclk_div_val = 64;
  209. break;
  210. default:
  211. return -EINVAL;
  212. }
  213. rtd->lrclk_div = lrclk_div_val;
  214. rtd->bclk_div = bclk_div_val;
  215. }
  216. return 0;
  217. }
  218. static int acp5x_i2s_trigger(struct snd_pcm_substream *substream,
  219. int cmd, struct snd_soc_dai *dai)
  220. {
  221. struct i2s_stream_instance *rtd;
  222. struct i2s_dev_data *adata;
  223. u32 ret, val, period_bytes, reg_val, ier_val, water_val;
  224. u32 buf_size, buf_reg;
  225. adata = snd_soc_dai_get_drvdata(dai);
  226. rtd = substream->runtime->private_data;
  227. period_bytes = frames_to_bytes(substream->runtime,
  228. substream->runtime->period_size);
  229. buf_size = frames_to_bytes(substream->runtime,
  230. substream->runtime->buffer_size);
  231. switch (cmd) {
  232. case SNDRV_PCM_TRIGGER_START:
  233. case SNDRV_PCM_TRIGGER_RESUME:
  234. case SNDRV_PCM_TRIGGER_PAUSE_RELEASE:
  235. rtd->bytescount = acp_get_byte_count(rtd,
  236. substream->stream);
  237. if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) {
  238. switch (rtd->i2s_instance) {
  239. case I2S_HS_INSTANCE:
  240. water_val =
  241. ACP_HS_TX_INTR_WATERMARK_SIZE;
  242. reg_val = ACP_HSTDM_ITER;
  243. ier_val = ACP_HSTDM_IER;
  244. buf_reg = ACP_HS_TX_RINGBUFSIZE;
  245. break;
  246. case I2S_SP_INSTANCE:
  247. default:
  248. water_val =
  249. ACP_I2S_TX_INTR_WATERMARK_SIZE;
  250. reg_val = ACP_I2STDM_ITER;
  251. ier_val = ACP_I2STDM_IER;
  252. buf_reg = ACP_I2S_TX_RINGBUFSIZE;
  253. }
  254. } else {
  255. switch (rtd->i2s_instance) {
  256. case I2S_HS_INSTANCE:
  257. water_val =
  258. ACP_HS_RX_INTR_WATERMARK_SIZE;
  259. reg_val = ACP_HSTDM_IRER;
  260. ier_val = ACP_HSTDM_IER;
  261. buf_reg = ACP_HS_RX_RINGBUFSIZE;
  262. break;
  263. case I2S_SP_INSTANCE:
  264. default:
  265. water_val =
  266. ACP_I2S_RX_INTR_WATERMARK_SIZE;
  267. reg_val = ACP_I2STDM_IRER;
  268. ier_val = ACP_I2STDM_IER;
  269. buf_reg = ACP_I2S_RX_RINGBUFSIZE;
  270. }
  271. }
  272. acp_writel(period_bytes, rtd->acp5x_base + water_val);
  273. acp_writel(buf_size, rtd->acp5x_base + buf_reg);
  274. if (adata->master_mode)
  275. acp5x_set_i2s_clk(adata, rtd);
  276. val = acp_readl(rtd->acp5x_base + reg_val);
  277. val = val | BIT(0);
  278. acp_writel(val, rtd->acp5x_base + reg_val);
  279. acp_writel(1, rtd->acp5x_base + ier_val);
  280. ret = 0;
  281. break;
  282. case SNDRV_PCM_TRIGGER_STOP:
  283. case SNDRV_PCM_TRIGGER_SUSPEND:
  284. case SNDRV_PCM_TRIGGER_PAUSE_PUSH:
  285. if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) {
  286. switch (rtd->i2s_instance) {
  287. case I2S_HS_INSTANCE:
  288. reg_val = ACP_HSTDM_ITER;
  289. break;
  290. case I2S_SP_INSTANCE:
  291. default:
  292. reg_val = ACP_I2STDM_ITER;
  293. }
  294. } else {
  295. switch (rtd->i2s_instance) {
  296. case I2S_HS_INSTANCE:
  297. reg_val = ACP_HSTDM_IRER;
  298. break;
  299. case I2S_SP_INSTANCE:
  300. default:
  301. reg_val = ACP_I2STDM_IRER;
  302. }
  303. }
  304. val = acp_readl(rtd->acp5x_base + reg_val);
  305. val = val & ~BIT(0);
  306. acp_writel(val, rtd->acp5x_base + reg_val);
  307. if (!(acp_readl(rtd->acp5x_base + ACP_HSTDM_ITER) & BIT(0)) &&
  308. !(acp_readl(rtd->acp5x_base + ACP_HSTDM_IRER) & BIT(0)))
  309. acp_writel(0, rtd->acp5x_base + ACP_HSTDM_IER);
  310. if (!(acp_readl(rtd->acp5x_base + ACP_I2STDM_ITER) & BIT(0)) &&
  311. !(acp_readl(rtd->acp5x_base + ACP_I2STDM_IRER) & BIT(0)))
  312. acp_writel(0, rtd->acp5x_base + ACP_I2STDM_IER);
  313. ret = 0;
  314. break;
  315. default:
  316. ret = -EINVAL;
  317. break;
  318. }
  319. return ret;
  320. }
  321. static const struct snd_soc_dai_ops acp5x_i2s_dai_ops = {
  322. .hw_params = acp5x_i2s_hwparams,
  323. .trigger = acp5x_i2s_trigger,
  324. .set_fmt = acp5x_i2s_set_fmt,
  325. .set_tdm_slot = acp5x_i2s_set_tdm_slot,
  326. };
  327. static const struct snd_soc_component_driver acp5x_dai_component = {
  328. .name = "acp5x-i2s",
  329. .legacy_dai_naming = 1,
  330. };
  331. static struct snd_soc_dai_driver acp5x_i2s_dai = {
  332. .playback = {
  333. .rates = SNDRV_PCM_RATE_8000_96000,
  334. .formats = SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S8 |
  335. SNDRV_PCM_FMTBIT_U8 | SNDRV_PCM_FMTBIT_S32_LE,
  336. .channels_min = 2,
  337. .channels_max = 2,
  338. .rate_min = 8000,
  339. .rate_max = 96000,
  340. },
  341. .capture = {
  342. .rates = SNDRV_PCM_RATE_8000_96000,
  343. .formats = SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S8 |
  344. SNDRV_PCM_FMTBIT_U8 | SNDRV_PCM_FMTBIT_S32_LE,
  345. .channels_min = 2,
  346. .channels_max = 2,
  347. .rate_min = 8000,
  348. .rate_max = 96000,
  349. },
  350. .ops = &acp5x_i2s_dai_ops,
  351. };
  352. static int acp5x_dai_probe(struct platform_device *pdev)
  353. {
  354. struct resource *res;
  355. struct i2s_dev_data *adata;
  356. int ret;
  357. adata = devm_kzalloc(&pdev->dev, sizeof(struct i2s_dev_data),
  358. GFP_KERNEL);
  359. if (!adata)
  360. return -ENOMEM;
  361. res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
  362. if (!res) {
  363. dev_err(&pdev->dev, "IORESOURCE_MEM FAILED\n");
  364. return -ENOMEM;
  365. }
  366. adata->acp5x_base = devm_ioremap(&pdev->dev, res->start,
  367. resource_size(res));
  368. if (!adata->acp5x_base)
  369. return -ENOMEM;
  370. adata->master_mode = I2S_MASTER_MODE_ENABLE;
  371. dev_set_drvdata(&pdev->dev, adata);
  372. ret = devm_snd_soc_register_component(&pdev->dev,
  373. &acp5x_dai_component,
  374. &acp5x_i2s_dai, 1);
  375. if (ret)
  376. dev_err(&pdev->dev, "Fail to register acp i2s dai\n");
  377. return ret;
  378. }
  379. static struct platform_driver acp5x_dai_driver = {
  380. .probe = acp5x_dai_probe,
  381. .driver = {
  382. .name = "acp5x_i2s_playcap",
  383. },
  384. };
  385. module_platform_driver(acp5x_dai_driver);
  386. MODULE_AUTHOR("[email protected]");
  387. MODULE_DESCRIPTION("AMD ACP5.x CPU DAI Driver");
  388. MODULE_ALIAS("platform:" DRV_NAME);
  389. MODULE_LICENSE("GPL v2");