es8388_audio_codec.cc 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205
  1. #include "es8388_audio_codec.h"
  2. #include <esp_log.h>
  3. #define TAG "Es8388AudioCodec"
  4. Es8388AudioCodec::Es8388AudioCodec(void* i2c_master_handle, i2c_port_t i2c_port, int input_sample_rate, int output_sample_rate,
  5. gpio_num_t mclk, gpio_num_t bclk, gpio_num_t ws, gpio_num_t dout, gpio_num_t din,
  6. gpio_num_t pa_pin, uint8_t es8388_addr) {
  7. duplex_ = true; // 是否双工
  8. input_reference_ = false; // 是否使用参考输入,实现回声消除
  9. input_channels_ = 1; // 输入通道数
  10. input_sample_rate_ = input_sample_rate;
  11. output_sample_rate_ = output_sample_rate;
  12. pa_pin_ = pa_pin; CreateDuplexChannels(mclk, bclk, ws, dout, din);
  13. // Do initialize of related interface: data_if, ctrl_if and gpio_if
  14. audio_codec_i2s_cfg_t i2s_cfg = {
  15. .port = I2S_NUM_0,
  16. .rx_handle = rx_handle_,
  17. .tx_handle = tx_handle_,
  18. };
  19. data_if_ = audio_codec_new_i2s_data(&i2s_cfg);
  20. assert(data_if_ != NULL);
  21. // Output
  22. audio_codec_i2c_cfg_t i2c_cfg = {
  23. .port = i2c_port,
  24. .addr = es8388_addr,
  25. .bus_handle = i2c_master_handle,
  26. };
  27. ctrl_if_ = audio_codec_new_i2c_ctrl(&i2c_cfg);
  28. assert(ctrl_if_ != NULL);
  29. gpio_if_ = audio_codec_new_gpio();
  30. assert(gpio_if_ != NULL);
  31. es8388_codec_cfg_t es8388_cfg = {};
  32. es8388_cfg.ctrl_if = ctrl_if_;
  33. es8388_cfg.gpio_if = gpio_if_;
  34. es8388_cfg.codec_mode = ESP_CODEC_DEV_WORK_MODE_BOTH;
  35. es8388_cfg.master_mode = true;
  36. es8388_cfg.pa_pin = pa_pin;
  37. es8388_cfg.pa_reverted = false;
  38. es8388_cfg.hw_gain.pa_voltage = 5.0;
  39. es8388_cfg.hw_gain.codec_dac_voltage = 3.3;
  40. codec_if_ = es8388_codec_new(&es8388_cfg);
  41. assert(codec_if_ != NULL);
  42. esp_codec_dev_cfg_t outdev_cfg = {
  43. .dev_type = ESP_CODEC_DEV_TYPE_OUT,
  44. .codec_if = codec_if_,
  45. .data_if = data_if_,
  46. };
  47. output_dev_ = esp_codec_dev_new(&outdev_cfg);
  48. assert(output_dev_ != NULL);
  49. esp_codec_dev_cfg_t indev_cfg = {
  50. .dev_type = ESP_CODEC_DEV_TYPE_IN,
  51. .codec_if = codec_if_,
  52. .data_if = data_if_,
  53. };
  54. input_dev_ = esp_codec_dev_new(&indev_cfg);
  55. assert(input_dev_ != NULL);
  56. esp_codec_set_disable_when_closed(output_dev_, false);
  57. esp_codec_set_disable_when_closed(input_dev_, false);
  58. ESP_LOGI(TAG, "Es8388AudioCodec initialized");
  59. }
  60. Es8388AudioCodec::~Es8388AudioCodec() {
  61. ESP_ERROR_CHECK(esp_codec_dev_close(output_dev_));
  62. esp_codec_dev_delete(output_dev_);
  63. ESP_ERROR_CHECK(esp_codec_dev_close(input_dev_));
  64. esp_codec_dev_delete(input_dev_);
  65. audio_codec_delete_codec_if(codec_if_);
  66. audio_codec_delete_ctrl_if(ctrl_if_);
  67. audio_codec_delete_gpio_if(gpio_if_);
  68. audio_codec_delete_data_if(data_if_);
  69. }
  70. void Es8388AudioCodec::CreateDuplexChannels(gpio_num_t mclk, gpio_num_t bclk, gpio_num_t ws, gpio_num_t dout, gpio_num_t din){
  71. assert(input_sample_rate_ == output_sample_rate_);
  72. i2s_chan_config_t chan_cfg = {
  73. .id = I2S_NUM_0,
  74. .role = I2S_ROLE_MASTER,
  75. .dma_desc_num = AUDIO_CODEC_DMA_DESC_NUM,
  76. .dma_frame_num = AUDIO_CODEC_DMA_FRAME_NUM,
  77. .auto_clear_after_cb = true,
  78. .auto_clear_before_cb = false,
  79. .intr_priority = 0,
  80. };
  81. ESP_ERROR_CHECK(i2s_new_channel(&chan_cfg, &tx_handle_, &rx_handle_));
  82. i2s_std_config_t std_cfg = {
  83. .clk_cfg = {
  84. .sample_rate_hz = (uint32_t)output_sample_rate_,
  85. .clk_src = I2S_CLK_SRC_DEFAULT,
  86. .ext_clk_freq_hz = 0,
  87. .mclk_multiple = I2S_MCLK_MULTIPLE_256
  88. },
  89. .slot_cfg = {
  90. .data_bit_width = I2S_DATA_BIT_WIDTH_16BIT,
  91. .slot_bit_width = I2S_SLOT_BIT_WIDTH_AUTO,
  92. .slot_mode = I2S_SLOT_MODE_STEREO,
  93. .slot_mask = I2S_STD_SLOT_BOTH,
  94. .ws_width = I2S_DATA_BIT_WIDTH_16BIT,
  95. .ws_pol = false,
  96. .bit_shift = true,
  97. .left_align = true,
  98. .big_endian = false,
  99. .bit_order_lsb = false
  100. },
  101. .gpio_cfg = {
  102. .mclk = mclk,
  103. .bclk = bclk,
  104. .ws = ws,
  105. .dout = dout,
  106. .din = din,
  107. .invert_flags = {
  108. .mclk_inv = false,
  109. .bclk_inv = false,
  110. .ws_inv = false
  111. }
  112. }
  113. };
  114. ESP_ERROR_CHECK(i2s_channel_init_std_mode(tx_handle_, &std_cfg));
  115. ESP_ERROR_CHECK(i2s_channel_init_std_mode(rx_handle_, &std_cfg));
  116. ESP_LOGI(TAG, "Duplex channels created");
  117. }
  118. void Es8388AudioCodec::SetOutputVolume(int volume) {
  119. ESP_ERROR_CHECK(esp_codec_dev_set_out_vol(output_dev_, volume));
  120. AudioCodec::SetOutputVolume(volume);
  121. }
  122. void Es8388AudioCodec::EnableInput(bool enable) {
  123. if (enable == input_enabled_) {
  124. return;
  125. }
  126. if (enable) {
  127. esp_codec_dev_sample_info_t fs = {
  128. .bits_per_sample = 16,
  129. .channel = 1,
  130. .channel_mask = 0,
  131. .sample_rate = (uint32_t)input_sample_rate_,
  132. .mclk_multiple = 0,
  133. };
  134. ESP_ERROR_CHECK(esp_codec_dev_open(input_dev_, &fs));
  135. ESP_ERROR_CHECK(esp_codec_dev_set_in_gain(input_dev_, 24.0));
  136. } else {
  137. ESP_ERROR_CHECK(esp_codec_dev_close(input_dev_));
  138. }
  139. AudioCodec::EnableInput(enable);
  140. }
  141. void Es8388AudioCodec::EnableOutput(bool enable) {
  142. if (enable == output_enabled_) {
  143. return;
  144. }
  145. if (enable) {
  146. esp_codec_dev_sample_info_t fs = {
  147. .bits_per_sample = 16,
  148. .channel = 1,
  149. .channel_mask = 0,
  150. .sample_rate = (uint32_t)output_sample_rate_,
  151. .mclk_multiple = 0,
  152. };
  153. ESP_ERROR_CHECK(esp_codec_dev_open(output_dev_, &fs));
  154. ESP_ERROR_CHECK(esp_codec_dev_set_out_vol(output_dev_, output_volume_));
  155. // Set analog output volume to 0dB, default is -45dB
  156. uint8_t reg_val = 30; // 0dB
  157. uint8_t regs[] = { 46, 47, 48, 49 }; // HP_LVOL, HP_RVOL, SPK_LVOL, SPK_RVOL
  158. for (uint8_t reg : regs) {
  159. ctrl_if_->write_reg(ctrl_if_, reg, 1, &reg_val, 1);
  160. }
  161. if (pa_pin_ != GPIO_NUM_NC) {
  162. gpio_set_level(pa_pin_, 1);
  163. }
  164. } else {
  165. ESP_ERROR_CHECK(esp_codec_dev_close(output_dev_));
  166. if (pa_pin_ != GPIO_NUM_NC) {
  167. gpio_set_level(pa_pin_, 0);
  168. }
  169. }
  170. AudioCodec::EnableOutput(enable);
  171. }
  172. int Es8388AudioCodec::Read(int16_t* dest, int samples) {
  173. if (input_enabled_) {
  174. ESP_ERROR_CHECK_WITHOUT_ABORT(esp_codec_dev_read(input_dev_, (void*)dest, samples * sizeof(int16_t)));
  175. }
  176. return samples;
  177. }
  178. int Es8388AudioCodec::Write(const int16_t* data, int samples) {
  179. if (output_enabled_) {
  180. ESP_ERROR_CHECK_WITHOUT_ABORT(esp_codec_dev_write(output_dev_, (void*)data, samples * sizeof(int16_t)));
  181. }
  182. return samples;
  183. }