您的位置 首页 知识

根据STM32单片机ADC接连收集和DMA循环转化

基于STM32单片机ADC连续采集和DMA循环转换-描述:用ADC连续采集11路模拟信号,并由DMA传输到内存。ADC配置为扫描并且连续转换模式,ADC的时钟配置为12MHZ。在每次转换结束后,由DMA循环将转换的数据传输到内存中。ADC可以连续采集N次求平均值。最后通过串口传输出最后转换的结果。

描绘:用ADC接连收集11路模仿信号,并由DMA传输到内存。ADC装备为扫描而且接连转化方式,ADC的时钟装备为12MHZ。在每次转化完毕后,由DMA循环将转化的数据传输到内存中。ADC能够接连收集N次求平均值。最终经过串口传输出最终转化的成果。

程序如下:

#i nclude “stm32f10x.h” //这个头文件包含STM32F10x一切外围存放器、位、内存映射的界说

#i nclude “eval.h” //头文件(包含串口、按键、LED的函数声明)

#i nclude “SysTIckDelay.h”

#i nclude “UART_INTERFACE.h”

#i nclude

#define N 50 //每通道采50次

#define M 12 //为12个通道

vu16 AD_Value[N][M]; //用来存放ADC转化成果,也是DMA的方针地址

vu16 After_filter[M]; //用来存放求平均值之后的成果

int i;

void GPIO_ConfiguraTIon(void)

{

GPIO_InitTypeDef GPIO_InitStructure;

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //由于USART1管脚是以复用的方式接到GPIO口上的,所以运用复用推挽式输出

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

GPIO_Init(GPIOA, &GPIO_InitStructure);

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;

GPIO_Init(GPIOA, &GPIO_InitStructure);

//PA0/1/2 作为模仿通道输入引脚

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0| GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_3;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模仿输入引脚

GPIO_Init(GPIOA, &GPIO_InitStructure);

//PB0/1 作为模仿通道输入引脚

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0|GPIO_Pin_1;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模仿输入引脚

GPIO_Init(GPIOB, &GPIO_InitStructure);

//PC0/1/2/3/4/5 作为模仿通道输入引脚

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0|GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_3|GPIO_Pin_4|GPIO_Pin_5;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模仿输入引脚

GPIO_Init(GPIOC, &GPIO_InitStructure);

}

}

void RCC_ConfiguraTIon(void)

{

ErrorStatus HSEStartUpStatus;

RCC_DeInit(); //RCC 体系复位

RCC_HSEConfig(RCC_HSE_ON); //敞开HSE

HSEStartUpStatus = RCC_WaitForHSEStartUp(); //等候HSE准备好

if(HSEStartUpStatus == SUCCESS)

{

FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable); //Enable Prefetch Buffer

FLASH_SetLatency(FLASH_Latency_2); //Set 2 Latency cycles

RCC_HCLKConfig(RCC_SYSCLK_Div1); //AHB clock = SYSCLK

RCC_PCLK2Config(RCC_HCLK_Div1); //APB2 clock = HCLK

RCC_PCLK1Config(RCC_HCLK_Div2); //APB1 clock = HCLK/2

RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_6); //PLLCLK = 12MHz * 6 = 72 MHz

RCC_PLLCmd(ENABLE); //Enable PLL

while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET); //Wait till PLL is ready

RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK); //Select PLL as system clock source

while(RCC_GetSYSCLKSource() != 0x08); //Wait till PLL is used as system clock source

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOB

| RCC_APB2Periph_GPIOC |RCC_APB2Periph_ADC1 | RCC_APB2Periph_AFIO |RCC_APB2Periph_USART1, ENABLE ); //使能ADC1通道时钟,各个管脚时钟

RCC_ADCCLKConfig(RCC_PCLK2_Div6); //72M/6=12,ADC最大时刻不能超过14M

RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); //使能DMA传输

}

}

void ADC1_Configuration(void)

{

ADC_InitTypeDef ADC_InitStructure;

ADC_DeInit(ADC1); //将外设 ADC1 的悉数存放器重设为缺省值

ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; //ADC作业方式:ADC1和ADC2作业在独立方式

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数据右对齐

ADC_InitStructure.ADC_NbrOfChannel = M; //次序进行规矩转化的ADC通道的数目

ADC_Init(ADC1, &ADC_InitStructure); //依据ADC_InitStruct中指定的参数初始化外设ADCx的存放器

//设置指定ADC的规矩组通道,设置它们的转化次序和采样时刻

//ADC1,ADC通道x,规矩采样次序值为y,采样时刻为239.5周期

ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_239Cycles5 );

ADC_RegularChannelConfig(ADC1, ADC_Channel_1, 2, ADC_SampleTime_239Cycles5 );

ADC_RegularChannelConfig(ADC1, ADC_Channel_2, 3, ADC_SampleTime_239Cycles5 );

ADC_RegularChannelConfig(ADC1, ADC_Channel_3, 4, ADC_SampleTime_239Cycles5 );

ADC_RegularChannelConfig(ADC1, ADC_Channel_8, 5, ADC_SampleTime_239Cycles5 );

ADC_RegularChannelConfig(ADC1, ADC_Channel_9, 6, ADC_SampleTime_239Cycles5 );

ADC_RegularChannelConfig(ADC1, ADC_Channel_10, 7, ADC_SampleTime_239Cycles5 );

ADC_RegularChannelConfig(ADC1, ADC_Channel_11, 8, ADC_SampleTime_239Cycles5 );

ADC_RegularChannelConfig(ADC1, ADC_Channel_12, 9, ADC_SampleTime_239Cycles5 );

ADC_RegularChannelConfig(ADC1, ADC_Channel_13, 10, ADC_SampleTime_239Cycles5 );

ADC_RegularChannelConfig(ADC1, ADC_Channel_14, 11, ADC_SampleTime_239Cycles5 );

ADC_RegularChannelConfig(ADC1, ADC_Channel_15, 12, ADC_SampleTime_239Cycles5 );

// 敞开ADC的DMA支撑(要完结DMA功用,还需独立装备DMA通道等参数)

ADC_DMACmd(ADC1, ENABLE);

ADC_Cmd(ADC1, ENABLE); //使能指定的ADC1

ADC_ResetCalibration(ADC1); //复位指定的ADC1的校准存放器

while(ADC_GetResetCalibrationStatu

本文引证地址: http://www.21ic.com/app/mcu/201810/778451.htm

s(ADC1)); //获取ADC1复位校准存放器的状况,设置状况则等候

ADC_StartCalibration(ADC1); //开端指定ADC1的校准状况

while(ADC_GetCalibrationStatus(ADC1)); //获取指定ADC1的校准程序,设置状况则等候

}

void DMA_Configuration(void)

{

DMA_InitTypeDef DMA_InitStructure;

DMA_DeInit(DMA1_Channel1); //将DMA的通道1存放器重设为缺省值

DMA_InitStructure.DMA_PeripheralBaseAddr = (u32)&ADC1-》DR; //DMA外设ADC基地址

DMA_InitStructure.DMA_MemoryBaseAddr = (u32)&AD_Value; //DMA内存基地址

DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC; //内存作为数据传输的目的地

DMA_InitStructure.DMA_BufferSize = N*M; //DMA通道的DMA缓存的巨细

DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; //外设地址存放器不变

DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; //内存地址存放器递加

DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord; //数据宽度为16位

DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord; //数据宽度为16位

DMA_InitStructure.DMA_Mode = DMA_Mode_Circular; //作业在循环缓存方式

DMA_InitStructure.DMA_Priority = DMA_Priority_High; //DMA通道 x具有高优先级

DMA_InitStructure.DMA_M2M = DMA_M2M_Disable; //DMA通道x没有设置为内存到内存传输

DMA_Init(DMA1_Channel1, &DMA_InitStructure); //依据DMA_InitStruct中指定的参数初始化DMA的通道

}

//装备一切外设

void Init_All_Periph(void)

{

RCC_Configuration();

GPIO_Configuration();

ADC1_Configuration();

DMA_Configuration();

//USART1_Configuration();

USART_Configuration(9600);

}

u16 GetVolt(u16 advalue)

{

return (u16)(advalue * 330 / 4096); //求的成果扩展了100倍,便利下面求出小数

}

void filter(void)

{

int sum = 0;

u8 count;

for(i=0;i《12;i++)

{

for ( count=0;count

{

sum += AD_Value[count][i];

}

After_filter[i]=sum/N;

sum=0;

}

}

int main(void)

{

u16 value[M];

init_All_Periph();

SysTick_Initaize();

ADC_SoftwareStartConvCmd(ADC1, ENABLE);

DMA_Cmd(DMA1_Channel1, ENABLE); //发动DMA通道

while(1)

{

while(USART_GetFlagStatus(USART1,USART_FLAG_TXE)==RESET);//等候传输完结不然第一位数据简单丢掉

filter();

for(i=0;i《12;i++)

{

value[i]= GetVolt(After_filter[i]);

printf(“value[%d]:\t%d.%dv\n”,i,value[i]/100,value[i]0) ;

delay_ms(100);

}

}

}

总结

该程序中的两个宏界说,M和N,别离代表有多少个通道,每个通道转化多少次,能够修正其值。

曾呈现的问题:装备时钟时要知道外部晶振是多少,以便精确装备时钟。将转化值由二进制转化为十进制时,要先扩展100倍,便利显现小数。最终串口输出时在 printf句子之前加这句代码,避免输出的第一位数据丢掉:while(USART_GetFlagStatus(USART1,USART_FLAG_TXE)==RESET);

声明:本文内容来自网络转载或用户投稿,文章版权归原作者和原出处所有。文中观点,不代表本站立场。若有侵权请联系本站删除(kf@86ic.com)https://www.86ic.net/zhishi/347167.html

为您推荐

联系我们

联系我们

在线咨询: QQ交谈

邮箱: kf@86ic.com

关注微信
微信扫一扫关注我们

微信扫一扫关注我们

返回顶部