双ADC多通道

该代码示例展示了如何在STM32微控制器上配置ADC1和ADC2,使用DMA进行数据传输。ADC配置包括连续转换模式,多通道选择,以及校准。同时,DMA被用来从ADC的两个通道收集16位数据,高位来自ADC2,低位来自ADC1。主函数中,读取的数据被解析并打印,用于显示光敏电阻和模拟摇杆的值。
摘要由CSDN通过智能技术生成

这里使用ADC1和ADC2,高位为ADC2的值,低位为ADC1的值,各为16位。
adc.c文件:

#include "adc.h"
#include "dma.h"

extern uint32_t Data1[2];//ADC传输过来的数据


static void GPIO_Config(void)//PA6和PA7分别连接到了ADC得通道6和7
{
	GPIO_InitTypeDef GPIO_InitStrcut;
	//GPIOA的时钟
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
	
	GPIO_InitStrcut.GPIO_Pin = GPIO_Pin_6 | GPIO_Pin_7 | GPIO_Pin_5;//Pin_6和Pin7是模拟摇杆
	GPIO_InitStrcut.GPIO_Mode = GPIO_Mode_AIN;//模拟输入
	GPIO_Init(GPIOA, &GPIO_InitStrcut);
}

static void DMA_Config(void)//外设的模拟数据传输到单片机内部,所以是外设作为SRC
{
	//DMA在ADC2中没有,所以只用初始化一个DMA
	DMA_InitTypeDef DMA_InitStrcut;
	RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);//开启DMA1的时钟
	
	DMA_DeInit(DMA1_Channel1);
	DMA_InitStrcut.DMA_PeripheralBaseAddr = (uint32_t)&(ADC1->DR);//ADC数据寄存器的地址0x4001244C
	DMA_InitStrcut.DMA_MemoryBaseAddr = (uint32_t)Data1;
	DMA_InitStrcut.DMA_DIR = DMA_DIR_PeripheralSRC;
	
	DMA_InitStrcut.DMA_BufferSize = 2;//ADC的两个通道对应两个Buffer
	DMA_InitStrcut.DMA_PeripheralInc = DMA_PeripheralInc_Disable;//外设基地址不变
	//32位,由于两个ADC在数据转化的是分低16和高16位,所以这里一次传输的数据为32位
	DMA_InitStrcut.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Word;
	
	DMA_InitStrcut.DMA_MemoryInc = DMA_MemoryInc_Enable;//内存地址定义的是数组,所以这里用地址自增模式
	DMA_InitStrcut.DMA_MemoryDataSize = DMA_MemoryDataSize_Word;
	
	DMA_InitStrcut.DMA_Mode = DMA_Mode_Circular;//
	DMA_InitStrcut.DMA_Priority = DMA_Priority_High;//DMA优先级
	DMA_InitStrcut.DMA_M2M = DMA_M2M_Disable;//只要不是flash到SRAM的,都用Disable
	
	DMA_Init(DMA1_Channel1, &DMA_InitStrcut);
	DMA_ClearFlag(DMA1_FLAG_TC1);//清除标志位
	DMA_Cmd(DMA1_Channel1, ENABLE);//DMA1的通道1与ADC1连接,使能DMA1的通道1
}

static void ADC_Config(void)
{
	ADC_InitTypeDef ADC_InitStruct;
	
	/* --------ADC1模式配置(摇杆)--------------*/
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);//开启ADC1的时钟
	ADC_InitStruct.ADC_Mode = ADC_Mode_RegSimult;//如果是使用1个ADC就选择独立模型,多个ADC则一般选择ADC_Mode_RegSimult
	ADC_InitStruct.ADC_DataAlign = ADC_DataAlign_Right;//一般选择右对齐
	ADC_InitStruct.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;//软件触发
	ADC_InitStruct.ADC_ContinuousConvMode = ENABLE;//使能连续转换模式
	ADC_InitStruct.ADC_NbrOfChannel = 2;//选择两通道
	ADC_InitStruct.ADC_ScanConvMode = ENABLE;//多通道时,要使能
	
	ADC_Init(ADC1, &ADC_InitStruct);
	RCC_ADCCLKConfig(RCC_PCLK2_Div8);
	ADC_RegularChannelConfig(ADC1, ADC_Channel_6, 1, ADC_SampleTime_55Cycles5);
	ADC_RegularChannelConfig(ADC1, ADC_Channel_7, 2, ADC_SampleTime_55Cycles5);
	
	/* --------ADC2模式配置(光敏电阻)--------------*/
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC2, ENABLE);//开启ADC1的时钟
	ADC_InitStruct.ADC_Mode = ADC_Mode_RegSimult;//如果是使用1个ADC就选择独立模型,多个ADC则一般选择ADC_Mode_RegSimult
	ADC_InitStruct.ADC_DataAlign = ADC_DataAlign_Right;//一般选择右对齐
	ADC_InitStruct.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;//软件触发
	ADC_InitStruct.ADC_ContinuousConvMode = ENABLE;//使能连续转换模式
	ADC_InitStruct.ADC_NbrOfChannel = 1;//选择1通道
	ADC_InitStruct.ADC_ScanConvMode = DISABLE;//多通道时,要使能
	
	ADC_Init(ADC2, &ADC_InitStruct);
	RCC_ADCCLKConfig(RCC_PCLK2_Div8);//ADC时钟分频,72MHz/8=9MHz
	//设置指定ADC的规则组通道,设置它们的转化顺序和采样时间
	ADC_RegularChannelConfig(ADC2, ADC_Channel_5, 1, ADC_SampleTime_55Cycles5);
	
	
			/* --------ADC1校正--------------*/
	//使能ADC1的DMA通道
	ADC_DMACmd(ADC1, ENABLE);
	//使能ADC
	ADC_Cmd(ADC1, ENABLE);
	//初始化ADC校准寄存器
	ADC_ResetCalibration(ADC1);
	//等待ADC1初始化ADC校准寄存器初始化完成
	while(ADC_GetResetCalibrationStatus(ADC1));
	//开启校正
	ADC_StartCalibration(ADC1);
	while(ADC_GetCalibrationStatus(ADC1));//等待ADC1校准完成
	
	
			/* --------ADC2校正--------------*/
	//使能ADC
	ADC_Cmd(ADC2, ENABLE);
	//初始化ADC校准寄存器
	ADC_ResetCalibration(ADC2);
	//等待ADC1初始化ADC校准寄存器初始化完成
	while(ADC_GetResetCalibrationStatus(ADC2));
	//开启校正
	ADC_StartCalibration(ADC2);
	while(ADC_GetCalibrationStatus(ADC2));//等待ADC2校准完成
	
	
	ADC_ExternalTrigConvCmd(ADC2, ENABLE); //使能ADC2外部触发开始转换
	ADC_SoftwareStartConvCmd(ADC1, ENABLE); //使能ADC1软件开始转换
	
	
	
}

void ADC_DMA_Config(void)
{
	GPIO_Config();
	DMA_Config();
	ADC_Config();
	
}

main.c文件

#include "stm32f10x.h"
#include "adc.h"
#include "uart.h"
#include "systick.h"


vu32 Data1[2] = {0};//ADC传输过来的数据
uint32_t temp1;
uint32_t temp2;
uint32_t temp3;
uint32_t temp4;
float Data2[2] = {0};

int main(void)
{
	
	USART1_Init(9600);
	ADC_DMA_Config();
	
	while(1)
	{
		//定义了一个数组,第一个位置用来存放[ADC2,ADC1],第二位存放[ADC1],ADC为16位。
		//高16位,ADC2的值,光敏电阻的值
		temp2 = (Data1[0]&0xFFFF0000) >> 16;
		//ADC1通道6的值
		temp1 = (Data1[0]&0xFFFF);
		//ADC1通道7的值
		temp3 = (Data1[1]&0xFFFF);
		temp4 = (Data1[1]&0xFFFF0000) >> 16;
		
		
		printf("\r\n 通道5光敏电阻: ");
		printf("%d", temp2);
		printf(" 通道6: ");
		printf("%d", temp1);
		printf(" 通道7: ");
		printf("%d", temp3);
		printf(" xxx: ");
		printf("%d", temp4);
		printf("\r\n");
		delay_nms(1000);
	}


}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值