audio_acts_utils.c 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521
  1. /*
  2. * Copyright (c) 2020 Actions Semiconductor Co., Ltd
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief Common utils for in/out audio drivers
  9. */
  10. #include <kernel.h>
  11. #include <device.h>
  12. #include <string.h>
  13. #include <errno.h>
  14. #include <soc.h>
  15. #include "audio_acts_utils.h"
  16. #include "phy_audio_common.h"
  17. #include <logging/log.h>
  18. LOG_MODULE_REGISTER(autils, CONFIG_LOG_DEFAULT_LEVEL);
  19. /*
  20. * @brief AUDIO PLL clock selection
  21. */
  22. /* CMU AUDIOPLL0 CTL bits */
  23. #define CMU_AUDIOPLL0_CTL_PMD (8)
  24. #define CMU_AUDIOPLL0_CTL_EN (4)
  25. #define CMU_AUDIOPLL0_CTL_APS0_SHIFT (0)
  26. #define CMU_AUDIOPLL0_CTL_APS0(x) ((x) << CMU_AUDIOPLL0_CTL_APS0_SHIFT)
  27. #define CMU_AUDIOPLL0_CTL_APS0_MASK CMU_AUDIOPLL0_CTL_APS0(0xF)
  28. /* CMU AUDIOPLL1 CTL bits */
  29. #define CMU_AUDIOPLL1_CTL_PMD (8)
  30. #define CMU_AUDIOPLL1_CTL_EN (4)
  31. #define CMU_AUDIOPLL1_CTL_APS1_SHIFT (0)
  32. #define CMU_AUDIOPLL1_CTL_APS1(x) ((x) << CMU_AUDIOPLL1_CTL_APS1_SHIFT)
  33. #define CMU_AUDIOPLL1_CTL_APS1_MASK CMU_AUDIOPLL1_CTL_APS1(0xF)
  34. /* 48ksr serials */
  35. #define PLL_65536 (65536000)
  36. #define PLL_49152 (49152000)
  37. #define PLL_24576 (24576000)
  38. #define PLL_16384 (16384000)
  39. #define PLL_12288 (12288000)
  40. #define PLL_8192 (8192000)
  41. #define PLL_6144 (6144000)
  42. #define PLL_4096 (4096000)
  43. #define PLL_3072 (3072000)
  44. #define PLL_2048 (2048000)
  45. /* 44.1ksr serials */
  46. #define PLL_602112 (60211200)
  47. #define PLL_451584 (45158400)
  48. #define PLL_225792 (22579200)
  49. #define PLL_112896 (11289600)
  50. #define PLL_56448 (5644800)
  51. #define PLL_28224 (2822400)
  52. /*
  53. * struct audio_pll_t
  54. * @brief The structure includes the pll clock and its setting parameters
  55. */
  56. typedef struct {
  57. uint32_t pll_clk; /* audio pll clock */
  58. uint8_t pre_div; /* clock pre-divisor */
  59. uint8_t clk_div; /* clock divisor */
  60. } audio_pll_t;
  61. static const audio_pll_t audio_pll_selections[] = {
  62. {PLL_49152, 0, 0}, /* 48ksr series */
  63. {PLL_24576, 0, 1}, /* 48ksr series */
  64. {PLL_16384, 0, 2}, /* 48ksr series */
  65. {PLL_12288, 0, 3}, /* 48ksr series */
  66. {PLL_8192, 0, 4}, /* 48ksr series */
  67. {PLL_6144, 0, 5}, /* 48ksr series */
  68. {PLL_4096, 0, 6}, /* 48ksr series */
  69. {PLL_3072, 1, 5}, /* 48ksr series */
  70. {PLL_2048, 1, 6}, /* 48ksr series */
  71. {PLL_451584, 0, 0}, /* 44.1ksr series */
  72. {PLL_225792, 0, 1}, /* 44.1ksr series */
  73. {PLL_112896, 0, 3}, /* 44.1ksr series */
  74. {PLL_56448, 0, 5}, /* 44.1ksr series */
  75. {PLL_28224, 1, 5} /* 44.1ksr series */
  76. };
  77. /*
  78. * struct audio_i2s_pll_t
  79. * @brief The structure includes the pll clock and its setting parameters version2 for i2s
  80. */
  81. typedef struct {
  82. uint32_t pll_clk; /* audio pll clock */
  83. uint8_t clk_div; /* divison */
  84. } audio_i2s_pll_t;
  85. static const audio_i2s_pll_t audio_i2s_pll_selections[] = {
  86. {PLL_65536, 0}, /* 48ksr series */
  87. {PLL_49152, 1}, /* 48ksr series */
  88. {PLL_24576, 3}, /* 48ksr series */
  89. {PLL_16384, 4}, /* 48ksr series */
  90. {PLL_12288, 5}, /* 48ksr series */
  91. {PLL_8192, 6}, /* 48ksr series */
  92. {PLL_6144, 7}, /* 48ksr series */
  93. {PLL_4096, 8}, /* 48ksr series */
  94. {PLL_3072, 9}, /* 48ksr series */
  95. {PLL_2048, 10}, /* 48ksr series */
  96. {PLL_602112, 0}, /* 44.1ksr series */
  97. {PLL_451584, 1}, /* 44.1ksr series */
  98. {PLL_225792, 3}, /* 44.1ksr series */
  99. {PLL_112896, 5}, /* 44.1ksr series */
  100. {PLL_56448, 7}, /* 44.1ksr series */
  101. {PLL_28224, 9} /* 44.1ksr series */
  102. };
  103. #ifdef CONFIG_AUDIO_DEBUG_TRACE
  104. /* audio debug trace global object */
  105. struct audio_debug_trace_t audio_debug_trace;
  106. #endif
  107. static uint8_t audiopll0_refcount;
  108. static uint8_t audiopll1_refcount;
  109. /* @brief Translate the sample rate from KHz to Hz */
  110. uint32_t audio_sr_khz_to_hz(audio_sr_sel_e sr_khz)
  111. {
  112. uint32_t ret_sample_hz;
  113. if (!(sr_khz % SAMPLE_RATE_11KHZ)) {
  114. /* 44.1KHz serials */
  115. ret_sample_hz = (sr_khz / SAMPLE_RATE_11KHZ) * 11025;
  116. } else {
  117. /* 48KHz serials */
  118. ret_sample_hz = sr_khz * 1000;
  119. }
  120. return ret_sample_hz;
  121. }
  122. /* @brief Translate the sample rate from Hz to KHz */
  123. audio_sr_sel_e audio_sr_hz_to_Khz(uint32_t sr_hz)
  124. {
  125. uint32_t sr_khz;
  126. if (!(sr_hz % 11025)) {
  127. sr_khz = (sr_hz / 11025) * SAMPLE_RATE_11KHZ;
  128. } else {
  129. sr_khz = sr_hz / 1000;
  130. }
  131. return (audio_sr_sel_e)sr_khz;
  132. }
  133. /* @brief Get the audio pll setting by specified sample rate and mclk*/
  134. int audio_get_pll_setting(audio_sr_sel_e sr_khz, a_mclk_type_e mclk,
  135. uint8_t *pre_div, uint8_t *clk_div, uint8_t *series)
  136. {
  137. uint32_t sr_hz;
  138. int i;
  139. /* FPGA audio_pll main clock is 24Mhz, but the formal chipset is 98.304Mhz */
  140. #ifdef CONFIG_SOC_SERIES_LARK_FPGA
  141. sr_khz *= 4;
  142. #endif
  143. sr_hz = audio_sr_khz_to_hz(sr_khz);
  144. /* calculate the constant clock */
  145. sr_hz *= mclk;
  146. for (i = 0; i < ARRAY_SIZE(audio_pll_selections); i++) {
  147. if (sr_hz == audio_pll_selections[i].pll_clk) {
  148. *pre_div = audio_pll_selections[i].pre_div;
  149. *clk_div = audio_pll_selections[i].clk_div;
  150. if (i > 8)
  151. *series = AUDIOPLL_44KSR;
  152. else
  153. *series = AUDIOPLL_48KSR;
  154. break;
  155. }
  156. }
  157. /* Can not find the corresponding PLL setting */
  158. if (i == ARRAY_SIZE(audio_pll_selections)) {
  159. LOG_ERR("Failed to find audio pll setting sr:%d mclk:%d", sr_khz, mclk);
  160. *pre_div = 0xFF;
  161. *pre_div = 0xFF;
  162. *series = 0xFF;
  163. return -ENOEXEC;
  164. }
  165. LOG_DBG("sr_khz %d sr_hz %d, pre_div %d, clk_div %d", sr_khz, sr_hz, *pre_div, *clk_div);
  166. return 0;
  167. }
  168. /* @brief Get the audio pll setting by specified sample rate and mclk for i2s*/
  169. int audio_get_pll_setting_i2s(uint16_t sr_khz, a_mclk_type_e mclk,
  170. uint8_t *div, uint8_t *series)
  171. {
  172. uint32_t sr_hz;
  173. int i;
  174. /* FPGA audio_pll main clock is 24Mhz, but the formal chipset is 98.304Mhz */
  175. #ifdef CONFIG_SOC_SERIES_LARK_FPGA
  176. sr_khz *= 4;
  177. #endif
  178. /* in case of MCLK from internal in slave mode */
  179. if (384 == sr_khz) { /* MCLK = 192fs x 2 */
  180. *series = AUDIOPLL_48KSR;
  181. *div = 0;
  182. } else if (352 == sr_khz) { /* MCLK = 176fs x 2 */
  183. *series = AUDIOPLL_44KSR;
  184. *div = 0;
  185. } else {
  186. sr_hz = audio_sr_khz_to_hz(sr_khz);
  187. /* calculate the constant clock */
  188. sr_hz *= mclk;
  189. for (i = 0; i < ARRAY_SIZE(audio_i2s_pll_selections); i++) {
  190. if (sr_hz == audio_i2s_pll_selections[i].pll_clk) {
  191. *div = audio_i2s_pll_selections[i].clk_div;
  192. if (i > 9)
  193. *series = AUDIOPLL_44KSR;
  194. else
  195. *series = AUDIOPLL_48KSR;
  196. break;
  197. }
  198. }
  199. /* Can not find the corresponding PLL setting */
  200. if (i == ARRAY_SIZE(audio_i2s_pll_selections)) {
  201. LOG_ERR("Failed to find audio pll setting sr:%d mclk:%d", sr_khz, mclk);
  202. *div = 0xFF;
  203. *series = 0xFF;
  204. return -ENOEXEC;
  205. }
  206. }
  207. LOG_DBG("sr_khz %d sr_hz %d, div %d", sr_khz, sr_hz, *div);
  208. return 0;
  209. }
  210. /* @brief Get the audio pll usage info by the pll index */
  211. static int audio_pll_get_usage(a_pll_type_e index, uint8_t *series)
  212. {
  213. /* AUDIO_PLL0 */
  214. if (AUDIOPLL_TYPE_0 == index) {
  215. /* check AUDIO_PLL0 enable or not */
  216. if ((sys_read32(AUDIO_PLL0_CTL) & (1 << CMU_AUDIOPLL0_CTL_EN)) == 0) {
  217. /* AUDIO_PLL0 disable */
  218. *series = 0xFF;
  219. } else if((sys_read32(AUDIO_PLL0_CTL) & CMU_AUDIOPLL0_CTL_APS0_MASK) >= 8) {
  220. /* AUDIO_PLL0 is 48k series */
  221. *series = AUDIOPLL_48KSR;
  222. } else {
  223. /* AUDIO_PLL0 is 44.1k series */
  224. *series = AUDIOPLL_44KSR;
  225. }
  226. } else if (AUDIOPLL_TYPE_1 == index) {
  227. /* AUDIO_PLL1 */
  228. if((sys_read32(AUDIO_PLL1_CTL) & (1 << CMU_AUDIOPLL1_CTL_EN)) == 0) {
  229. /* AUDIO_PLL1 disable */
  230. *series = 0xFF;
  231. } else if ((sys_read32(AUDIO_PLL1_CTL) & CMU_AUDIOPLL1_CTL_APS1_MASK) >= 8) {
  232. /* AUDIO_PLL1 is 48k series */
  233. *series = AUDIOPLL_48KSR;
  234. } else {
  235. /* AUDIO_PLL1 is 44.1k series */
  236. *series = AUDIOPLL_44KSR;
  237. }
  238. } else {
  239. LOG_ERR("Invalid AUDIO_PLL type %d", index);
  240. return -EINVAL;
  241. }
  242. LOG_DBG("use AUDIO_PLL@%d => series:0x%x", index, *series);
  243. return 0;
  244. }
  245. /* @brief Get the audio pll aps by specified pll index */
  246. int audio_pll_get_aps(a_pll_type_e index)
  247. {
  248. uint32_t reg = -1;
  249. if (AUDIOPLL_TYPE_0 == index) {
  250. if ((sys_read32(AUDIO_PLL0_CTL) & (1 << CMU_AUDIOPLL0_CTL_EN)) == 0) {
  251. LOG_ERR("AUDIO_PLL0 is not enable yet");
  252. return -EPERM;
  253. }
  254. reg = sys_read32(AUDIO_PLL0_CTL) & CMU_AUDIOPLL0_CTL_APS0_MASK;
  255. if (reg >= 8) {
  256. /* 48KHz sample rate seires */
  257. reg -= 8;
  258. }
  259. } else if (AUDIOPLL_TYPE_1 == index) {
  260. if ((sys_read32(AUDIO_PLL1_CTL) & (1 << CMU_AUDIOPLL1_CTL_EN)) == 0) {
  261. LOG_ERR("AUDIO_PLL1 is not enable yet");
  262. return -EPERM;
  263. }
  264. reg = sys_read32(AUDIO_PLL1_CTL) & CMU_AUDIOPLL1_CTL_APS1_MASK;
  265. if (reg >= 8) {
  266. /* 48KHz sample rate seires */
  267. reg -= 8;
  268. }
  269. } else {
  270. LOG_ERR("Invalid AUDIO_PLL type %d", index);
  271. return -EINVAL;
  272. }
  273. return reg;
  274. }
  275. /* @brief Set the audio pll aps by specified pll index */
  276. int audio_pll_set_aps(a_pll_type_e index, audio_aps_level_e level)
  277. {
  278. uint32_t reg;
  279. if (level > APS_LEVEL_8) {
  280. LOG_ERR("Invalid APS level setting %d", level);
  281. return -EINVAL;
  282. }
  283. if (AUDIOPLL_TYPE_0 == index) {
  284. if ((sys_read32(AUDIO_PLL0_CTL) & (1 << CMU_AUDIOPLL0_CTL_EN)) == 0) {
  285. LOG_INF("AUDIO_PLL0 is not enable");
  286. return -EPERM;
  287. }
  288. reg = sys_read32(AUDIO_PLL0_CTL);
  289. if ((reg & CMU_AUDIOPLL0_CTL_APS0_MASK) >= 8) /* 48KHz sample rate seires */
  290. level += 8;
  291. reg &= ~CMU_AUDIOPLL0_CTL_APS0_MASK;
  292. sys_write32(reg | CMU_AUDIOPLL0_CTL_APS0(level), AUDIO_PLL0_CTL);
  293. } else if (AUDIOPLL_TYPE_1 == index) {
  294. if ((sys_read32(AUDIO_PLL1_CTL) & (1 << CMU_AUDIOPLL1_CTL_EN)) == 0) {
  295. LOG_ERR("AUDIO_PLL1 is not enable yet");
  296. return -EPERM;
  297. }
  298. reg = sys_read32(AUDIO_PLL1_CTL);
  299. if ((reg & CMU_AUDIOPLL1_CTL_APS1_MASK) >= 8) /* 48KHz sample rate seires */
  300. level += 8;
  301. reg &= ~CMU_AUDIOPLL1_CTL_APS1_MASK;
  302. sys_write32(reg | CMU_AUDIOPLL1_CTL_APS1(level), AUDIO_PLL1_CTL);
  303. } else {
  304. LOG_ERR("Invalid AUDIO_PLL type %d", index);
  305. return -EINVAL;
  306. }
  307. return 0;
  308. }
  309. /* @brief Get the pll clock in HZ for the sample rate translation */
  310. int audio_get_pll_sample_rate(a_mclk_type_e mclk, uint8_t pre_div, uint8_t clk_div, a_pll_type_e index)
  311. {
  312. uint8_t series, start, end;
  313. int ret, i;
  314. ret = audio_pll_get_usage(index, &series);
  315. if (ret)
  316. return ret;
  317. if (AUDIOPLL_48KSR == series) {
  318. start = 0;
  319. end = 9;
  320. } else if (AUDIOPLL_44KSR == series) {
  321. start = 9;
  322. end = ARRAY_SIZE(audio_pll_selections);
  323. } else {
  324. LOG_ERR("Error series %d", series);
  325. return -EPERM;
  326. }
  327. for (i = start; i < end; i++) {
  328. if ((audio_pll_selections[i].pre_div == pre_div)
  329. && (audio_pll_selections[i].clk_div) == clk_div) {
  330. ret = audio_pll_selections[i].pll_clk / mclk;
  331. ret = (int)audio_sr_hz_to_Khz(ret);
  332. break;
  333. }
  334. }
  335. if (i == end) {
  336. LOG_ERR("Failed to translate sr pre_div:%d clk_div%d pll_index:%d",
  337. pre_div, clk_div, index);
  338. ret = -EFAULT;
  339. }
  340. return ret;
  341. }
  342. /* @brief Get the pll clock in HZ for the sample rate translation for i2s */
  343. int audio_get_pll_sample_rate_i2s(a_mclk_type_e mclk, uint8_t clk_div, a_pll_type_e index)
  344. {
  345. uint8_t series, start, end;
  346. int ret, i;
  347. ret = audio_pll_get_usage(index, &series);
  348. if (ret)
  349. return ret;
  350. if (AUDIOPLL_48KSR == series) {
  351. start = 0;
  352. end = 10;
  353. } else if (AUDIOPLL_44KSR == series) {
  354. start = 10;
  355. end = ARRAY_SIZE(audio_i2s_pll_selections);
  356. } else {
  357. LOG_ERR("Error series %d", series);
  358. return -EPERM;
  359. }
  360. for (i = start; i < end; i++) {
  361. if (audio_i2s_pll_selections[i].clk_div == clk_div) {
  362. ret = audio_i2s_pll_selections[i].pll_clk / mclk;
  363. /* 235.2kfs is not divisible by 11025 */
  364. if (235200 == ret)
  365. ret = 235;
  366. else
  367. ret = (int)audio_sr_hz_to_Khz(ret);
  368. break;
  369. }
  370. }
  371. if (i == end) {
  372. LOG_ERR("Failed to translate sr clk_div%d pll_index:%d",
  373. clk_div, index);
  374. ret = -EFAULT;
  375. }
  376. return ret;
  377. }
  378. /* @brief audio pll set */
  379. void audio_pll_set(a_pll_type_e index, a_pll_series_e series)
  380. {
  381. uint32_t reg;
  382. if (AUDIOPLL_TYPE_0 == index) {
  383. /* Enable AUDIO_PLL0 */
  384. reg = sys_read32(AUDIO_PLL0_CTL) & (~CMU_AUDIOPLL0_CTL_APS0_MASK);
  385. reg |= (1 << CMU_AUDIOPLL0_CTL_EN);
  386. if (AUDIOPLL_44KSR == series)
  387. reg |= (0x04 << CMU_AUDIOPLL0_CTL_APS0_SHIFT);
  388. else
  389. reg |= (0x0c << CMU_AUDIOPLL0_CTL_APS0_SHIFT);
  390. sys_write32(reg, AUDIO_PLL0_CTL);
  391. ++audiopll0_refcount;
  392. LOG_DBG("AUDIO_PLL0_CTL - 0x%x", sys_read32(AUDIO_PLL0_CTL));
  393. } else if (AUDIOPLL_TYPE_1 == index) {
  394. /* Enable AUDIO_PLL1 */
  395. reg = sys_read32(AUDIO_PLL1_CTL) & (~CMU_AUDIOPLL1_CTL_APS1_MASK);
  396. reg |= (1 << CMU_AUDIOPLL1_CTL_EN);
  397. if (AUDIOPLL_44KSR == series)
  398. reg |= (0x04 << CMU_AUDIOPLL1_CTL_APS1_SHIFT);
  399. else
  400. reg |= (0x0c << CMU_AUDIOPLL1_CTL_APS1_SHIFT);
  401. sys_write32(reg, AUDIO_PLL1_CTL);
  402. ++audiopll1_refcount;
  403. LOG_DBG("AUDIO_PLL1_CTL - 0x%x", sys_read32(AUDIO_PLL1_CTL));
  404. }
  405. }
  406. /* @brief unset audio pll */
  407. void audio_pll_unset(a_pll_type_e index)
  408. {
  409. if (AUDIOPLL_TYPE_0 == index) {
  410. --audiopll0_refcount;
  411. if (!audiopll0_refcount)
  412. sys_write32(0, AUDIO_PLL0_CTL);
  413. } else if (AUDIOPLL_TYPE_1 == index) {
  414. --audiopll1_refcount;
  415. if (!audiopll1_refcount)
  416. sys_write32(0, AUDIO_PLL1_CTL);
  417. }
  418. }
  419. /* @brief Check and config the audio pll */
  420. int audio_pll_check_config(a_pll_series_e series, uint8_t *index)
  421. {
  422. int ret;
  423. uint8_t get_series, pll_index = 0xFF;
  424. ret = audio_pll_get_usage(AUDIOPLL_TYPE_0, &get_series);
  425. if (ret) {
  426. LOG_ERR("Get AUDIO_PLL0 error %d", ret);
  427. return ret;
  428. }
  429. /* Keep the same series within the pll setting */
  430. if ((0xFF == get_series) || (series == (a_pll_series_e)get_series)) {
  431. pll_index = AUDIOPLL_TYPE_0;
  432. } else {
  433. ret = audio_pll_get_usage(AUDIOPLL_TYPE_1, &get_series);
  434. if (ret) {
  435. LOG_ERR("Get AUDIO_PLL1 error %d", ret);
  436. return ret;
  437. }
  438. if ((0xFF == get_series) || (series == (a_pll_series_e)get_series))
  439. pll_index = AUDIOPLL_TYPE_1;
  440. }
  441. if (0xFF == pll_index) {
  442. LOG_ERR("Failed to find the available pll %d", series);
  443. LOG_INF("AUDIO_PLL0: 0x%x", sys_read32(AUDIO_PLL0_CTL));
  444. LOG_INF("AUDIO_PLL1: 0x%x", sys_read32(AUDIO_PLL1_CTL));
  445. *index = 0xFF;
  446. return -ENOENT;
  447. }
  448. *index = pll_index;
  449. audio_pll_set(pll_index, series);
  450. return 0;
  451. }