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代码拆分成多个配置引脚或者其他配置方法这里就不详细的讲了
接线同上
现象:
效果有点不好,充电宝会断电