STM32CubeMX使用(四)之DMA方式使用ADC、ADC使用遇到的问题

本文详述了在STM32G0B1RCT6芯片上使用ADC和DMA进行数据采集的过程,包括配置端口、开启DMA中断、测试ADC以及遇到的两个问题:通道间干扰和无输入电压时的非零读数。作者通过调整代码和增加独立的ADC数据获取函数解决了这些问题。
摘要由CSDN通过智能技术生成

一、概述

本文记录下ADC的使用过程,主要是后续使用,能有个参考。同样,以后遇到的ADC问题,也都收录在本文后面。
芯片: STM32G0B1RCT6

二、Pinout&Configuration

2.1 选择端口

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

2.2 开启DMA中断

在这里插入图片描述

三、测试ADC

测试代码可参考demo工程,同样是DMA方式:
在这里插入图片描述
主要步骤为:

  1. 定义采集值缓存
    在这里插入图片描述
  2. 开启ADC校准在这里插入图片描述
  3. 开启DMA功能*。在这里插入图片描述
  4. 主循环内重复开启ADC转换并打印采集结果:在这里插入图片描述
  5. 发现打印结果如下,每次只转换了一个通道:
    在这里插入图片描述
    开启ADC后1S还不够采集完成吗?
  6. 回到demo中查看,是在回调函数中设置标志,然后在主循环中判断标志才进行打印。
    在这里插入图片描述
    修改后果然正常了:
    在这里插入图片描述
    这样看来每次开启ADC,只转换了一个通道。CubeMX设置有些问题。使能了scan扫描模式,应该会转换完所有通道吧?回头再看下。

注意:HAL_ADC_Start_DMA后,不要使用立即调用HAL_ADC_Start。

2021-12-9补充:

  1. 前面混合使用了HAL_ADC_Start_DMA和HAL_ADC_Start,是有些问题的。可以只使用HAL_ADC_Start_DMA,但要注意,不能重复调用,每次转换完要先使用HAL_ADC_Stop_DMA关闭。

  2. 修改cube设置
    在这里插入图片描述
    转换完所有通道后关闭ADC。

四、遇到的问题

4.1 采集到其他通道的值

使用N32G031C8开发板,采集channel1和channel8两路ADC。发现只给channel1输入2V电压,channel8采集值也随之改变。两次采集之间也增加了500ms延时,是什么原因呢?
暂时没有细究。开发板只是把IO直接引出来方便测试,并没有包含外围分压电路。在channel8也输入一路电压,可以看到channel1和channel8都能够正常采集了。

继续4.1的问题

硬件做好后测试,采集channel1、channel8、channel7三路电压,发现仍然存在问题:
在这里插入图片描述
正常情况下,只有ADU路是有电压的,但有时ADU会采集到0,或者HW_VER路采集到ADU的电压值。测试了多通道采集和连续转换,发现都不是问题的原因。开发板例程的ADC采集了浪路,参考设置是一样的,仍然有问题。最终发现,增加一个ADC_GetData函数,就可以了。

static uint16_t ADC_GetData(uint8_t ADC_Channel)
{
    uint16_t dat;
    
    ADC_ConfigRegularChannel(ADC1, ADC_Channel, 1, ADC_SAMP_TIME_56CYCLES5);
    /* Start ADC Software Conversion */
    ADC_EnableSoftwareStartConv(ADC1, ENABLE);
    while(ADC_GetFlagStatus(ADC1, ADC_FLAG_ENDC)==0){
    }
    ADC_ClearFlag(ADC1, ADC_FLAG_ENDC);
    ADC_ClearFlag(ADC1, ADC_FLAG_STR);
    dat = ADC_GetDat(ADC1);
    
    return dat;
}


#if USE_ADC1
static int32_t n32_adc1_read(uint8_t channel)
{
    int32_t read_value = 0;
    uint16_t dat;
    
//    ADC_ConfigRegularChannel(ADC1, channel, 1, ADC_SAMP_TIME_56CYCLES5);
//    /* Start ADC Software Conversion */
//    ADC_EnableSoftwareStartConv(ADC1, ENABLE);
//    while(ADC_GetFlagStatus(ADC1, ADC_FLAG_ENDC)==0){
//    }
//    ADC_ClearFlag(ADC1, ADC_FLAG_ENDC);
//    ADC_ClearFlag(ADC1, ADC_FLAG_STR);
//    
    read_value = ADC_GetDat(ADC1);
//    dat = ADC_GetDat(ADC1);
//    read_value = dat;
    
    read_value = ADC_GetData(channel);

    return read_value;
}
#endif

为什么单独封装一个函数,效果就不一样了呢?暂时没搞明白

4.2 STM32 ADC没有输入电压时,采集结果不为0

单独写为一篇:添加链接描述

  • 5
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
为了实现ADC多通道DMA滤波,需要进行以下步骤: 1. 使用STM32CubeMX配置ADCDMASTM32CubeMX中,选择正确的芯片型号,打开“Pinout & Configuration”选项卡,然后配置ADCDMA的参数。在“ADC”选项卡中,选择要使用的通道和采样时间,然后在“DMA”选项卡中,配置DMA通道和传输模式。 2. 设置ADC DMA模式 在代码中,需要将ADC设置为DMA模式。这可以通过在HAL_ADC_Start_DMA函数中设置相应的参数来实现。例如,以下代码将ADC设置为连续DMA模式: HAL_ADC_Start_DMA(&hadc1, (uint32_t*)ADCBuffer, ADC_BUFFER_SIZE); 3. 实现数据滤波 在ADC数据传输到缓冲区之后,可以对数据进行滤波处理。可以使用各种滤波算法,例如移动平均、中值滤波等。以下是一个使用移动平均滤波的示例代码: #define N 10 uint32_t FilteredData[ADC_NUM_CHANNELS]; void FilterData(uint32_t* ADCData) { static uint32_t FilterBuffer[ADC_NUM_CHANNELS][N]; static uint32_t FilterIndex[ADC_NUM_CHANNELS] = {0}; uint32_t i, j; for(i=0; i<ADC_NUM_CHANNELS; i++) { FilterBuffer[i][FilterIndex[i]] = ADCData[i]; FilterIndex[i] = (FilterIndex[i]+1) % N; FilteredData[i] = 0; for(j=0; j<N; j++) { FilteredData[i] += FilterBuffer[i][j]; } FilteredData[i] /= N; } } 在此示例中,FilterBuffer是一个二维数组,用于存储ADC数据。FilterIndex是一个数组,用于跟踪每个通道的滤波器缓冲区中的位置。FilteredData是一个数组,用于存储滤波后的数据。 4. 处理滤波后的数据 滤波后的数据现在可以用于您的应用程序。可以将其发送到计算机或其他设备,或者用于控制硬件。以下是一个将滤波后的数据通过串口发送到计算机的示例代码: #define UART_TX_BUFFER_SIZE 256 uint8_t UARTTxBuffer[UART_TX_BUFFER_SIZE]; void SendData(void) { uint32_t i; uint32_t len = 0; for(i=0; i<ADC_NUM_CHANNELS; i++) { len += sprintf((char*)(UARTTxBuffer+len), "%d,", FilteredData[i]); } UARTTxBuffer[len-1] = '\n'; HAL_UART_Transmit(&huart1, UARTTxBuffer, len, 100); } 在此示例中,使用sprintf函数将滤波后的数据转换为字符串,并将其存储在UARTTxBuffer中。然后,使用HAL_UART_Transmit函数将数据发送到计算机。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值