有几种方法可以将模拟音频数据输入到ESP32中。
直接从内置的模数转换器(ADC)读取
使用I2S通过DMA读取内置ADC
使用I2S直接从I2S兼容外设读取
ESP32上有两个内置ADC,ADC1和ADC2。
ADC1具有8个通道:
渠道
通用输入输出
渠道
通用输入输出
ADC1_CHANNEL_0
GPIO36
ADC1_CHANNEL_4
GPIO32
ADC1_CHANNEL_1
GPIO37
ADC1_CHANNEL_5
GPIO33
ADC1_CHANNEL_2
GPIO38
ADC1_CHANNEL_6
GPIO34
ADC1_CHANNEL_3
GPIO39
ADC1_CHANNEL_7
GPIO35
ADC2有10个通道:
渠道
通用输入输出
渠道
通用输入输出
ADC2_CHANNEL_0
GPIO4
ADC2_CHANNEL_5
GPIO12
ADC2_CHANNEL_1
GPIO0
ADC2_CHANNEL_6
GPIO14
ADC2_CHANNEL_2
GPIO2
ADC2_CHANNEL_7
GPIO27
ADC2_CHANNEL_3
GPIO15
ADC2_CHANNEL_8
GPIO25
ADC2_CHANNEL_4
GPIO13
ADC2_CHANNEL_9
GPIO26
尽管有一些限制-WiFi子系统也使用ADC2,并且某些引脚还用于控制启动行为的捆绑引脚。这意味着在项目中坚持使用ADC1是最安全的。
从ADC读取非常简单-您可以使用Arduino函数或直接使用Espressif函数:
// read using Arduino
int sample = analogRead(35)
// read using Espressif
int sample = adc1_get_raw(ADC1_CHANNEL_7);
ESP32 ADC非常不准确,如果您想获得准确的读数,可以使用校准设置。现在,这些操作大多在工厂完成,因此您的ESP32应该已经具有一些校准设置。也可以手动校准ADC。
要读取校准值,请使用以下代码,它将为您提供以毫伏为单位的值。这两个调用adc1_config_width
和adc1_config_channel_atten
是至关重要的,因为校准特性需要匹配ADC配置。
// calibration values for the adc
#define DEFAULT_VREF 1100
esp_adc_cal_characteristics_t *adc_chars;
//Range 0-4096
adc1_config_width(ADC_WIDTH_BIT_12);
// full voltage range
adc1_config_channel_atten(ADC1_CHANNEL_7, ADC_ATTEN_DB_11);
// get the ADC characteristics
esp_adc_cal_characterize(
ADC_UNIT_1,
ADC_ATTEN_DB_11,
ADC_WIDTH_BIT_12,
DEFAULT_VREF,
adc_chars);
// read a sample from the ADC
int sample = adc1_get_raw(ADC1_CHANNEL_7);
// get the calibrated value
int milliVolts = esp_adc_cal_raw_to_voltage(sample, adc_chars);
直接使用ADC可以进行低频和一次性采样。为了采样高质量的音频数据,您将需要以16-40KHz的频率采样。您可以使用计时器来执行此操作,但这并不是ESP32的CPU资源的最佳用途。
更好的方法是使用内置的I2S外设将ADC的样本直接读取到内存中。
这是使用I2S读取内置ADC的基本设置。
i2s_config_t i2s_config = {
.mode = (i2s_mode_t)(I2S_MODE_MASTER | I2S_MODE_RX | I2S_MODE_ADC_BUILT_IN),
.sample_rate = 40000,
.bits_per_sample = I2S_BITS_PER_SAMPLE_16BIT,
.channel_format = I2S_CHANNEL_FMT_ONLY_LEFT,
.communication_format = I2S_COMM_FORMAT_I2S_LSB,
.intr_alloc_flags = ESP_INTR_FLAG_LEVEL1,
.dma_buf_count = 2,
.dma_buf_len = 1024,
.use_apll = false,
.tx_desc_auto_clear = false,
.fixed_mclk = 0};
//install and start i2s driver
i2s_driver_install(I2S_NUM_0, &i2s_config, 4, &i2s_queue);
//init ADC pad
i2s_set_adc_mode(ADC_UNIT_1, ADC1_CHANNEL_7);
// enable the ADC
i2s_adc_enable(I2S_NUM_0);
// start a task to read samples from I2S
TaskHandle_t readerTaskHandle;
xTaskCreatePinnedToCore(readerTask, "Reader Task", 8192, this, 1, &readerTaskHandle, 0);
然后,您可以使用以下任务从ADC读取样本:
void readerTask(void *param)
{
I2SSampler *sampler = (I2SSampler *)param;
while (true)
{
// wait for some data to arrive on the queue
i2s_event_t evt;
if (xQueueReceive(sampler->i2s_queue, &evt, portMAX_DELAY) == pdPASS)
{
if (evt.type == I2S_EVENT_RX_DONE)
{
size_t bytesRead = 0;
do
{
// try and fill up our audio buffer
size_t bytesToRead = (ADC_SAMPLES_COUNT - sampler->audioBufferPos) * 2;
void *bufferPosition = (void *)(sampler->currentAudioBuffer + sampler->audioBufferPos);
// read from i2s
i2s_read(I2S_NUM_0, bufferPosition, bytesToRead, &bytesRead, 10 / portTICK_PERIOD_MS);
sampler->audioBufferPos += bytesRead / 2;
if (sampler->audioBufferPos == ADC_SAMPLES_COUNT)
{
// do something with the sample - e.g. notify another task to do some processing
}
} while (bytesRead > 0);
}
}
}
}
阅读完样本后,您可以执行所需的任何处理,I2S外设将在后台继续将样本从ADC读取到DMA缓冲区中。
MA4466的接线非常简单,只需将VCC连接至3v3,将GND连接至GND,将Out连接至与您要从其采样的ADC通道相对应的GPIO引脚。
尝试MAX9814的方法相同-您也可以通过将增益引脚连接至VCC或GND来发挥MAX9814的增益。
原文地址:https://blog.cmgresearch.com/2020/09/12/esp32-audio-input.html
手机扫一扫
移动阅读更方便
你可能感兴趣的文章