audio_codec.cc 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104
  1. #include "audio_codec.h"
  2. #include "board.h"
  3. #include "settings.h"
  4. #include <esp_log.h>
  5. #include <cstring>
  6. #include <driver/i2s_common.h>
  7. #define TAG "AudioCodec"
  8. AudioCodec::AudioCodec() {
  9. }
  10. AudioCodec::~AudioCodec() {
  11. }
  12. void AudioCodec::OnInputReady(std::function<bool()> callback) {
  13. on_input_ready_ = callback;
  14. }
  15. void AudioCodec::OnOutputReady(std::function<bool()> callback) {
  16. on_output_ready_ = callback;
  17. }
  18. void AudioCodec::OutputData(std::vector<int16_t>& data) {
  19. Write(data.data(), data.size());
  20. }
  21. bool AudioCodec::InputData(std::vector<int16_t>& data) {
  22. int duration = 30;
  23. int input_frame_size = input_sample_rate_ / 1000 * duration * input_channels_;
  24. data.resize(input_frame_size);
  25. int samples = Read(data.data(), data.size());
  26. if (samples > 0) {
  27. return true;
  28. }
  29. return false;
  30. }
  31. IRAM_ATTR bool AudioCodec::on_sent(i2s_chan_handle_t handle, i2s_event_data_t *event, void *user_ctx) {
  32. auto audio_codec = (AudioCodec*)user_ctx;
  33. if (audio_codec->output_enabled_ && audio_codec->on_output_ready_) {
  34. return audio_codec->on_output_ready_();
  35. }
  36. return false;
  37. }
  38. IRAM_ATTR bool AudioCodec::on_recv(i2s_chan_handle_t handle, i2s_event_data_t *event, void *user_ctx) {
  39. auto audio_codec = (AudioCodec*)user_ctx;
  40. if (audio_codec->input_enabled_ && audio_codec->on_input_ready_) {
  41. return audio_codec->on_input_ready_();
  42. }
  43. return false;
  44. }
  45. void AudioCodec::Start() {
  46. Settings settings("audio", false);
  47. output_volume_ = settings.GetInt("output_volume", output_volume_);
  48. if (output_volume_ <= 0) {
  49. ESP_LOGW(TAG, "Output volume value (%d) is too small, setting to default (10)", output_volume_);
  50. output_volume_ = 10;
  51. }
  52. // 注册音频数据回调
  53. i2s_event_callbacks_t rx_callbacks = {};
  54. rx_callbacks.on_recv = on_recv;
  55. i2s_channel_register_event_callback(rx_handle_, &rx_callbacks, this);
  56. i2s_event_callbacks_t tx_callbacks = {};
  57. tx_callbacks.on_sent = on_sent;
  58. i2s_channel_register_event_callback(tx_handle_, &tx_callbacks, this);
  59. ESP_ERROR_CHECK(i2s_channel_enable(tx_handle_));
  60. ESP_ERROR_CHECK(i2s_channel_enable(rx_handle_));
  61. EnableInput(true);
  62. EnableOutput(true);
  63. ESP_LOGI(TAG, "Audio codec started");
  64. }
  65. void AudioCodec::SetOutputVolume(int volume) {
  66. output_volume_ = volume;
  67. ESP_LOGI(TAG, "Set output volume to %d", output_volume_);
  68. Settings settings("audio", true);
  69. settings.SetInt("output_volume", output_volume_);
  70. }
  71. void AudioCodec::EnableInput(bool enable) {
  72. if (enable == input_enabled_) {
  73. return;
  74. }
  75. input_enabled_ = enable;
  76. ESP_LOGI(TAG, "Set input enable to %s", enable ? "true" : "false");
  77. }
  78. void AudioCodec::EnableOutput(bool enable) {
  79. if (enable == output_enabled_) {
  80. return;
  81. }
  82. output_enabled_ = enable;
  83. ESP_LOGI(TAG, "Set output enable to %s", enable ? "true" : "false");
  84. }