tcamerapluss3_audio_codec.cc 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162
  1. #include "tcamerapluss3_audio_codec.h"
  2. #include <esp_log.h>
  3. #include <driver/i2c_master.h>
  4. #include <driver/i2s_tdm.h>
  5. #include <driver/i2s_pdm.h>
  6. #include "config.h"
  7. static const char TAG[] = "Tcamerapluss3AudioCodec";
  8. Tcamerapluss3AudioCodec::Tcamerapluss3AudioCodec(int input_sample_rate, int output_sample_rate,
  9. gpio_num_t mic_bclk, gpio_num_t mic_ws, gpio_num_t mic_data,
  10. gpio_num_t spkr_bclk, gpio_num_t spkr_lrclk, gpio_num_t spkr_data,
  11. bool input_reference) {
  12. duplex_ = true; // 是否双工
  13. input_reference_ = input_reference; // 是否使用参考输入,实现回声消除
  14. input_channels_ = input_reference_ ? 2 : 1; // 输入通道数
  15. input_sample_rate_ = input_sample_rate;
  16. output_sample_rate_ = output_sample_rate;
  17. CreateVoiceHardware(mic_bclk, mic_ws, mic_data, spkr_bclk, spkr_lrclk, spkr_data);
  18. ESP_LOGI(TAG, "Tcamerapluss3AudioCodec initialized");
  19. #ifdef CONFIG_BOARD_TYPE_LILYGO_T_CAMERAPLUS_S3_V1_2
  20. gpio_config_t config_mic_spkr_en;
  21. config_mic_spkr_en.pin_bit_mask = BIT64(AUDIO_MIC_SPKR_EN);
  22. config_mic_spkr_en.mode = GPIO_MODE_OUTPUT;
  23. config_mic_spkr_en.pull_up_en = GPIO_PULLUP_ENABLE;
  24. config_mic_spkr_en.pull_down_en = GPIO_PULLDOWN_DISABLE;
  25. config_mic_spkr_en.intr_type = GPIO_INTR_DISABLE;
  26. #if SOC_GPIO_SUPPORT_PIN_HYS_FILTER
  27. config_mic_spkr_en.hys_ctrl_mode = GPIO_HYS_SOFT_ENABLE;
  28. #endif
  29. gpio_config(&config_mic_spkr_en);
  30. gpio_set_level(AUDIO_MIC_SPKR_EN, 0);
  31. #endif
  32. }
  33. Tcamerapluss3AudioCodec::~Tcamerapluss3AudioCodec() {
  34. audio_codec_delete_codec_if(in_codec_if_);
  35. audio_codec_delete_ctrl_if(in_ctrl_if_);
  36. audio_codec_delete_codec_if(out_codec_if_);
  37. audio_codec_delete_ctrl_if(out_ctrl_if_);
  38. audio_codec_delete_gpio_if(gpio_if_);
  39. audio_codec_delete_data_if(data_if_);
  40. }
  41. void Tcamerapluss3AudioCodec::CreateVoiceHardware(gpio_num_t mic_bclk, gpio_num_t mic_ws, gpio_num_t mic_data,
  42. gpio_num_t spkr_bclk, gpio_num_t spkr_lrclk, gpio_num_t spkr_data) {
  43. i2s_chan_config_t mic_chan_config = I2S_CHANNEL_DEFAULT_CONFIG(I2S_NUM_0, I2S_ROLE_MASTER);
  44. mic_chan_config.auto_clear = true; // Auto clear the legacy data in the DMA buffer
  45. i2s_chan_config_t spkr_chan_config = I2S_CHANNEL_DEFAULT_CONFIG(I2S_NUM_1, I2S_ROLE_MASTER);
  46. spkr_chan_config.auto_clear = true; // Auto clear the legacy data in the DMA buffer
  47. ESP_ERROR_CHECK(i2s_new_channel(&mic_chan_config, NULL, &rx_handle_));
  48. ESP_ERROR_CHECK(i2s_new_channel(&spkr_chan_config, &tx_handle_, NULL));
  49. #ifdef CONFIG_BOARD_TYPE_LILYGO_T_CAMERAPLUS_S3_V1_0_V1_1
  50. i2s_std_config_t mic_config = {
  51. .clk_cfg = {
  52. .sample_rate_hz = (uint32_t)input_sample_rate_,
  53. .clk_src = I2S_CLK_SRC_DEFAULT,
  54. .mclk_multiple = I2S_MCLK_MULTIPLE_256,
  55. #ifdef I2S_HW_VERSION_2
  56. .ext_clk_freq_hz = 0,
  57. #endif
  58. },
  59. .slot_cfg = I2S_STD_PHILIPS_SLOT_DEFAULT_CONFIG(I2S_DATA_BIT_WIDTH_16BIT, I2S_SLOT_MODE_MONO),
  60. .gpio_cfg = {
  61. .mclk = I2S_GPIO_UNUSED,
  62. .bclk = mic_bclk,
  63. .ws = mic_ws,
  64. .dout = I2S_GPIO_UNUSED,
  65. .din = mic_data,
  66. .invert_flags = {
  67. .mclk_inv = false,
  68. .bclk_inv = false,
  69. .ws_inv = true // 默认右通道
  70. }
  71. }
  72. };
  73. ESP_ERROR_CHECK(i2s_channel_init_std_mode(rx_handle_, &mic_config));
  74. #elif defined CONFIG_BOARD_TYPE_LILYGO_T_CAMERAPLUS_S3_V1_2
  75. i2s_pdm_rx_config_t mic_config = {
  76. .clk_cfg = I2S_PDM_RX_CLK_DEFAULT_CONFIG(static_cast<uint32_t>(input_sample_rate_)),
  77. /* The data bit-width of PDM mode is fixed to 16 */
  78. .slot_cfg = I2S_PDM_RX_SLOT_DEFAULT_CONFIG(I2S_DATA_BIT_WIDTH_16BIT, I2S_SLOT_MODE_MONO),
  79. .gpio_cfg = {
  80. .clk = mic_ws,
  81. .din = mic_data,
  82. .invert_flags = {
  83. .clk_inv = false,
  84. },
  85. },
  86. };
  87. ESP_ERROR_CHECK(i2s_channel_init_pdm_rx_mode(rx_handle_, &mic_config));
  88. #endif
  89. i2s_std_config_t spkr_config = {
  90. .clk_cfg ={
  91. .sample_rate_hz = static_cast<uint32_t>(11025),
  92. .clk_src = I2S_CLK_SRC_DEFAULT,
  93. .ext_clk_freq_hz = 0,
  94. .mclk_multiple = I2S_MCLK_MULTIPLE_256,
  95. },
  96. .slot_cfg = I2S_STD_PHILIPS_SLOT_DEFAULT_CONFIG(I2S_DATA_BIT_WIDTH_16BIT, I2S_SLOT_MODE_STEREO),
  97. .gpio_cfg ={
  98. .mclk = I2S_GPIO_UNUSED,
  99. .bclk = spkr_bclk,
  100. .ws = spkr_lrclk,
  101. .dout = spkr_data,
  102. .din = I2S_GPIO_UNUSED,
  103. .invert_flags = {
  104. .mclk_inv = false,
  105. .bclk_inv = false,
  106. .ws_inv = false
  107. }
  108. }
  109. };
  110. ESP_ERROR_CHECK(i2s_channel_init_std_mode(tx_handle_, &spkr_config));
  111. ESP_LOGI(TAG, "Voice hardware created");
  112. }
  113. void Tcamerapluss3AudioCodec::SetOutputVolume(int volume) {
  114. volume_ = volume;
  115. AudioCodec::SetOutputVolume(volume);
  116. }
  117. void Tcamerapluss3AudioCodec::EnableInput(bool enable) {
  118. AudioCodec::EnableInput(enable);
  119. }
  120. void Tcamerapluss3AudioCodec::EnableOutput(bool enable) {
  121. AudioCodec::EnableOutput(enable);
  122. }
  123. int Tcamerapluss3AudioCodec::Read(int16_t *dest, int samples){
  124. if (input_enabled_){
  125. size_t bytes_read;
  126. i2s_channel_read(rx_handle_, dest, samples * sizeof(int16_t), &bytes_read, portMAX_DELAY);
  127. }
  128. return samples;
  129. }
  130. int Tcamerapluss3AudioCodec::Write(const int16_t *data, int samples){
  131. if (output_enabled_){
  132. size_t bytes_read;
  133. auto output_data = (int16_t *)malloc(samples * sizeof(int16_t));
  134. for (size_t i = 0; i < samples; i++){
  135. output_data[i] = (float)data[i] * (float)(volume_ / 100.0);
  136. }
  137. i2s_channel_write(tx_handle_, output_data, samples * sizeof(int16_t), &bytes_read, portMAX_DELAY);
  138. free(output_data);
  139. }
  140. return samples;
  141. }