ux500_msp_i2s.c 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (C) ST-Ericsson SA 2012
  4. *
  5. * Author: Ola Lilja <ola.o.lilja@stericsson.com>,
  6. * Roger Nilsson <roger.xr.nilsson@stericsson.com>,
  7. * Sandeep Kaushik <sandeep.kaushik@st.com>
  8. * for ST-Ericsson.
  9. */
  10. #include <linux/module.h>
  11. #include <linux/platform_device.h>
  12. #include <linux/delay.h>
  13. #include <linux/slab.h>
  14. #include <linux/io.h>
  15. #include <linux/of.h>
  16. #include <sound/soc.h>
  17. #include "ux500_msp_i2s.h"
  18. /* Protocol desciptors */
  19. static const struct msp_protdesc prot_descs[] = {
  20. { /* I2S */
  21. MSP_SINGLE_PHASE,
  22. MSP_SINGLE_PHASE,
  23. MSP_PHASE2_START_MODE_IMEDIATE,
  24. MSP_PHASE2_START_MODE_IMEDIATE,
  25. MSP_BTF_MS_BIT_FIRST,
  26. MSP_BTF_MS_BIT_FIRST,
  27. MSP_FRAME_LEN_1,
  28. MSP_FRAME_LEN_1,
  29. MSP_FRAME_LEN_1,
  30. MSP_FRAME_LEN_1,
  31. MSP_ELEM_LEN_32,
  32. MSP_ELEM_LEN_32,
  33. MSP_ELEM_LEN_32,
  34. MSP_ELEM_LEN_32,
  35. MSP_DELAY_1,
  36. MSP_DELAY_1,
  37. MSP_RISING_EDGE,
  38. MSP_FALLING_EDGE,
  39. MSP_FSYNC_POL_ACT_LO,
  40. MSP_FSYNC_POL_ACT_LO,
  41. MSP_SWAP_NONE,
  42. MSP_SWAP_NONE,
  43. MSP_COMPRESS_MODE_LINEAR,
  44. MSP_EXPAND_MODE_LINEAR,
  45. MSP_FSYNC_IGNORE,
  46. 31,
  47. 15,
  48. 32,
  49. }, { /* PCM */
  50. MSP_DUAL_PHASE,
  51. MSP_DUAL_PHASE,
  52. MSP_PHASE2_START_MODE_FSYNC,
  53. MSP_PHASE2_START_MODE_FSYNC,
  54. MSP_BTF_MS_BIT_FIRST,
  55. MSP_BTF_MS_BIT_FIRST,
  56. MSP_FRAME_LEN_1,
  57. MSP_FRAME_LEN_1,
  58. MSP_FRAME_LEN_1,
  59. MSP_FRAME_LEN_1,
  60. MSP_ELEM_LEN_16,
  61. MSP_ELEM_LEN_16,
  62. MSP_ELEM_LEN_16,
  63. MSP_ELEM_LEN_16,
  64. MSP_DELAY_0,
  65. MSP_DELAY_0,
  66. MSP_RISING_EDGE,
  67. MSP_FALLING_EDGE,
  68. MSP_FSYNC_POL_ACT_HI,
  69. MSP_FSYNC_POL_ACT_HI,
  70. MSP_SWAP_NONE,
  71. MSP_SWAP_NONE,
  72. MSP_COMPRESS_MODE_LINEAR,
  73. MSP_EXPAND_MODE_LINEAR,
  74. MSP_FSYNC_IGNORE,
  75. 255,
  76. 0,
  77. 256,
  78. }, { /* Companded PCM */
  79. MSP_SINGLE_PHASE,
  80. MSP_SINGLE_PHASE,
  81. MSP_PHASE2_START_MODE_FSYNC,
  82. MSP_PHASE2_START_MODE_FSYNC,
  83. MSP_BTF_MS_BIT_FIRST,
  84. MSP_BTF_MS_BIT_FIRST,
  85. MSP_FRAME_LEN_1,
  86. MSP_FRAME_LEN_1,
  87. MSP_FRAME_LEN_1,
  88. MSP_FRAME_LEN_1,
  89. MSP_ELEM_LEN_8,
  90. MSP_ELEM_LEN_8,
  91. MSP_ELEM_LEN_8,
  92. MSP_ELEM_LEN_8,
  93. MSP_DELAY_0,
  94. MSP_DELAY_0,
  95. MSP_RISING_EDGE,
  96. MSP_RISING_EDGE,
  97. MSP_FSYNC_POL_ACT_HI,
  98. MSP_FSYNC_POL_ACT_HI,
  99. MSP_SWAP_NONE,
  100. MSP_SWAP_NONE,
  101. MSP_COMPRESS_MODE_LINEAR,
  102. MSP_EXPAND_MODE_LINEAR,
  103. MSP_FSYNC_IGNORE,
  104. 255,
  105. 0,
  106. 256,
  107. },
  108. };
  109. static void set_prot_desc_tx(struct ux500_msp *msp,
  110. struct msp_protdesc *protdesc,
  111. enum msp_data_size data_size)
  112. {
  113. u32 temp_reg = 0;
  114. temp_reg |= MSP_P2_ENABLE_BIT(protdesc->tx_phase_mode);
  115. temp_reg |= MSP_P2_START_MODE_BIT(protdesc->tx_phase2_start_mode);
  116. temp_reg |= MSP_P1_FRAME_LEN_BITS(protdesc->tx_frame_len_1);
  117. temp_reg |= MSP_P2_FRAME_LEN_BITS(protdesc->tx_frame_len_2);
  118. if (msp->def_elem_len) {
  119. temp_reg |= MSP_P1_ELEM_LEN_BITS(protdesc->tx_elem_len_1);
  120. temp_reg |= MSP_P2_ELEM_LEN_BITS(protdesc->tx_elem_len_2);
  121. } else {
  122. temp_reg |= MSP_P1_ELEM_LEN_BITS(data_size);
  123. temp_reg |= MSP_P2_ELEM_LEN_BITS(data_size);
  124. }
  125. temp_reg |= MSP_DATA_DELAY_BITS(protdesc->tx_data_delay);
  126. temp_reg |= MSP_SET_ENDIANNES_BIT(protdesc->tx_byte_order);
  127. temp_reg |= MSP_FSYNC_POL(protdesc->tx_fsync_pol);
  128. temp_reg |= MSP_DATA_WORD_SWAP(protdesc->tx_half_word_swap);
  129. temp_reg |= MSP_SET_COMPANDING_MODE(protdesc->compression_mode);
  130. temp_reg |= MSP_SET_FSYNC_IGNORE(protdesc->frame_sync_ignore);
  131. writel(temp_reg, msp->registers + MSP_TCF);
  132. }
  133. static void set_prot_desc_rx(struct ux500_msp *msp,
  134. struct msp_protdesc *protdesc,
  135. enum msp_data_size data_size)
  136. {
  137. u32 temp_reg = 0;
  138. temp_reg |= MSP_P2_ENABLE_BIT(protdesc->rx_phase_mode);
  139. temp_reg |= MSP_P2_START_MODE_BIT(protdesc->rx_phase2_start_mode);
  140. temp_reg |= MSP_P1_FRAME_LEN_BITS(protdesc->rx_frame_len_1);
  141. temp_reg |= MSP_P2_FRAME_LEN_BITS(protdesc->rx_frame_len_2);
  142. if (msp->def_elem_len) {
  143. temp_reg |= MSP_P1_ELEM_LEN_BITS(protdesc->rx_elem_len_1);
  144. temp_reg |= MSP_P2_ELEM_LEN_BITS(protdesc->rx_elem_len_2);
  145. } else {
  146. temp_reg |= MSP_P1_ELEM_LEN_BITS(data_size);
  147. temp_reg |= MSP_P2_ELEM_LEN_BITS(data_size);
  148. }
  149. temp_reg |= MSP_DATA_DELAY_BITS(protdesc->rx_data_delay);
  150. temp_reg |= MSP_SET_ENDIANNES_BIT(protdesc->rx_byte_order);
  151. temp_reg |= MSP_FSYNC_POL(protdesc->rx_fsync_pol);
  152. temp_reg |= MSP_DATA_WORD_SWAP(protdesc->rx_half_word_swap);
  153. temp_reg |= MSP_SET_COMPANDING_MODE(protdesc->expansion_mode);
  154. temp_reg |= MSP_SET_FSYNC_IGNORE(protdesc->frame_sync_ignore);
  155. writel(temp_reg, msp->registers + MSP_RCF);
  156. }
  157. static int configure_protocol(struct ux500_msp *msp,
  158. struct ux500_msp_config *config)
  159. {
  160. struct msp_protdesc *protdesc;
  161. enum msp_data_size data_size;
  162. u32 temp_reg = 0;
  163. data_size = config->data_size;
  164. msp->def_elem_len = config->def_elem_len;
  165. if (config->default_protdesc == 1) {
  166. if (config->protocol >= MSP_INVALID_PROTOCOL) {
  167. dev_err(msp->dev, "%s: ERROR: Invalid protocol!\n",
  168. __func__);
  169. return -EINVAL;
  170. }
  171. protdesc =
  172. (struct msp_protdesc *)&prot_descs[config->protocol];
  173. } else {
  174. protdesc = (struct msp_protdesc *)&config->protdesc;
  175. }
  176. if (data_size < MSP_DATA_BITS_DEFAULT || data_size > MSP_DATA_BITS_32) {
  177. dev_err(msp->dev,
  178. "%s: ERROR: Invalid data-size requested (data_size = %d)!\n",
  179. __func__, data_size);
  180. return -EINVAL;
  181. }
  182. if (config->direction & MSP_DIR_TX)
  183. set_prot_desc_tx(msp, protdesc, data_size);
  184. if (config->direction & MSP_DIR_RX)
  185. set_prot_desc_rx(msp, protdesc, data_size);
  186. /* The code below should not be separated. */
  187. temp_reg = readl(msp->registers + MSP_GCR) & ~TX_CLK_POL_RISING;
  188. temp_reg |= MSP_TX_CLKPOL_BIT(~protdesc->tx_clk_pol);
  189. writel(temp_reg, msp->registers + MSP_GCR);
  190. temp_reg = readl(msp->registers + MSP_GCR) & ~RX_CLK_POL_RISING;
  191. temp_reg |= MSP_RX_CLKPOL_BIT(protdesc->rx_clk_pol);
  192. writel(temp_reg, msp->registers + MSP_GCR);
  193. return 0;
  194. }
  195. static int setup_bitclk(struct ux500_msp *msp, struct ux500_msp_config *config)
  196. {
  197. u32 reg_val_GCR;
  198. u32 frame_per = 0;
  199. u32 sck_div = 0;
  200. u32 frame_width = 0;
  201. u32 temp_reg = 0;
  202. struct msp_protdesc *protdesc = NULL;
  203. reg_val_GCR = readl(msp->registers + MSP_GCR);
  204. writel(reg_val_GCR & ~SRG_ENABLE, msp->registers + MSP_GCR);
  205. if (config->default_protdesc)
  206. protdesc =
  207. (struct msp_protdesc *)&prot_descs[config->protocol];
  208. else
  209. protdesc = (struct msp_protdesc *)&config->protdesc;
  210. switch (config->protocol) {
  211. case MSP_PCM_PROTOCOL:
  212. case MSP_PCM_COMPAND_PROTOCOL:
  213. frame_width = protdesc->frame_width;
  214. sck_div = config->f_inputclk / (config->frame_freq *
  215. (protdesc->clocks_per_frame));
  216. frame_per = protdesc->frame_period;
  217. break;
  218. case MSP_I2S_PROTOCOL:
  219. frame_width = protdesc->frame_width;
  220. sck_div = config->f_inputclk / (config->frame_freq *
  221. (protdesc->clocks_per_frame));
  222. frame_per = protdesc->frame_period;
  223. break;
  224. default:
  225. dev_err(msp->dev, "%s: ERROR: Unknown protocol (%d)!\n",
  226. __func__,
  227. config->protocol);
  228. return -EINVAL;
  229. }
  230. temp_reg = (sck_div - 1) & SCK_DIV_MASK;
  231. temp_reg |= FRAME_WIDTH_BITS(frame_width);
  232. temp_reg |= FRAME_PERIOD_BITS(frame_per);
  233. writel(temp_reg, msp->registers + MSP_SRG);
  234. msp->f_bitclk = (config->f_inputclk)/(sck_div + 1);
  235. /* Enable bit-clock */
  236. udelay(100);
  237. reg_val_GCR = readl(msp->registers + MSP_GCR);
  238. writel(reg_val_GCR | SRG_ENABLE, msp->registers + MSP_GCR);
  239. udelay(100);
  240. return 0;
  241. }
  242. static int configure_multichannel(struct ux500_msp *msp,
  243. struct ux500_msp_config *config)
  244. {
  245. struct msp_protdesc *protdesc;
  246. struct msp_multichannel_config *mcfg;
  247. u32 reg_val_MCR;
  248. if (config->default_protdesc == 1) {
  249. if (config->protocol >= MSP_INVALID_PROTOCOL) {
  250. dev_err(msp->dev,
  251. "%s: ERROR: Invalid protocol (%d)!\n",
  252. __func__, config->protocol);
  253. return -EINVAL;
  254. }
  255. protdesc = (struct msp_protdesc *)
  256. &prot_descs[config->protocol];
  257. } else {
  258. protdesc = (struct msp_protdesc *)&config->protdesc;
  259. }
  260. mcfg = &config->multichannel_config;
  261. if (mcfg->tx_multichannel_enable) {
  262. if (protdesc->tx_phase_mode == MSP_SINGLE_PHASE) {
  263. reg_val_MCR = readl(msp->registers + MSP_MCR);
  264. writel(reg_val_MCR | (mcfg->tx_multichannel_enable ?
  265. 1 << TMCEN_BIT : 0),
  266. msp->registers + MSP_MCR);
  267. writel(mcfg->tx_channel_0_enable,
  268. msp->registers + MSP_TCE0);
  269. writel(mcfg->tx_channel_1_enable,
  270. msp->registers + MSP_TCE1);
  271. writel(mcfg->tx_channel_2_enable,
  272. msp->registers + MSP_TCE2);
  273. writel(mcfg->tx_channel_3_enable,
  274. msp->registers + MSP_TCE3);
  275. } else {
  276. dev_err(msp->dev,
  277. "%s: ERROR: Only single-phase supported (TX-mode: %d)!\n",
  278. __func__, protdesc->tx_phase_mode);
  279. return -EINVAL;
  280. }
  281. }
  282. if (mcfg->rx_multichannel_enable) {
  283. if (protdesc->rx_phase_mode == MSP_SINGLE_PHASE) {
  284. reg_val_MCR = readl(msp->registers + MSP_MCR);
  285. writel(reg_val_MCR | (mcfg->rx_multichannel_enable ?
  286. 1 << RMCEN_BIT : 0),
  287. msp->registers + MSP_MCR);
  288. writel(mcfg->rx_channel_0_enable,
  289. msp->registers + MSP_RCE0);
  290. writel(mcfg->rx_channel_1_enable,
  291. msp->registers + MSP_RCE1);
  292. writel(mcfg->rx_channel_2_enable,
  293. msp->registers + MSP_RCE2);
  294. writel(mcfg->rx_channel_3_enable,
  295. msp->registers + MSP_RCE3);
  296. } else {
  297. dev_err(msp->dev,
  298. "%s: ERROR: Only single-phase supported (RX-mode: %d)!\n",
  299. __func__, protdesc->rx_phase_mode);
  300. return -EINVAL;
  301. }
  302. if (mcfg->rx_comparison_enable_mode) {
  303. reg_val_MCR = readl(msp->registers + MSP_MCR);
  304. writel(reg_val_MCR |
  305. (mcfg->rx_comparison_enable_mode << RCMPM_BIT),
  306. msp->registers + MSP_MCR);
  307. writel(mcfg->comparison_mask,
  308. msp->registers + MSP_RCM);
  309. writel(mcfg->comparison_value,
  310. msp->registers + MSP_RCV);
  311. }
  312. }
  313. return 0;
  314. }
  315. static int enable_msp(struct ux500_msp *msp, struct ux500_msp_config *config)
  316. {
  317. int status = 0;
  318. u32 reg_val_DMACR, reg_val_GCR;
  319. /* Configure msp with protocol dependent settings */
  320. configure_protocol(msp, config);
  321. setup_bitclk(msp, config);
  322. if (config->multichannel_configured == 1) {
  323. status = configure_multichannel(msp, config);
  324. if (status)
  325. dev_warn(msp->dev,
  326. "%s: WARN: configure_multichannel failed (%d)!\n",
  327. __func__, status);
  328. }
  329. reg_val_DMACR = readl(msp->registers + MSP_DMACR);
  330. if (config->direction & MSP_DIR_RX)
  331. reg_val_DMACR |= RX_DMA_ENABLE;
  332. if (config->direction & MSP_DIR_TX)
  333. reg_val_DMACR |= TX_DMA_ENABLE;
  334. writel(reg_val_DMACR, msp->registers + MSP_DMACR);
  335. writel(config->iodelay, msp->registers + MSP_IODLY);
  336. /* Enable frame generation logic */
  337. reg_val_GCR = readl(msp->registers + MSP_GCR);
  338. writel(reg_val_GCR | FRAME_GEN_ENABLE, msp->registers + MSP_GCR);
  339. return status;
  340. }
  341. static void flush_fifo_rx(struct ux500_msp *msp)
  342. {
  343. u32 reg_val_GCR, reg_val_FLR;
  344. u32 limit = 32;
  345. reg_val_GCR = readl(msp->registers + MSP_GCR);
  346. writel(reg_val_GCR | RX_ENABLE, msp->registers + MSP_GCR);
  347. reg_val_FLR = readl(msp->registers + MSP_FLR);
  348. while (!(reg_val_FLR & RX_FIFO_EMPTY) && limit--) {
  349. readl(msp->registers + MSP_DR);
  350. reg_val_FLR = readl(msp->registers + MSP_FLR);
  351. }
  352. writel(reg_val_GCR, msp->registers + MSP_GCR);
  353. }
  354. static void flush_fifo_tx(struct ux500_msp *msp)
  355. {
  356. u32 reg_val_GCR, reg_val_FLR;
  357. u32 limit = 32;
  358. reg_val_GCR = readl(msp->registers + MSP_GCR);
  359. writel(reg_val_GCR | TX_ENABLE, msp->registers + MSP_GCR);
  360. writel(MSP_ITCR_ITEN | MSP_ITCR_TESTFIFO, msp->registers + MSP_ITCR);
  361. reg_val_FLR = readl(msp->registers + MSP_FLR);
  362. while (!(reg_val_FLR & TX_FIFO_EMPTY) && limit--) {
  363. readl(msp->registers + MSP_TSTDR);
  364. reg_val_FLR = readl(msp->registers + MSP_FLR);
  365. }
  366. writel(0x0, msp->registers + MSP_ITCR);
  367. writel(reg_val_GCR, msp->registers + MSP_GCR);
  368. }
  369. int ux500_msp_i2s_open(struct ux500_msp *msp,
  370. struct ux500_msp_config *config)
  371. {
  372. u32 old_reg, new_reg, mask;
  373. int res;
  374. unsigned int tx_sel, rx_sel, tx_busy, rx_busy;
  375. if (in_interrupt()) {
  376. dev_err(msp->dev,
  377. "%s: ERROR: Open called in interrupt context!\n",
  378. __func__);
  379. return -1;
  380. }
  381. tx_sel = (config->direction & MSP_DIR_TX) > 0;
  382. rx_sel = (config->direction & MSP_DIR_RX) > 0;
  383. if (!tx_sel && !rx_sel) {
  384. dev_err(msp->dev, "%s: Error: No direction selected!\n",
  385. __func__);
  386. return -EINVAL;
  387. }
  388. tx_busy = (msp->dir_busy & MSP_DIR_TX) > 0;
  389. rx_busy = (msp->dir_busy & MSP_DIR_RX) > 0;
  390. if (tx_busy && tx_sel) {
  391. dev_err(msp->dev, "%s: Error: TX is in use!\n", __func__);
  392. return -EBUSY;
  393. }
  394. if (rx_busy && rx_sel) {
  395. dev_err(msp->dev, "%s: Error: RX is in use!\n", __func__);
  396. return -EBUSY;
  397. }
  398. msp->dir_busy |= (tx_sel ? MSP_DIR_TX : 0) | (rx_sel ? MSP_DIR_RX : 0);
  399. /* First do the global config register */
  400. mask = RX_CLK_SEL_MASK | TX_CLK_SEL_MASK | RX_FSYNC_MASK |
  401. TX_FSYNC_MASK | RX_SYNC_SEL_MASK | TX_SYNC_SEL_MASK |
  402. RX_FIFO_ENABLE_MASK | TX_FIFO_ENABLE_MASK | SRG_CLK_SEL_MASK |
  403. LOOPBACK_MASK | TX_EXTRA_DELAY_MASK;
  404. new_reg = (config->tx_clk_sel | config->rx_clk_sel |
  405. config->rx_fsync_pol | config->tx_fsync_pol |
  406. config->rx_fsync_sel | config->tx_fsync_sel |
  407. config->rx_fifo_config | config->tx_fifo_config |
  408. config->srg_clk_sel | config->loopback_enable |
  409. config->tx_data_enable);
  410. old_reg = readl(msp->registers + MSP_GCR);
  411. old_reg &= ~mask;
  412. new_reg |= old_reg;
  413. writel(new_reg, msp->registers + MSP_GCR);
  414. res = enable_msp(msp, config);
  415. if (res < 0) {
  416. dev_err(msp->dev, "%s: ERROR: enable_msp failed (%d)!\n",
  417. __func__, res);
  418. return -EBUSY;
  419. }
  420. if (config->loopback_enable & 0x80)
  421. msp->loopback_enable = 1;
  422. /* Flush FIFOs */
  423. flush_fifo_tx(msp);
  424. flush_fifo_rx(msp);
  425. msp->msp_state = MSP_STATE_CONFIGURED;
  426. return 0;
  427. }
  428. static void disable_msp_rx(struct ux500_msp *msp)
  429. {
  430. u32 reg_val_GCR, reg_val_DMACR, reg_val_IMSC;
  431. reg_val_GCR = readl(msp->registers + MSP_GCR);
  432. writel(reg_val_GCR & ~RX_ENABLE, msp->registers + MSP_GCR);
  433. reg_val_DMACR = readl(msp->registers + MSP_DMACR);
  434. writel(reg_val_DMACR & ~RX_DMA_ENABLE, msp->registers + MSP_DMACR);
  435. reg_val_IMSC = readl(msp->registers + MSP_IMSC);
  436. writel(reg_val_IMSC &
  437. ~(RX_SERVICE_INT | RX_OVERRUN_ERROR_INT),
  438. msp->registers + MSP_IMSC);
  439. msp->dir_busy &= ~MSP_DIR_RX;
  440. }
  441. static void disable_msp_tx(struct ux500_msp *msp)
  442. {
  443. u32 reg_val_GCR, reg_val_DMACR, reg_val_IMSC;
  444. reg_val_GCR = readl(msp->registers + MSP_GCR);
  445. writel(reg_val_GCR & ~TX_ENABLE, msp->registers + MSP_GCR);
  446. reg_val_DMACR = readl(msp->registers + MSP_DMACR);
  447. writel(reg_val_DMACR & ~TX_DMA_ENABLE, msp->registers + MSP_DMACR);
  448. reg_val_IMSC = readl(msp->registers + MSP_IMSC);
  449. writel(reg_val_IMSC &
  450. ~(TX_SERVICE_INT | TX_UNDERRUN_ERR_INT),
  451. msp->registers + MSP_IMSC);
  452. msp->dir_busy &= ~MSP_DIR_TX;
  453. }
  454. static int disable_msp(struct ux500_msp *msp, unsigned int dir)
  455. {
  456. u32 reg_val_GCR;
  457. unsigned int disable_tx, disable_rx;
  458. reg_val_GCR = readl(msp->registers + MSP_GCR);
  459. disable_tx = dir & MSP_DIR_TX;
  460. disable_rx = dir & MSP_DIR_TX;
  461. if (disable_tx && disable_rx) {
  462. reg_val_GCR = readl(msp->registers + MSP_GCR);
  463. writel(reg_val_GCR | LOOPBACK_MASK,
  464. msp->registers + MSP_GCR);
  465. /* Flush TX-FIFO */
  466. flush_fifo_tx(msp);
  467. /* Disable TX-channel */
  468. writel((readl(msp->registers + MSP_GCR) &
  469. (~TX_ENABLE)), msp->registers + MSP_GCR);
  470. /* Flush RX-FIFO */
  471. flush_fifo_rx(msp);
  472. /* Disable Loopback and Receive channel */
  473. writel((readl(msp->registers + MSP_GCR) &
  474. (~(RX_ENABLE | LOOPBACK_MASK))),
  475. msp->registers + MSP_GCR);
  476. disable_msp_tx(msp);
  477. disable_msp_rx(msp);
  478. } else if (disable_tx)
  479. disable_msp_tx(msp);
  480. else if (disable_rx)
  481. disable_msp_rx(msp);
  482. return 0;
  483. }
  484. int ux500_msp_i2s_trigger(struct ux500_msp *msp, int cmd, int direction)
  485. {
  486. u32 reg_val_GCR, enable_bit;
  487. if (msp->msp_state == MSP_STATE_IDLE) {
  488. dev_err(msp->dev, "%s: ERROR: MSP is not configured!\n",
  489. __func__);
  490. return -EINVAL;
  491. }
  492. switch (cmd) {
  493. case SNDRV_PCM_TRIGGER_START:
  494. case SNDRV_PCM_TRIGGER_RESUME:
  495. case SNDRV_PCM_TRIGGER_PAUSE_RELEASE:
  496. if (direction == SNDRV_PCM_STREAM_PLAYBACK)
  497. enable_bit = TX_ENABLE;
  498. else
  499. enable_bit = RX_ENABLE;
  500. reg_val_GCR = readl(msp->registers + MSP_GCR);
  501. writel(reg_val_GCR | enable_bit, msp->registers + MSP_GCR);
  502. break;
  503. case SNDRV_PCM_TRIGGER_STOP:
  504. case SNDRV_PCM_TRIGGER_SUSPEND:
  505. case SNDRV_PCM_TRIGGER_PAUSE_PUSH:
  506. if (direction == SNDRV_PCM_STREAM_PLAYBACK)
  507. disable_msp_tx(msp);
  508. else
  509. disable_msp_rx(msp);
  510. break;
  511. default:
  512. return -EINVAL;
  513. }
  514. return 0;
  515. }
  516. int ux500_msp_i2s_close(struct ux500_msp *msp, unsigned int dir)
  517. {
  518. int status = 0;
  519. dev_dbg(msp->dev, "%s: Enter (dir = 0x%01x).\n", __func__, dir);
  520. status = disable_msp(msp, dir);
  521. if (msp->dir_busy == 0) {
  522. /* disable sample rate and frame generators */
  523. msp->msp_state = MSP_STATE_IDLE;
  524. writel((readl(msp->registers + MSP_GCR) &
  525. (~(FRAME_GEN_ENABLE | SRG_ENABLE))),
  526. msp->registers + MSP_GCR);
  527. writel(0, msp->registers + MSP_GCR);
  528. writel(0, msp->registers + MSP_TCF);
  529. writel(0, msp->registers + MSP_RCF);
  530. writel(0, msp->registers + MSP_DMACR);
  531. writel(0, msp->registers + MSP_SRG);
  532. writel(0, msp->registers + MSP_MCR);
  533. writel(0, msp->registers + MSP_RCM);
  534. writel(0, msp->registers + MSP_RCV);
  535. writel(0, msp->registers + MSP_TCE0);
  536. writel(0, msp->registers + MSP_TCE1);
  537. writel(0, msp->registers + MSP_TCE2);
  538. writel(0, msp->registers + MSP_TCE3);
  539. writel(0, msp->registers + MSP_RCE0);
  540. writel(0, msp->registers + MSP_RCE1);
  541. writel(0, msp->registers + MSP_RCE2);
  542. writel(0, msp->registers + MSP_RCE3);
  543. }
  544. return status;
  545. }
  546. int ux500_msp_i2s_init_msp(struct platform_device *pdev,
  547. struct ux500_msp **msp_p)
  548. {
  549. struct resource *res = NULL;
  550. struct ux500_msp *msp;
  551. *msp_p = devm_kzalloc(&pdev->dev, sizeof(struct ux500_msp), GFP_KERNEL);
  552. msp = *msp_p;
  553. if (!msp)
  554. return -ENOMEM;
  555. msp->dev = &pdev->dev;
  556. res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
  557. if (res == NULL) {
  558. dev_err(&pdev->dev, "%s: ERROR: Unable to get resource!\n",
  559. __func__);
  560. return -ENOMEM;
  561. }
  562. msp->tx_rx_addr = res->start + MSP_DR;
  563. msp->registers = devm_ioremap(&pdev->dev, res->start,
  564. resource_size(res));
  565. if (msp->registers == NULL) {
  566. dev_err(&pdev->dev, "%s: ERROR: ioremap failed!\n", __func__);
  567. return -ENOMEM;
  568. }
  569. msp->msp_state = MSP_STATE_IDLE;
  570. msp->loopback_enable = 0;
  571. return 0;
  572. }
  573. void ux500_msp_i2s_cleanup_msp(struct platform_device *pdev,
  574. struct ux500_msp *msp)
  575. {
  576. dev_dbg(msp->dev, "%s: Enter (id = %d).\n", __func__, msp->id);
  577. }
  578. MODULE_LICENSE("GPL v2");