STM32 ADC 多通道16路电压采集

下面介绍一种利用STM32单片机制作的16路多通道ADC采集电路图和源程序。采用USB接口与电脑连接,实则USB转串口方式,所以上位机可以用串口作为接口。电路图中利用LM324作为电压跟随器,起到保护单片机引脚的作用。直接在电脑USB取点,省去外接电源麻烦,实测耗电电流不到20ma.

1.主控电路图:

stm3216adc

2. USB转串口电路图

USB转TTL串口电路图

3.LM324电压跟随器电路图

电压跟随器

4.滤波电路图

滤波电路

5.16路接口电路图

16路接口电路图

6.电源电路图

电源电路图

7.16路ADC初始化程序:

void Adc_Init(void) 
{
//先初始化IO口
RCC->APB2ENR|=0X7<<2; //使能PORTA\PORAB\PORTC口时钟
GPIOA->CRL&=0X00000000;//PA0 1 2 3 4 5 6 7 anolog输入
GPIOB->CRL&=0XFFFFFF00;//PB0 1 anolog输入
GPIOC->CRL&=0XFF000000;//PC0 1 2 3 4 5 anolog输入
//通道10/11设置
RCC->APB2ENR|=1<<9; //ADC1时钟使能
RCC->APB2RSTR|=1<<9; //ADC1复位
RCC->APB2RSTR&=~(1<<9);//复位结束
RCC->CFGR&=~(3<<14); //分频因子清零
//SYSCLK/DIV2=12M ADC时钟设置为12M,ADC最大时钟不能超过14M!
//否则将导致ADC准确度下降!
RCC->CFGR|=2<<14;
ADC1->CR1&=0XF0FFFF; //工作模式清零
ADC1->CR1|=0<<16; //独立工作模式
ADC1->CR1&=~(1<<8); //非扫描模式
ADC1->CR2&=~(1<<1); //单次转换模式
ADC1->CR2&=~(7<<17);
ADC1->CR2|=7<<17; //软件控制转换
ADC1->CR2|=1<<20; //使用用外部触发(SWSTART)!!! 必须使用一个事件来触发
ADC1->CR2&=~(1<<11); //右对齐ADC1->SQR1&=~(0XF<<20);
ADC1->SQR1&=0<<20; //1个转换在规则序列中 也就是只转换规则序列1
//设置通道采样时间
ADC1->SMPR2&=0X00000000;//通道0,1,2,3,4,5,6,7,8,9采样时间清空
ADC1->SMPR2|=7<<27; //通道9 239.5周期,提高采样时间可以提高精确度
ADC1->SMPR2|=7<<24; //通道8 239.5周期,提高采样时间可以提高精确度
ADC1->SMPR2|=7<<21; //通道7 239.5周期,提高采样时间可以提高精确度
ADC1->SMPR2|=7<<18; //通道6 239.5周期,提高采样时间可以提高精确度
ADC1->SMPR2|=7<<15; //通道5 239.5周期,提高采样时间可以提高精确度
ADC1->SMPR2|=7<<12; //通道4 239.5周期,提高采样时间可以提高精确度
ADC1->SMPR2|=7<<9; //通道3 239.5周期,提高采样时间可以提高精确度
ADC1->SMPR2|=7<<6; //通道2 239.5周期,提高采样时间可以提高精确度
ADC1->SMPR2|=7<<3; //通道1 239.5周期,提高采样时间可以提高精确度
ADC1->SMPR2|=7<<0; //通道0 239.5周期,提高采样时间可以提高精确度
ADC1->SMPR1&=0XFFFC0000;//通道10,11,12,13,14,15采样时间清空
ADC1->SMPR1|=7<<15; //通道15 239.5周期,提高采样时间可以提高精确度
ADC1->SMPR1|=7<<12; //通道14 239.5周期,提高采样时间可以提高精确度
ADC1->SMPR1|=7<<9; //通道13 239.5周期,提高采样时间可以提高精确度
ADC1->SMPR1|=7<<6; //通道12 239.5周期,提高采样时间可以提高精确度
ADC1->SMPR1|=7<<3; //通道11 239.5周期,提高采样时间可以提高精确度
ADC1->SMPR1|=7<<0; //通道10 239.5周期,提高采样时间可以提高精确度
ADC1->CR2|=1<<0; //开启AD转换器
ADC1->CR2|=1<<3; //使能复位校准
while(ADC1->CR2&1<<3); //等待校准结束
//该位由软件设置并由硬件清除。在校准寄存器被初始化后该位将被清除。
ADC1->CR2|=1<<2; //开启AD校准
while(ADC1->CR2&1<<2); //等待校准结束
//该位由软件设置以开始校准,并在校准结束时由硬件清除
}

8.获取ADC值的程序:

//获得ADC值 
//ch:通道值 1~16
u16 Get_Adc(u8 ch)
{
u8 ch_ch;
switch(ch)
{
case 1:ch_ch = 8; break;
case 2:ch_ch = 9; break;
case 3:ch_ch = 14; break;
case 4:ch_ch = 15; break;
case 5:ch_ch = 6; break;
case 6:ch_ch = 7; break;
case 7:ch_ch = 4; break;
case 8:ch_ch = 5; break;
case 9:ch_ch = 2; break;
case 10:ch_ch = 3; break;
case 11:ch_ch = 1; break;
case 12:ch_ch = 0; break;
case 13:ch_ch = 12; break;
case 14:ch_ch = 13; break;
case 15:ch_ch = 11; break;
case 16:ch_ch = 10; break;
default:ch_ch = 88; break;
}
if(ch_ch==88)return 0;
//设置转换序列
ADC1->SQR3&=0XFFFFFFE0;//规则序列1 通道ch
ADC1->SQR3|=ch_ch;
ADC1->CR2|=1<<22; //启动规则转换通道
while(!(ADC1->SR&1<<1));//等待转换结束
return ADC1->DR; //返回adc值
}

9.把ADC值转换成电压值的函数:

// ch范围1~16 
void fetch_adc(u8 ch)
{
u16 adcx;
u32 temp;
if((ch==0)||(ch>16))return; // 如果不是1到16,通道无效,退出函数
adcx=Get_Adc(ch); // 获得ADC值
temp=(u32)adcx*3300/4096; // 计算电压值 单位mv
adcx=temp; // 获得计算出的电压值
adc_buf[ch*2-2] = adcx>>8; // 给adc_buf赋值 ,先赋高8位,后赋低8位
adc_buf[ch*2-1] = adcx;
}


  • 20
    点赞
  • 198
    收藏
    觉得还不错? 一键收藏
  • 6
    评论
STM32中,ADC是一个十分重要的模块,它可以用来对外部信号进行模拟量采集。对于多通道电压采集STM32ADC可以通过DMA方式实现多通道采集,这样可以提高采样效率和精度。 下面是一个示例代码,演示如何使用STM32ADC进行多通道电压采集: ```c #include "stm32f10x.h" #define ADC1_DR_Address ((u32)0x4001244C) void ADC1_GPIO_Config(void); void ADC1_Mode_Config(void); void ADC1_DMA_Config(void); u16 ADC_ConvertedValue[3]; int main(void) { DMA_InitTypeDef DMA_InitStructure; RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); /* DMA channel1 configuration ----------------------------------------------*/ DMA_DeInit(DMA1_Channel1); DMA_InitStructure.DMA_PeripheralBaseAddr = ADC1_DR_Address; DMA_InitStructure.DMA_MemoryBaseAddr = (u32)&ADC_ConvertedValue; DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC; DMA_InitStructure.DMA_BufferSize = 3; 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_InitStructure.DMA_Mode = DMA_Mode_Circular; DMA_InitStructure.DMA_Priority = DMA_Priority_High; DMA_InitStructure.DMA_M2M = DMA_M2M_Disable; DMA_Init(DMA1_Channel1, &DMA_InitStructure); DMA_Cmd(DMA1_Channel1, ENABLE); ADC1_GPIO_Config(); ADC1_Mode_Config(); ADC1_DMA_Config(); while (1); } void ADC1_GPIO_Config(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_AFIO, ENABLE); /* Configure PA0, PA1 and PA2 as analog input -----------------------------------------*/ GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_2; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; GPIO_Init(GPIOA, &GPIO_InitStructure); } void ADC1_Mode_Config(void) { ADC_InitTypeDef ADC_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE); /* ADC1 configuration ------------------------------------------------------*/ ADC_DeInit(ADC1); 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_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; ADC_InitStructure.ADC_NbrOfChannel = 3; ADC_Init(ADC1, &ADC_InitStructure); /* ADC1 regular channel configuration ---------------------------------------*/ ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_55Cycles5); ADC_RegularChannelConfig(ADC1, ADC_Channel_1, 2, ADC_SampleTime_55Cycles5); ADC_RegularChannelConfig(ADC1, ADC_Channel_2, 3, ADC_SampleTime_55Cycles5); /* Enable ADC1 DMA ----------------------------------------------------------*/ ADC_DMACmd(ADC1, ENABLE); /* Enable ADC1 --------------------------------------------------------------*/ ADC_Cmd(ADC1, ENABLE); /* ADC1 reset calibration ---------------------------------------------------*/ ADC_ResetCalibration(ADC1); while (ADC_GetResetCalibrationStatus(ADC1)); /* ADC1 calibration ---------------------------------------------------------*/ ADC_StartCalibration(ADC1); while (ADC_GetCalibrationStatus(ADC1)); } void ADC1_DMA_Config(void) { DMA_InitTypeDef DMA_InitStructure; RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); /* DMA channel1 configuration ----------------------------------------------*/ DMA_DeInit(DMA1_Channel1); DMA_InitStructure.DMA_PeripheralBaseAddr = ADC1_DR_Address; DMA_InitStructure.DMA_MemoryBaseAddr = (u32)&ADC_ConvertedValue; DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC; DMA_InitStructure.DMA_BufferSize = 3; 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_InitStructure.DMA_Mode = DMA_Mode_Circular; DMA_InitStructure.DMA_Priority = DMA_Priority_High; DMA_InitStructure.DMA_M2M = DMA_M2M_Disable; DMA_Init(DMA1_Channel1, &DMA_InitStructure); DMA_Cmd(DMA1_Channel1, ENABLE); } ``` 在代码中,使用了ADC1、DMA1和GPIOA模块。首先是GPIO的配置,将PA0、PA1和PA2配置为模拟输入。然后是ADC的配置,设置了ADC1的扫描模式和DMA模式,并将PA0、PA1和PA2分别作为ADC1的第1、2和3个通道。最后是DMA的配置,使能了DMA1的通道1,并设置了DMA的传输方向、缓存地址和大小等参数。在主函数中,使用了一个无限循环,这样可以让程序一直运行,不会退出。 在本示例中,使用了连续转换模式,因此ADC会不断地采集多个通道的电压值,并通过DMA将采集到的数据传输到内存中。因此,可以通过访问ADC_ConvertedValue数组来获取采样数据。 需要注意的是,在使用多通道采集时,采样顺序与通道顺序有关。在本示例中,采样顺序为ADC_Channel_0、ADC_Channel_1和ADC_Channel_2,因此ADC_ConvertedValue数组中的第一个元素是ADC_Channel_0的采样值,第二个元素是ADC_Channel_1的采样值,第三个元素是ADC_Channel_2的采样值。如果需要改变采样顺序,只需要修改ADC_RegularChannelConfig函数的通道顺序即可。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值