2401_85308619 2024-11-05 22:08 采纳率: 44.4%
浏览 5
已结题

stm32配置多通道ADC

stm32多通道ADC是这样配置的吗?
adc.c

 #include "adc.h"
 #include "delay.h"
#include "stm32f10x_gpio.h"
       
void Adc_NVIC_Config(void)
{
    NVIC_InitTypeDef NVIC_InitStructure; 
    
    NVIC_InitStructure.NVIC_IRQChannel = ADC1_2_IRQn;
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
    
    NVIC_Init(&NVIC_InitStructure);
}
//初始化ADC
//这里我们仅以规则通道为例
//我们默认将开启通道1                                                                       
void  Adc_Init(void)
{     
    ADC_InitTypeDef ADC_InitStructure; 
    GPIO_InitTypeDef GPIO_InitStructure;

    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA |RCC_APB2Periph_ADC1    , ENABLE );      //使能ADC1通道时钟
    RCC_ADCCLKConfig(RCC_PCLK2_Div6);   //设置ADC分频因子6 72M/6=12,ADC最大时间不能超过14M

    //PA6 作为模拟通道输入引脚                       
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1 |GPIO_Pin_7;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;        //模拟输入引脚
    GPIO_Init(GPIOA, &GPIO_InitStructure);    

    ADC_DeInit(ADC1);  //复位ADC1 

    ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;//ADC工作模式
    ADC_InitStructure.ADC_ScanConvMode = DISABLE;    //模数转换工作在单通道模式
    ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;    //模数转换工作在多次转换模式
    ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;    //转换由软件而不是外部触发启动
    ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;    //ADC数据右对齐
    ADC_InitStructure.ADC_NbrOfChannel = 2;    //顺序进行规则转换的ADC通道的数目
    ADC_Init(ADC1, &ADC_InitStructure);    //根据ADC_InitStruct中指定的参数初始化外设ADCx的寄存器

  
    ADC_Cmd(ADC1, ENABLE);//使能指定的ADC1
    
    ADC_ResetCalibration(ADC1);    //使能复位校准
     
    while(ADC_GetResetCalibrationStatus(ADC1));    //等待复位校准结束
    
    ADC_StartCalibration(ADC1);    //开启AD校准
 
    while(ADC_GetCalibrationStatus(ADC1));    //等待校准结束
    
    
    //ADC_ITConfig(ADC1, ADC_IT_EOC, ENABLE);
    //ADC_RegularChannelConfig(ADC1, ADC_Channel_6, 1, ADC_SampleTime_1Cycles5);
    //ADC_SoftwareStartConvCmd(ADC1, ENABLE);
}                  
//获得ADC值
//ch:通道值 1
u16 Get_Adc(u8 ch)   
{
//设置指定ADC的规则组通道,一个序列,采样时间
    ADC_RegularChannelConfig(ADC1, ch, 2, ADC_SampleTime_1Cycles5 );//ADC1,ADC通道,采样时间为239.5周期            
  
    ADC_SoftwareStartConvCmd(ADC1, ENABLE);        //使能指定的ADC1的软件转换启动功能
     
    while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC ));//等待转换结束

    return ADC_GetConversionValue(ADC1);    //返回最近一次ADC1规则组的转换结果
}

u16 Get_Adc_Average(u8 ch,u8 times)
{
    u32 temp_val=0;
    u8 t;
    for(t=0;t<times;t++)
    {
        temp_val+=Get_Adc(ch);
        delay_ms(1);
    }
    return temp_val/times;
}      

void ADC1_2_IRQHandler(void)
{
    if (ADC_GetITStatus(ADC1, ADC_IT_EOC) == SET)
    {

    }
    ADC_ClearITPendingBit(ADC1, ADC_IT_EOC);
}

// 存放 ADC 通道采样值
volatile unsigned short int ADC_ConvertedValue[5] ;


void ADC1_DMA_Config(void)
{
    DMA_InitTypeDef  DMA_InitStructure;
    ADC_InitTypeDef  ADC_InitStructure;
    GPIO_InitTypeDef GPIO_InitStructure;
    
    // 开启 DMA 时钟
    RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);
    // 开启 ADC 时钟
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);
    // 开启 GPIO 时钟
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
    
    /* * * * * * ADC 引脚配置 * * * * * */
    

    // PA0/1/2/3/4 作为模拟通道输入引脚                         
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1|GPIO_Pin_7;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;        //模拟输入引脚
    GPIO_Init(GPIOA, &GPIO_InitStructure);    

        /* * * * * * DMA 模式配置 * * * * * */
    // 复位 DMA 控制器
    DMA_DeInit(DMA1_Channel1);
    
    // 源数据地址
    DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)&(ADC1->DR);
    // 目标地址
    DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)ADC_ConvertedValue;
    // 方向:外设到存储器
    DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
    // 传输大小
    DMA_InitStructure.DMA_BufferSize = 2;
    // 外设地址递增
    DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
    // 内存地址递增
    DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
    // 外设数据单位
    DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;
    // 内存数据单位
    DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;
    // DMA模式,循环
    // DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;
    DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
    // 优先级
    DMA_InitStructure.DMA_Priority = DMA_Priority_High;
    // 关闭内存到内存的传输
    DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
    // 初始化DMA
    DMA_Init(DMA1_Channel1, &DMA_InitStructure);
    // 使能DMA
    DMA_Cmd(DMA1_Channel1, ENABLE);
    
    /* * * * * * ADC 模式配置 * * * * * */
    
    // ADC_DeInit(ADC1);  //将外设 ADC1 的全部寄存器重设为缺省值

    // 只使用一个ADC,属于独立模式
    ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;    
    // 扫描模式
    ADC_InitStructure.ADC_ScanConvMode = ENABLE;    
    // 连续转换模式
    ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;    
    // 转换由软件开启
    ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;    
    // ADC数据右对齐
    ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;    
    // 顺序进行规则转换的ADC通道的数目
    ADC_InitStructure.ADC_NbrOfChannel = 2;    
    // 根据ADC_InitStruct中指定的参数初始化外设ADCx的寄存器   
    ADC_Init(ADC1, &ADC_InitStructure);    

    // 配置ADC时钟N狿CLK2的8分频,即9MHz
    RCC_ADCCLKConfig(RCC_PCLK2_Div8); 
    
    ADC_RegularChannelConfig( ADC1, ADC_Channel_1, 1, ADC_SampleTime_1Cycles5);
    ADC_RegularChannelConfig( ADC1, ADC_Channel_7, 2, ADC_SampleTime_1Cycles5);


    // 使能 ADC1 DMA 请求
    ADC_DMACmd(ADC1, ENABLE);    
    // 使能 ADC1, 开始转换
    ADC_Cmd(ADC1, ENABLE);
    
    ADC_ResetCalibration(ADC1);    //重置指定的ADC1的校准寄存器
     
    while(ADC_GetResetCalibrationStatus(ADC1));    //获取ADC1重置校准寄存器的状态,设置状态则等待
    
    ADC_StartCalibration(ADC1);        //开始指定ADC1的校准状态
 
    while(ADC_GetCalibrationStatus(ADC1));        //获取指定ADC1的校准程序,设置状态则等待
 
    
    // ADC1的软件转换启动
    ADC_SoftwareStartConvCmd(ADC1, ENABLE);        
}



adc.h

#ifndef __ADC_H
#define __ADC_H
#include "sys.h"

extern uint16_t ADC_Value[4];

void Adc_Init(void);
u16 Get_Adc_Average(u8 ch,u8 times);
u16 Get_Adc(u8 ch);
void ADC1_2_IRQHandler(void);
void ADC1_DMA_Config(void);

#endif


  • 写回答

2条回答 默认 最新

  • 香格里辣 2024-11-05 22:09
    关注

    以下回复参考:皆我百晓生券券喵儿等免费微信小程序作答:

    你的代码大致上是正确的,但有一些细节需要检查和调整。在配置STM32的多通道ADC时,你需要注意以下几点:

    1. 时钟配置:确保你的ADC时钟配置正确。你的代码中有RCC_APB2PeriphClockCmd()RCC_ADCCLKConfig()来配置ADC的时钟,你需要根据你的STM32的具体型号和时钟树配置进行相应的调整。
    2. 通道配置:在Adc_Init()函数中,你设置了ADC通道的配置,但是看起来你只配置了通道1。如果你想要使用多个通道,你需要为每个通道都调用ADC_RegularChannelConfig()。在你的ADC1_DMA_Config()函数中,你为通道1和通道7设置了采样时间并启用了DMA,这看起来是正确的。
    3. 中断服务程序(ISR):在ADC1_2_IRQHandler()中,你检查了转换完成中断(EOC),但没有进行任何处理。在实际应用中,你可能需要在中断服务程序中处理转换结果,例如将结果存储到数组中。
    4. DMA配置:在ADC1_DMA_Config()中,你配置了DMA来自动处理ADC的转换结果。确保你的DMA配置(如通道、地址、传输大小等)与你的实际需求相匹配。
    5. ADC模式:你选择了独立模式ADC(ADC_Mode_Independent)。这是多ADC系统中的一个常见模式,确保这是你想要的模式。
    6. 代码注释:虽然代码注释有助于理解代码的功能和结构,但请确保注释与实际代码的功能相匹配。一些注释可能是遗留的或者已经过时。
    7. 测试和验证:配置完成后,确保通过实际的硬件测试你的代码。多通道ADC的配置可能因STM32的不同型号和工具链而略有不同,所以总是需要进行实际的硬件测试来验证配置是否正确。

    总的来说,你的代码提供了一个基本的框架来配置多通道ADC。在实际使用中,你可能还需要添加更多的功能和错误处理。确保你完全理解代码的功能并进行了充分的测试。

    本回答被题主选为最佳回答 , 对您是否有帮助呢?
    评论
查看更多回答(1条)

报告相同问题?

问题事件

  • 系统已结题 11月14日
  • 已采纳回答 11月6日
  • 创建了问题 11月5日