6.2 DMA+AD多通道

DMA+AD多通道

1.循环模式自动转运

AD.c

#include "stm32f10x.h"                  // Device header

uint16_t AD_Value[3];

void AD_Init(void)
{
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
	RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);
	
	RCC_ADCCLKConfig(RCC_PCLK2_Div6);
	
	GPIO_InitTypeDef GPIO_InitStructure;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0|GPIO_Pin_1|GPIO_Pin_2;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA, &GPIO_InitStructure);
	
	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);
	
	
	ADC_InitTypeDef ADC_IS;
	ADC_IS.ADC_ContinuousConvMode=ENABLE; //连续转换模式:连续
	ADC_IS.ADC_DataAlign=ADC_DataAlign_Right;    //数据对齐:右对齐
	ADC_IS.ADC_ExternalTrigConv=ADC_ExternalTrigConv_None; //触发源:软件触发
	ADC_IS.ADC_Mode=ADC_Mode_Independent;   //工作模式:独立模式
	ADC_IS.ADC_NbrOfChannel= 3;
	ADC_IS.ADC_ScanConvMode=ENABLE; //扫描转换模式
	ADC_Init(ADC1,&ADC_IS);
	
	DMA_InitTypeDef DMA_IS;
	DMA_IS.DMA_PeripheralBaseAddr= (uint32_t)&ADC1->DR ;//外设起始地址
	DMA_IS.DMA_PeripheralDataSize=  DMA_PeripheralDataSize_HalfWord  ; //外设数据宽度
	DMA_IS.DMA_PeripheralInc=  DMA_PeripheralInc_Disable ;  //外设是否自增
	DMA_IS.DMA_MemoryBaseAddr= (uint32_t)AD_Value ; //存储器起始地址
	DMA_IS.DMA_MemoryDataSize=  DMA_MemoryDataSize_HalfWord ;  //存储器数据宽度
	DMA_IS.DMA_MemoryInc=  DMA_MemoryInc_Enable ; //存储器是否自增
	DMA_IS.DMA_DIR=DMA_DIR_PeripheralSRC; //方向:外设到存储器,若要反向则为DMA_DIR_PeripheralDST
	DMA_IS.DMA_BufferSize=3; //传输数量
	DMA_IS.DMA_Mode=DMA_Mode_Circular; //循环模式
	DMA_IS.DMA_M2M=DMA_M2M_Disable;
	DMA_IS.DMA_Priority=DMA_Priority_VeryHigh ;//优先级:最高优先级
	DMA_Init(DMA1_Channel1, &DMA_IS);
	
	DMA_Cmd(DMA1_Channel1,ENABLE);//使能
	ADC_DMACmd(ADC1,ENABLE);//开启ADC到DMA的输出
	ADC_Cmd(ADC1,ENABLE);//上电
	
	ADC_ResetCalibration(ADC1);
	while( ADC_GetResetCalibrationStatus(ADC1) == SET);
	ADC_StartCalibration(ADC1);
	while( ADC_GetCalibrationStatus(ADC1) == SET);
	
	ADC_SoftwareStartConvCmd(ADC1,ENABLE);
}

//uint16_t AD_GetV(void)
//{
//	DMA_Cmd(DMA1_Channel1,DISABLE);
//	DMA_SetCurrDataCounter(DMA1_Channel1,3);
//	DMA_Cmd(DMA1_Channel1,ENABLE);	
//	
//	ADC_SoftwareStartConvCmd(ADC1,ENABLE);
//	
//	while (DMA_GetFlagStatus(DMA1_FLAG_TC1)==RESET);//完成标志位
//	DMA_ClearFlag(DMA1_FLAG_TC1);//清除标志位
//}

注释掉的部分是一会会用到的单次扫描转运,代码可以理解为DMA+AD,但要注意和之前的代码比扫描模式和循环模式的改动,并用ADC_DMACmd开启了DMA转运

main.c

#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "AD.h"


int main(void)
{
	OLED_Init();
	AD_Init();
	
	OLED_ShowString(1,1,"AD0:");
	OLED_ShowString(2,1,"AD1:");
	OLED_ShowString(3,1,"AD2:");
	while (1)
	{
//		AD_GetV();
		
		OLED_ShowNum(1, 5, AD_Value[0], 4);
		OLED_ShowNum(2, 5, AD_Value[1], 4);
		OLED_ShowNum(3, 5, AD_Value[2], 4);
		
		Delay_ms(100);
	}
}

简单的显示三个引脚的数据

接线图

现象同AD多通道

2.单次扫描+单次转运

AD.c

#include "stm32f10x.h"                  // Device header

uint16_t AD_Value[3];

void AD_Init(void)
{
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
	RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);
	
	RCC_ADCCLKConfig(RCC_PCLK2_Div6);
	
	GPIO_InitTypeDef GPIO_InitStructure;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0|GPIO_Pin_1|GPIO_Pin_2;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA, &GPIO_InitStructure);
	
	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);
	
	
	ADC_InitTypeDef ADC_IS;
	ADC_IS.ADC_ContinuousConvMode=DISABLE; //连续转换模式:连续
	ADC_IS.ADC_DataAlign=ADC_DataAlign_Right;    //数据对齐:右对齐
	ADC_IS.ADC_ExternalTrigConv=ADC_ExternalTrigConv_None; //触发源:软件触发
	ADC_IS.ADC_Mode=ADC_Mode_Independent;   //工作模式:独立模式
	ADC_IS.ADC_NbrOfChannel= 3;
	ADC_IS.ADC_ScanConvMode=ENABLE; //扫描转换模式
	ADC_Init(ADC1,&ADC_IS);
	
	DMA_InitTypeDef DMA_IS;
	DMA_IS.DMA_PeripheralBaseAddr= (uint32_t)&ADC1->DR ;//外设起始地址
	DMA_IS.DMA_PeripheralDataSize=  DMA_PeripheralDataSize_HalfWord  ; //外设数据宽度
	DMA_IS.DMA_PeripheralInc=  DMA_PeripheralInc_Disable ;  //外设是否自增
	DMA_IS.DMA_MemoryBaseAddr= (uint32_t)AD_Value ; //存储器起始地址
	DMA_IS.DMA_MemoryDataSize=  DMA_MemoryDataSize_HalfWord ;  //存储器数据宽度
	DMA_IS.DMA_MemoryInc=  DMA_MemoryInc_Enable ; //存储器是否自增
	DMA_IS.DMA_DIR=DMA_DIR_PeripheralSRC; //方向:外设到存储器,若要反向则为DMA_DIR_PeripheralDST
	DMA_IS.DMA_BufferSize=3; //传输数量
	DMA_IS.DMA_Mode=DMA_Mode_Normal; //循环模式
	DMA_IS.DMA_M2M=DMA_M2M_Disable;
	DMA_IS.DMA_Priority=DMA_Priority_VeryHigh ;//优先级:最高优先级
	DMA_Init(DMA1_Channel1, &DMA_IS);
	
	DMA_Cmd(DMA1_Channel1,ENABLE);//使能
	ADC_DMACmd(ADC1,ENABLE);//开启ADC到DMA的输出
	ADC_Cmd(ADC1,ENABLE);//上电
	
	ADC_ResetCalibration(ADC1);
	while( ADC_GetResetCalibrationStatus(ADC1) == SET);
	ADC_StartCalibration(ADC1);
	while( ADC_GetCalibrationStatus(ADC1) == SET);
	
//	ADC_SoftwareStartConvCmd(ADC1,ENABLE);
}

void AD_GetV(void)
{
	DMA_Cmd(DMA1_Channel1,DISABLE);
	DMA_SetCurrDataCounter(DMA1_Channel1,3);
	DMA_Cmd(DMA1_Channel1,ENABLE);	
	
	ADC_SoftwareStartConvCmd(ADC1,ENABLE);
	
	while (DMA_GetFlagStatus(DMA1_FLAG_TC1)==RESET);//完成标志位
	DMA_ClearFlag(DMA1_FLAG_TC1);//清除标志位
}

代码有所改动,注意对比

main.c

#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "AD.h"
#include "Key.h"
uint8_t KeyNum;
int main(void)
{
	OLED_Init();
	AD_Init();
	Key_Init();
	
	OLED_ShowString(1,1,"AD0:");
	OLED_ShowString(2,1,"AD1:");
	OLED_ShowString(3,1,"AD2:");
	while (1)
	{
		KeyNum=Key_GetNum();
		if(KeyNum==1){AD_GetV();}
		
		
		OLED_ShowNum(1, 5, AD_Value[0], 4);
		OLED_ShowNum(2, 5, AD_Value[1], 4);
		OLED_ShowNum(3, 5, AD_Value[2], 4);
		
		Delay_ms(100);
	}
}

这边为了直观的显示现象,我加入了一个按键控制搬运,每按一次按键可以转运一次数据,如果你想按不同的按钮转运指定引脚的数据可以把AD代码拆分成多个配置引脚或者其他配置方法这里就不详细的讲了

接线同上

现象:

效果有点不好,充电宝会断电

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值