i2s_hal.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281
  1. /*
  2. * SPDX-FileCopyrightText: 2020-2022 Espressif Systems (Shanghai) CO LTD
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. // The HAL layer for I2S (common part)
  7. #include "soc/soc.h"
  8. #include "hal/i2s_hal.h"
  9. #include "hal/i2s_std.h"
  10. #include "hal/i2s_pdm.h"
  11. #include "hal/i2s_tdm.h"
  12. #if SOC_I2S_HW_VERSION_2 && SOC_I2S_SUPPORTS_PDM_TX
  13. /* PDM tx high pass filter cut-off frequency and coeffecients list
  14. * [0]: cut-off frequency; [1]: param0; [2]: param5 */
  15. static const float cut_off_coef[21][3] = {
  16. {185, 0, 0}, {172, 0, 1}, {160, 1, 1},
  17. {150, 1, 2}, {137, 2, 2}, {126, 2, 3},
  18. {120, 0, 3}, {115, 3, 3}, {106, 1, 7},
  19. {104, 2, 4}, {92, 4, 4}, {91.5, 2, 7},
  20. {81, 4, 5}, {77.2, 3, 7}, {69, 5, 5},
  21. {63, 4, 7}, {58, 5, 6}, {49, 5, 7},
  22. {46, 6, 6}, {35.5, 6, 7}, {23.3, 7, 7}};
  23. #endif
  24. void i2s_hal_init(i2s_hal_context_t *hal, int port_id)
  25. {
  26. /* Get hardware instance */
  27. hal->dev = I2S_LL_GET_HW(port_id);
  28. }
  29. void i2s_hal_set_tx_clock(i2s_hal_context_t *hal, const i2s_clock_info_t *clk_info, i2s_clock_src_t clk_src)
  30. {
  31. #if SOC_I2S_HW_VERSION_2
  32. i2s_ll_tx_enable_clock(hal->dev);
  33. i2s_ll_mclk_bind_to_tx_clk(hal->dev);
  34. #endif
  35. i2s_ll_tx_clk_set_src(hal->dev, clk_src);
  36. i2s_ll_tx_set_mclk(hal->dev, clk_info->sclk, clk_info->mclk, clk_info->mclk_div);
  37. i2s_ll_tx_set_bck_div_num(hal->dev, clk_info->bclk_div);
  38. }
  39. void i2s_hal_set_rx_clock(i2s_hal_context_t *hal, const i2s_clock_info_t *clk_info, i2s_clock_src_t clk_src)
  40. {
  41. #if SOC_I2S_HW_VERSION_2
  42. i2s_ll_rx_enable_clock(hal->dev);
  43. i2s_ll_mclk_bind_to_rx_clk(hal->dev);
  44. #endif
  45. i2s_ll_rx_clk_set_src(hal->dev, clk_src);
  46. i2s_ll_rx_set_mclk(hal->dev, clk_info->sclk, clk_info->mclk, clk_info->mclk_div);
  47. i2s_ll_rx_set_bck_div_num(hal->dev, clk_info->bclk_div);
  48. }
  49. /*-------------------------------------------------------------------------
  50. | STD Specific Slot Configurations |
  51. -------------------------------------------------------------------------*/
  52. void i2s_hal_std_set_tx_slot(i2s_hal_context_t *hal, bool is_slave, const i2s_slot_config_t *slot_config)
  53. {
  54. i2s_std_slot_config_t *slot_cfg = (i2s_std_slot_config_t*)slot_config;
  55. uint32_t slot_bit_width = (int)slot_cfg->slot_bit_width < (int)slot_cfg->data_bit_width ?
  56. slot_cfg->data_bit_width : slot_cfg->slot_bit_width;
  57. i2s_ll_tx_reset(hal->dev);
  58. i2s_ll_tx_set_slave_mod(hal->dev, is_slave); //TX Slave
  59. i2s_ll_tx_set_sample_bit(hal->dev, slot_bit_width, slot_cfg->data_bit_width);
  60. i2s_ll_tx_enable_mono_mode(hal->dev, slot_cfg->slot_mode == I2S_SLOT_MODE_MONO);
  61. i2s_ll_tx_enable_msb_shift(hal->dev, slot_cfg->bit_shift);
  62. i2s_ll_tx_set_ws_width(hal->dev, slot_cfg->ws_width);
  63. i2s_ll_tx_select_slot(hal->dev, slot_cfg->slot_sel);
  64. #if SOC_I2S_HW_VERSION_1
  65. i2s_ll_tx_enable_msb_right(hal->dev, slot_cfg->msb_right);
  66. i2s_ll_tx_enable_right_first(hal->dev, slot_cfg->ws_pol);
  67. /* Should always enable fifo */
  68. i2s_ll_tx_force_enable_fifo_mod(hal->dev, true);
  69. #elif SOC_I2S_HW_VERSION_2
  70. i2s_ll_tx_set_half_sample_bit(hal->dev, slot_bit_width);
  71. i2s_ll_tx_set_ws_idle_pol(hal->dev, slot_cfg->ws_pol);
  72. i2s_ll_tx_set_bit_order(hal->dev, slot_cfg->bit_order_lsb);
  73. i2s_ll_tx_enable_left_align(hal->dev, slot_cfg->left_align);
  74. i2s_ll_tx_enable_big_endian(hal->dev, slot_cfg->big_endian);
  75. #endif
  76. }
  77. void i2s_hal_std_set_rx_slot(i2s_hal_context_t *hal, bool is_slave, const i2s_slot_config_t *slot_config)
  78. {
  79. i2s_std_slot_config_t *slot_cfg = (i2s_std_slot_config_t*)slot_config;
  80. uint32_t slot_bit_width = (int)slot_cfg->slot_bit_width < (int)slot_cfg->data_bit_width ?
  81. slot_cfg->data_bit_width : slot_cfg->slot_bit_width;
  82. i2s_ll_rx_reset(hal->dev);
  83. i2s_ll_rx_set_slave_mod(hal->dev, is_slave); //RX Slave
  84. i2s_ll_rx_set_sample_bit(hal->dev, slot_bit_width, slot_cfg->data_bit_width);
  85. i2s_ll_rx_enable_mono_mode(hal->dev, slot_cfg->slot_mode == I2S_SLOT_MODE_MONO);
  86. i2s_ll_rx_enable_msb_shift(hal->dev, slot_cfg->bit_shift);
  87. i2s_ll_rx_set_ws_width(hal->dev, slot_cfg->ws_width);
  88. i2s_ll_rx_select_slot(hal->dev, slot_cfg->slot_sel);
  89. #if SOC_I2S_HW_VERSION_1
  90. i2s_ll_rx_enable_msb_right(hal->dev, slot_cfg->msb_right);
  91. i2s_ll_rx_enable_right_first(hal->dev, slot_cfg->ws_pol);
  92. /* Should always enable fifo */
  93. i2s_ll_rx_force_enable_fifo_mod(hal->dev, true);
  94. #elif SOC_I2S_HW_VERSION_2
  95. i2s_ll_rx_set_half_sample_bit(hal->dev, slot_bit_width);
  96. i2s_ll_rx_set_ws_idle_pol(hal->dev, slot_cfg->ws_pol);
  97. i2s_ll_rx_set_bit_order(hal->dev, slot_cfg->bit_order_lsb);
  98. i2s_ll_rx_enable_left_align(hal->dev, slot_cfg->left_align);
  99. i2s_ll_rx_enable_big_endian(hal->dev, slot_cfg->big_endian);
  100. #endif
  101. }
  102. void i2s_hal_std_enable_tx_channel(i2s_hal_context_t *hal)
  103. {
  104. i2s_ll_tx_enable_std(hal->dev);
  105. }
  106. void i2s_hal_std_enable_rx_channel(i2s_hal_context_t *hal)
  107. {
  108. i2s_ll_rx_enable_std(hal->dev);
  109. }
  110. /*-------------------------------------------------------------------------
  111. | PDM Specific Slot Configurations |
  112. -------------------------------------------------------------------------*/
  113. #if SOC_I2S_SUPPORTS_PDM_TX
  114. void i2s_hal_pdm_set_tx_slot(i2s_hal_context_t *hal, bool is_slave, const i2s_slot_config_t *slot_config)
  115. {
  116. i2s_pdm_tx_slot_config_t *slot_cfg = (i2s_pdm_tx_slot_config_t *)slot_config;
  117. uint32_t slot_bit_width = (int)slot_cfg->slot_bit_width < (int)slot_cfg->data_bit_width ?
  118. slot_cfg->data_bit_width : slot_cfg->slot_bit_width;
  119. i2s_ll_tx_reset(hal->dev);
  120. i2s_ll_tx_set_slave_mod(hal->dev, is_slave); //TX Slave
  121. i2s_ll_tx_set_sample_bit(hal->dev, slot_bit_width, slot_cfg->data_bit_width);
  122. i2s_ll_tx_enable_mono_mode(hal->dev, slot_cfg->slot_mode == I2S_SLOT_MODE_MONO);
  123. i2s_ll_tx_set_pdm_prescale(hal->dev, slot_cfg->sd_prescale);
  124. i2s_ll_tx_set_pdm_hp_scale(hal->dev, slot_cfg->hp_scale);
  125. i2s_ll_tx_set_pdm_lp_scale(hal->dev, slot_cfg->lp_scale);
  126. i2s_ll_tx_set_pdm_sinc_scale(hal->dev, slot_cfg->sinc_scale);
  127. i2s_ll_tx_set_pdm_sd_scale(hal->dev, slot_cfg->sd_scale);
  128. #if SOC_I2S_HW_VERSION_1
  129. i2s_ll_tx_force_enable_fifo_mod(hal->dev, true);
  130. #elif SOC_I2S_HW_VERSION_2
  131. /* Still need to enable the first 2 TDM channel mask to get the correct number of frame */
  132. i2s_ll_tx_set_active_chan_mask(hal->dev, I2S_TDM_SLOT0 | I2S_TDM_SLOT1);
  133. i2s_ll_tx_enable_pdm_hp_filter(hal->dev, slot_cfg->hp_en);
  134. uint8_t cnt = 0;
  135. float min = 1000;
  136. float expt_cut_off = slot_cfg->hp_cut_off_freq_hz;
  137. /* Find the closest cut-off frequency and its coefficients */
  138. for (int i = 0; i < 21; i++) {
  139. float tmp = cut_off_coef[i][0] < expt_cut_off ? expt_cut_off - cut_off_coef[i][0] : cut_off_coef[i][0] - expt_cut_off;
  140. if (tmp < min) {
  141. min = tmp;
  142. cnt = i;
  143. }
  144. }
  145. i2s_ll_tx_set_pdm_hp_filter_param0(hal->dev, cut_off_coef[cnt][1]);
  146. i2s_ll_tx_set_pdm_hp_filter_param5(hal->dev, cut_off_coef[cnt][2]);
  147. i2s_ll_tx_enable_pdm_sd_codec(hal->dev, slot_cfg->sd_en);
  148. i2s_ll_tx_set_pdm_sd_dither(hal->dev, slot_cfg->sd_dither);
  149. i2s_ll_tx_set_pdm_sd_dither2(hal->dev, slot_cfg->sd_dither2);
  150. #endif
  151. }
  152. void i2s_hal_pdm_enable_tx_channel(i2s_hal_context_t *hal)
  153. {
  154. i2s_ll_tx_enable_pdm(hal->dev);
  155. }
  156. #endif
  157. #if SOC_I2S_SUPPORTS_PDM_RX
  158. void i2s_hal_pdm_set_rx_slot(i2s_hal_context_t *hal, bool is_slave, const i2s_slot_config_t *slot_config)
  159. {
  160. i2s_pdm_rx_slot_config_t *slot_cfg = (i2s_pdm_rx_slot_config_t *)slot_config;
  161. uint32_t slot_bit_width = (int)slot_cfg->slot_bit_width < (int)slot_cfg->data_bit_width ?
  162. slot_cfg->data_bit_width : slot_cfg->slot_bit_width;
  163. i2s_ll_rx_reset(hal->dev);
  164. i2s_ll_rx_set_slave_mod(hal->dev, is_slave); //RX Slave
  165. i2s_ll_rx_set_sample_bit(hal->dev, slot_bit_width, slot_cfg->data_bit_width);
  166. i2s_ll_rx_enable_mono_mode(hal->dev, slot_cfg->slot_mode == I2S_SLOT_MODE_MONO);
  167. #if SOC_I2S_HW_VERSION_1
  168. i2s_ll_rx_force_enable_fifo_mod(hal->dev, true);
  169. #elif SOC_I2S_HW_VERSION_2
  170. /* Still need to enable the first 2 TDM channel mask to get the correct number of frame */
  171. i2s_ll_rx_set_active_chan_mask(hal->dev, I2S_TDM_SLOT0 | I2S_TDM_SLOT1);
  172. #endif
  173. }
  174. void i2s_hal_pdm_enable_rx_channel(i2s_hal_context_t *hal)
  175. {
  176. i2s_ll_rx_enable_pdm(hal->dev);
  177. }
  178. #endif
  179. /*-------------------------------------------------------------------------
  180. | TDM Specific Slot Configurations |
  181. -------------------------------------------------------------------------*/
  182. #if SOC_I2S_SUPPORTS_TDM
  183. void i2s_hal_tdm_set_tx_slot(i2s_hal_context_t *hal, bool is_slave, const i2s_slot_config_t *slot_config)
  184. {
  185. i2s_tdm_slot_config_t *slot_cfg = (i2s_tdm_slot_config_t*)slot_config;
  186. uint32_t slot_bit_width = (int)slot_cfg->slot_bit_width < (int)slot_cfg->data_bit_width ?
  187. slot_cfg->data_bit_width : slot_cfg->slot_bit_width;
  188. uint32_t cnt;
  189. uint32_t msk = slot_cfg->slot_mask;
  190. /* Get the maximum slot number */
  191. cnt = 32 - __builtin_clz(msk);
  192. /* There should be at least 2 slots in total even for mono mode */
  193. cnt = cnt < 2 ? 2 : cnt;
  194. uint32_t total_slot = slot_cfg->total_slot > cnt ? slot_cfg->total_slot : cnt;
  195. i2s_ll_tx_reset(hal->dev);
  196. i2s_ll_tx_set_slave_mod(hal->dev, is_slave); //TX Slave
  197. i2s_ll_tx_set_sample_bit(hal->dev, slot_bit_width, slot_cfg->data_bit_width);
  198. i2s_ll_tx_enable_mono_mode(hal->dev, slot_cfg->slot_mode == I2S_SLOT_MODE_MONO);
  199. i2s_ll_tx_enable_msb_shift(hal->dev, slot_cfg->bit_shift);
  200. if (slot_cfg->ws_width == I2S_TDM_AUTO_WS_WIDTH) {
  201. i2s_ll_tx_set_ws_width(hal->dev, (total_slot * slot_bit_width) / 2);
  202. } else {
  203. i2s_ll_tx_set_ws_width(hal->dev, slot_cfg->ws_width);
  204. }
  205. i2s_ll_tx_set_ws_idle_pol(hal->dev, slot_cfg->ws_pol);
  206. i2s_ll_tx_set_chan_num(hal->dev, total_slot);
  207. /* In mono mode, there only should be one slot enabled, other inactive slots will transmit same data as enabled slot */
  208. i2s_ll_tx_set_active_chan_mask(hal->dev, (slot_cfg->slot_mode == I2S_SLOT_MODE_MONO) ?
  209. I2S_TDM_SLOT0 : (uint32_t)slot_cfg->slot_mask);
  210. i2s_ll_tx_set_skip_mask(hal->dev, slot_cfg->skip_mask);
  211. i2s_ll_tx_set_half_sample_bit(hal->dev, total_slot * slot_bit_width / 2);
  212. i2s_ll_tx_set_bit_order(hal->dev, slot_cfg->bit_order_lsb);
  213. i2s_ll_tx_enable_left_align(hal->dev, slot_cfg->left_align);
  214. i2s_ll_tx_enable_big_endian(hal->dev, slot_cfg->big_endian);
  215. }
  216. void i2s_hal_tdm_set_rx_slot(i2s_hal_context_t *hal, bool is_slave, const i2s_slot_config_t *slot_config)
  217. {
  218. i2s_tdm_slot_config_t *slot_cfg = (i2s_tdm_slot_config_t*)slot_config;
  219. uint32_t slot_bit_width = (int)slot_cfg->slot_bit_width < (int)slot_cfg->data_bit_width ?
  220. slot_cfg->data_bit_width : slot_cfg->slot_bit_width;
  221. uint32_t cnt;
  222. uint32_t msk = slot_cfg->slot_mask;
  223. for (cnt = 0; msk; cnt++, msk >>= 1);
  224. /* Get the maximum slot number */
  225. cnt = 32 - __builtin_clz(msk);
  226. /* There should be at least 2 slots in total even for mono mode */
  227. cnt = cnt < 2 ? 2 : cnt;
  228. uint32_t total_slot = slot_cfg->total_slot > cnt ? slot_cfg->total_slot : cnt;
  229. i2s_ll_rx_reset(hal->dev);
  230. i2s_ll_rx_set_slave_mod(hal->dev, is_slave); //RX Slave
  231. i2s_ll_rx_set_sample_bit(hal->dev, slot_bit_width, slot_cfg->data_bit_width);
  232. i2s_ll_rx_enable_mono_mode(hal->dev, slot_cfg->slot_mode == I2S_SLOT_MODE_MONO);
  233. i2s_ll_rx_enable_msb_shift(hal->dev, slot_cfg->bit_shift);
  234. if (slot_cfg->ws_width == I2S_TDM_AUTO_WS_WIDTH) {
  235. i2s_ll_rx_set_ws_width(hal->dev, (total_slot * slot_bit_width) / 2);
  236. } else {
  237. i2s_ll_rx_set_ws_width(hal->dev, slot_cfg->ws_width);
  238. }
  239. i2s_ll_rx_set_ws_idle_pol(hal->dev, slot_cfg->ws_pol);
  240. i2s_ll_rx_set_chan_num(hal->dev, total_slot);
  241. /* In mono mode, there only should be one slot enabled, other inactive slots will transmit same data as enabled slot */
  242. i2s_ll_rx_set_active_chan_mask(hal->dev, (slot_cfg->slot_mode == I2S_SLOT_MODE_MONO) ?
  243. I2S_TDM_SLOT0 : (uint32_t)slot_cfg->slot_mask);
  244. i2s_ll_rx_set_half_sample_bit(hal->dev, total_slot * slot_bit_width / 2);
  245. i2s_ll_rx_set_bit_order(hal->dev, slot_cfg->bit_order_lsb);
  246. i2s_ll_rx_enable_left_align(hal->dev, slot_cfg->left_align);
  247. i2s_ll_rx_enable_big_endian(hal->dev, slot_cfg->big_endian);
  248. }
  249. void i2s_hal_tdm_enable_tx_channel(i2s_hal_context_t *hal)
  250. {
  251. i2s_ll_tx_enable_tdm(hal->dev);
  252. }
  253. void i2s_hal_tdm_enable_rx_channel(i2s_hal_context_t *hal)
  254. {
  255. i2s_ll_rx_enable_tdm(hal->dev);
  256. }
  257. #endif