STM32八路ADC采用DMA方式

发布者:平和心态最新更新时间:2019-09-07 来源: eefocus关键字:STM32  八路ADC  DMA方式 手机看文章 扫描二维码
随时随地手机看文章

STM32八路ADC采用DMA方式


采集的数据如下 


程序打包地址 有DMA方式和非DMA 方式


https://download.csdn.net/download/qq_36958104/11352188


#include "adc.h"

#include "sys.h"

#define ADC1_DR_Address    ((u32)0x4001244C)

 

__IO u16 ADC_ConvertedValue;  

 

/*配置采样通道端口 使能GPIO时钟   设置ADC采样PA0端口信号*/

 void ADC1_GPIO_Config(void)

  GPIO_InitTypeDef GPIO_InitStructure;    

  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);   

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

  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;     //GPIO设置为模拟输入

  GPIO_Init(GPIOA, &GPIO_InitStructure);   

}

 

 

/*配置ADC1的工作模式为MDA模式  */

 void ADC1_Mode_Config(void)

{

  DMA_InitTypeDef DMA_InitStructure;

  ADC_InitTypeDef ADC_InitStructure;

  RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); //使能MDA1时钟

/* DMA channel1 configuration */

  DMA_DeInit(DMA1_Channel1);  //指定DMA通道

  DMA_InitStructure.DMA_PeripheralBaseAddr = ADC1_DR_Address;////ADC1地址---代表ADC1保存转换值的寄存器

  DMA_InitStructure.DMA_MemoryBaseAddr = (u32)&ADC_ConvertedValue; //设置DMA内存地址,ADC转换结果直接放入该地址

  DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC; //外设为设置为数据传输的来源

  DMA_InitStructure.DMA_BufferSize = 16; //传输总数据---2通道需要传输2个数据

  DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; //外设地址固定

  DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; //内存地址自增---总体表示始终从外设ADC1地址处取值---依次保存到连续的两个内存变量中---

  DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord; //内存传输数据单元---半字16位

  DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;

  DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;//循环模式---2个数据依次循环接收从外设ADC1传输过来的ADC值---

  DMA_InitStructure.DMA_Priority = DMA_Priority_High;

  DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;

  DMA_Init(DMA1_Channel1, &DMA_InitStructure);

  

  /* Enable DMA channel1 */

  DMA_Cmd(DMA1_Channel1, ENABLE);  //使能DMA通道

 

  RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE); //使能ADC1时钟

     

  /* ADC1 configuration */

  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_DataAlign = ADC_DataAlign_Left;   //使用数据左对齐

  ADC_InitStructure.ADC_NbrOfChannel =8;  // ADC转换通道个数

  ADC_Init(ADC1, &ADC_InitStructure);

 

  /* ADC1 regular channel11 configuration */ 

ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_55Cycles5); //通道1采样周期55.5个时钟周期

ADC_RegularChannelConfig(ADC1, ADC_Channel_1, 2, ADC_SampleTime_55Cycles5); 

ADC_RegularChannelConfig(ADC1, ADC_Channel_2, 3, ADC_SampleTime_55Cycles5);

ADC_RegularChannelConfig(ADC1, ADC_Channel_3, 4, ADC_SampleTime_55Cycles5);

ADC_RegularChannelConfig(ADC1, ADC_Channel_4, 5, ADC_SampleTime_55Cycles5);

ADC_RegularChannelConfig(ADC1, ADC_Channel_5, 6, ADC_SampleTime_55Cycles5);

ADC_RegularChannelConfig(ADC1, ADC_Channel_6, 7, ADC_SampleTime_55Cycles5);

ADC_RegularChannelConfig(ADC1, ADC_Channel_7, 8, ADC_SampleTime_55Cycles5);

  /* Enable ADC1 DMA */

  ADC_DMACmd(ADC1, ENABLE); //使能ADC的DMA

  

  /* Enable ADC1 */

  ADC_Cmd(ADC1, ENABLE); //使能ADC1

 

  /* Enable ADC1 reset calibaration register */   

  ADC_ResetCalibration(ADC1);   //复位校准

  /* Check the end of ADC1 reset calibration register */

  while(ADC_GetResetCalibrationStatus(ADC1));

 

  /* Start ADC1 calibaration */

  ADC_StartCalibration(ADC1);

  /* Check the end of ADC1 calibration */

  while(ADC_GetCalibrationStatus(ADC1));

     

  /* Start ADC1 Software Conversion */ 

  ADC_SoftwareStartConvCmd(ADC1, ENABLE);  //开始转换

}

 

/*初始化ADC1 */

void ADC1_Init(void)

{

ADC1_GPIO_Config();

ADC1_Mode_Config();

}

 

 

 

 

#ifndef __ADC_H

#define __ADC_H

 

 

#include "stm32f10x.h"

void ADC1_Init(void);

 

 

#endif /* __ADC_H */

 

 


 


#include "led.h"

#include "delay.h"

#include "key.h"

#include "sys.h"

#include "DMA_ADC.h"

#include "usart.h"  

#include "adc.h"

#include "stdio.h"

/************************************************

 ALIENTEK 战舰STM32F103开发板实验17

 ADC 实验 

 技术支持:www.openedv.com

 淘宝店铺:http://eboard.taobao.com 

 关注微信公众平台微信号:"正点原子",免费获取STM32资料。

 广州市星翼电子科技有限公司  

 作者:正点原子 @ALIENTEK

************************************************/

uint8_t table[8]={0x00,0x00};

 

//ADC转换的电压通过DMA传输到ARM

extern __IO u16 ADC_ConvertedValue[8];

//保存adc数据再数组内

float adc_vola[8];

void SoftReset(void)    //软件复位

{

    __set_FAULTMASK(1); // 关闭所有中断

    NVIC_SystemReset(); // 复位

}

//普通ADC检测数据

/*******************************

void adc_collect()

{

u16 PA0, PA1, PA4, PA5, PA6, PA7;

float temp0,temp1,temp4,temp5;

PA0=Get_Adc(ADC_Channel_0,1);//PA0

temp0=(float)PA0*(3.3/4096);

printf("PA0ADC:");

printf("%0.3f",temp0);//接口数据的id  5445

PA1=Get_Adc(ADC_Channel_1,2);

temp1=(float)PA1*(3.3/4096);

printf("PA1ADC:");

printf("%0.3f",temp1);//接口数据的id  5445

PA4=Get_Adc(ADC_Channel_4,3);

temp4=(float)PA4*(3.3/4096);

printf("PA4ADC:");

printf("%0.3f",temp4);//接口数据的id  5445

PA5=Get_Adc(ADC_Channel_5,4);

temp5=(float)PA5*(3.3/4096);

printf("PA5ADC:");

printf("%0.3f",temp5);//接口数据的id  5445

delay_ms(500);

}

*******************************************************/

 

void dma_adc_collect()

{

uint16_t i;

for(i=0;i<7;i++)

{

adc_vola[i]=(float)ADC_ConvertedValue[i]/4096*3.3;

printf("ADC_ConvertedValue[%d]=%4.3f vrn",i,adc_vola[i]);

}

 

 

 

}

 int main(void)

 {  

  u16 adcx;

uint8_t a;

float temp;

delay_init();     //延时函数初始化   

NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//设置中断优先级分组为组2:2位抢占优先级,2位响应优先级

uart_init(115200); //串口初始化为115200

  LED_Init();      //LED端口初始化

 // Adc_Init();   //ADC初始化

ADC1_Init(); //DMA   ADC初始化

 

      

while(1)

{

//ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_239Cycles5 ); //ADC1,ADC通道,采样时间为239.5周期

//ADC_RegularChannelConfig(ADC1, ADC_Channel_1, 1, ADC_SampleTime_239Cycles5 ); //ADC1,ADC通道,采样时间为239.5周期

//ADC_RegularChannelConfig(ADC1, ADC_Channel_2, 1, ADC_SampleTime_239Cycles5 ); //ADC1,ADC通道,采样时间为239.5周期

//ADC_RegularChannelConfig(ADC1, ADC_Channel_3, 1, ADC_SampleTime_239Cycles5 ); //ADC1,ADC通道,采样时间为239.5周期

//ADC_RegularChannelConfig(ADC1, ADC_Channel_4, 1, ADC_SampleTime_239Cycles5 ); //ADC1,ADC通道,采样时间为239.5周期

//ADC_RegularChannelConfig(ADC1, ADC_Channel_5, 1, ADC_SampleTime_239Cycles5 ); //ADC1,ADC通道,采样时间为239.5周期

//ADC_RegularChannelConfig(ADC1, ADC_Channel_6, 1, ADC_SampleTime_239Cycles5 ); //ADC1,ADC通道,采样时间为239.5周期

//ADC_RegularChannelConfig(ADC1, ADC_Channel_7, 1, ADC_SampleTime_239Cycles5 ); //ADC1,ADC通道,采样时间为239.5周期

 

 

// temp=(float)adcx*(3.3/4096);

// //USART_SendData(USART1, temp);

// printf("pA0 ADC:");

//// USART1_SendOneData(temp);

// printf("%0.3f",temp);//接口数据的id  5445

//adc_collect();

dma_adc_collect();

//SoftReset();

LED0=!LED0;

delay_ms(1000);

}

 }

 

USART3_Sned_Char('+');

   sprintf(usar_buff,"%d",shuju5);  //    烟雾   3

USART3_Sned_Char_Buff2(usar_buff);//

#include "timer.h"

#include "usart.h"

 

//V1.1 20120904

//1,增加TIM3_PWM_Init函数。

//2,增加LED0_PWM_VAL宏定义,控制TIM3_CH2脉宽   

//////////////////////////////////////////////////////////////////////////////////  

      

//通用定时器3中断初始化

//这里时钟选择为APB1的2倍,而APB1为36M

//arr:自动重装值。

//psc:时钟预分频数

//这里使用的是定时器3!

 

void TIM3_Int_Init(u16 arr,u16 psc)

{

  TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;

NVIC_InitTypeDef NVIC_InitStructure;

 

RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE); //时钟使能

 

TIM_TimeBaseStructure.TIM_Period = arr; //设置在下一个更新事件装入活动的自动重装载寄存器周期的值 计数到5000为500ms

TIM_TimeBaseStructure.TIM_Prescaler =psc; //设置用来作为TIMx时钟频率除数的预分频值  10Khz的计数频率  

TIM_TimeBaseStructure.TIM_ClockDivision = 0; //设置时钟分割:TDTS = Tck_tim

TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;  //TIM向上计数模式

TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure); //根据TIM_TimeBaseInitStruct中指定的参数初始化TIMx的时间基数单位

 

TIM_ITConfig(TIM3,TIM_IT_Update,ENABLE ); //使能指定的TIM3中断,允许更新中断

 

NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQn;  //TIM3中断

NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;  //先占优先级0级

NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;  //从优先级3级

NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道被使能

NVIC_Init(&NVIC_InitStructure);  //根据NVIC_InitStruct中指定的参数初始化外设NVIC寄存器

 

TIM_Cmd(TIM3, ENABLE);  //使能TIMx外设

 

}

//定时器3中断服务程序

void TIM3_IRQHandler(void)   //TIM3中断

{

if (TIM_GetITStatus(TIM3, TIM_IT_Update) != RESET) //检查指定的TIM中断发生与否:TIM 中断源 

{

TIM_ClearITPendingBit(TIM3, TIM_IT_Update  );  //清除TIMx的中断待处理位:TIM 中断源 

//LED1=!LED1;

}

}

 

 

 

 

//TIM3 PWM部分初始化 

//PWM输出初始化

//arr:自动重装值

//psc:时钟预分频数

void TIM3_PWM_Init(u16 arr,u16 psc)

{  

GPIO_InitTypeDef GPIO_InitStructure;

TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;

TIM_OCInitTypeDef  TIM_OCInitStructure;

 

RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE); //使能定时器3时钟

  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB  | RCC_APB2Periph_AFIO, ENABLE);  //使能GPIO外设和AFIO复用功能模块时钟

[1] [2]
关键字:STM32  八路ADC  DMA方式 引用地址:STM32八路ADC采用DMA方式

上一篇:STM32F1系列之常用外设说明
下一篇:结构体变量和结构体类型的定义

推荐阅读最新更新时间:2024-11-10 03:41

STM32——利用STM32CubeMX来点亮GPIO口的led
GPIO口的 LED 点亮(以STM32F407IGTb芯片为例 ) 1、掌握的GPIO口的使用方法 2、熟悉Keil uVision5工具软件的使用 原理: 只要是对硬件操作,就要首先查看原理图。 查看外设是给模块的MCU的哪个引脚相连。 FS-STM32F407开发平台上的LED的亮灭状态,与芯片上的引脚I/O输出电平有关。 FS-STM32F407开发平台上LED的I/O扩展: IO操作重要结构体:GPIO_InitTypeDef typedef struct { uint32_t Pin; 操作的管脚 uint32_t Mode; 模式选择 uint32_t Pull;
[单片机]
<font color='red'>STM32</font>——利用STM32CubeMX来点亮GPIO口的led
STM32 + GCC + J-Link :开源的力量
发现Keil自带的编译器有时候优化有问题,跑得跟没优化时完全不一样。以前使用GCC,现在自然而然想再次使用GCC。 搭配1:Keil+GCC+Ulink2 这类的介绍也较多,主要是一个链接脚本、启动代码、IDE选项配置的问题,大家可以google一下,个人觉得搭配一般般。 搭配2:emIDE+JLink 纯粹的Opensource血统,这个EMIDE已经带了GCC,只需要一个JLink,然后构建好工程就可以debug了,如果发展迅猛,估计Keil也会扛不住了。
[单片机]
STM32开发笔记73: C++中子类调用父类同名函数的处理方法
单片机型号:STM32L053R8T6 1、问题 父类有1方法:Enable_RS485,如下所示: class CUart { public: uint8_t u8_UartNumber; //端口号1-8 uint32_t u32_BaudRate; //波特率 uint8_t u8_Parity; //效验位 CC0 C0; //485控制引脚C0 CC1 C1; //485控制引脚C1 UART_HandleTypeDef hUART; uint8_t u8_UartReceiveBuffer ; public: CUart(uint8_t u8_UartNumber, uint3
[单片机]
基于STM32单片机设计的红外测温仪(带人脸检测)
由于医学发展的需要,在很多情况下,一般的温度计己经满足不了快速而又准确的测温要求,例如:车站、地铁、机场等人口密度较大的地方进行人体温度测量。 当前设计的这款红外非接触式测温仪由测温硬件+上位机软件组合而成,主要用在地铁、车站入口等地方,可以准确识别人脸进行测温,如果有人温度超标会进行语音提示并且保存当前人脸照片。 1、 硬件选型与设计思路 (1). 设备端 主控单片机采用STM32F103C8T6,人体测温功能采用非接触式红外测温模块。 (2). 上位机设计思路 上位机采用Qt5设计,Qt5是一套基于C++语言的跨平台软件库,性能非常强大,目前桌面端很多主流的软件都是采用QT开发。比如: 金山办公旗下的-WPS,字节
[单片机]
基于<font color='red'>STM32</font>单片机设计的红外测温仪(带人脸检测)
Arduino_Core_STM32---pinMode()实现分析
pinMode()定义 Arduino平台的易于使用性主要就体现在屏蔽的大量底层细节的实现,对于该函数来说也不例外。虽然该函数只有两个参数(arduino引脚号和模式),但这两个参数需要多层的映射才能转化为具体适合STM32芯片的配置信息,并调用底层接口完成配置。 从下面源码中可以看出主要由两个函数来完成引脚模式配置的:digitalPinToPinName()和pinfunction()。在具体深入下面两个函数之前我们需要一些预备知识—该库对于STM32硬件端口和引脚的封装。 void pinMode(uint32_t ulPin, uint32_t ulMode) { PinName p = digitalPinTo
[单片机]
STM32 USB HID设置(STM32CubeMX)
STM32F070F6P USB HID设置 1、打开STM32CubeMX软件,选择“NEW”新建一个工程 2、选择芯片型号STM32F070F6P 3、在Swap PA9/10- PA11/12前打勾 4、在“USB”下的“Device(FS)”前打勾 5、设置晶振,在“RCC”下的“High Speed Clock(HSE)”下选择第三项(此芯片需外接晶振) 6、在MiddleWares下的USB_DEVICE选择“Custon Human Interface Device Class(HID)”选项 7、点击Clock Configuration 标签,在弹出的对话框点YES.自动配置时
[单片机]
<font color='red'>STM32</font> USB HID设置(STM32CubeMX)
STM32速成笔记(8)—DMA
一、什么是DMA DMA全程Direct Memory Access,即直接存储器访问。简单来讲,它的功能是把数据从一个地址搬运到另一个地址。通常有三个传输方向,分别是内存到内存,内存到外设和外设到内存。 DMA示意图 二、DMA有什么作用 直接存储器存取(DMA)用来提供在外设和存储器之间或者存储器和存储器之间的高速数据传输。无须CPU干预,数据可以通过DMA快速地移动,这就节省了CPU的资源来做其他操作。 比如在串口接收或者发送时可以直接利用DMA将接收内容直接搬运到接收数组。或者利用DMA将准备发送的数据搬运到发送的缓冲区。再或者利用DMA把数据搬运到特定的地址,或者从特定的地址利用DMA搬运数据出来。总而言之,在平时的
[单片机]
<font color='red'>STM32</font>速成笔记(8)—<font color='red'>DMA</font>
STM32的IO的配置和模式
STM32的IO,配置的时候,首先要打开时钟,即RCC。配置寄存器,可以通过库,也可以直接控制寄存器。他的寄存器有好几个,但基本上常用的就几个,GPIOx- CRL,GPIO- CRH,GPIOx- IDR,GPIOx- ODR. CRL配置低8位IO,CRH配置高8位IO 配置时,需要配置模式与速度。 模式有:普通推挽输出,普通开漏输出,复用推挽输出,复用开漏输出,模拟输入,浮空输入,上下拉输入(需要GPIOx- ODR配合)。 其中复用的意思是带功能的,比如串口输出,即要使用复用输出,这时,IO口,会被串口模块控制。 如果要开复用的话,需要在RCC中,打开复用时钟。 速度主要是说输出,有:2M,10M,50M三种极限频率,据说配
[单片机]
小广播
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
随便看看

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved