STM32 八路AD转换用DMA传输调试成功,DMA传输不错位

发布者:堕落的猫最新更新时间:2016-02-25 来源: eefocus关键字:STM32  AD转换  DMA  传输调试 手机看文章 扫描二维码
随时随地手机看文章
//


//
#include"stm32f10x_conf.h"              //把DMA.h和ADC.h 的注释去掉
#include "stm32f10x.h"
#include "stm32_eval.h"

#include

#define  50      //每通道采50次
#define      //为8个通道


#define ADC1_DR_Address    ((u32)0x4001244C)

vu16  After_filter[M];    //用来存放求平均值之后的结果
vu16  AD_Value[N][M];     //用来存放ADC转换结果,也是DMA的目标地址
                          //这个变量存放ADC转换后的值,在后面DMA设置的时候取了这个变了的地址,将转

                          //换结果直接传输到这个地址。当需要多路AD转换时,定义此为一个数组后面

                          //DMA设置时取数组的基地址,依次传输

void IO_cfg(void);
void EXTI_cfg(void);
void NVIC_cfg(void);
void USART_cfg(void);
void ADC_cfg(void);
void DMA_cfg(void);

u16 GetVolt(u16 advalue);  
void filter(void);

void SerialPutChar(uint8_t c);
void SerialReceivechar(uint8_t c);

void Delay(vu32 nCount)
{
  for(; nCount != 0; nCount--);
}


void RCC_cfg()
{
       //打开PA端口时钟,并且打开复用时钟
       RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOB | RCC_APB2Periph_GPIOC | RCC_APB2Periph_AFIO, ENABLE);   
   
    RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);   //使能DMA时钟
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);   //使能ADC

}


////////////////////////////几乎没用,为了系统兼容/////////////////////////////////////
#ifdef __GNUC__
 
  #define PUTCHAR_PROTOTYPE int __io_putchar(int ch)
#else
  #define PUTCHAR_PROTOTYPE int fputc(int ch, FILE *f)
#endif
//////////////////////////////////////////////////////////////////////////////////////////////////
 
int main()
{
  int i;      //用于串口输出

 u16 value[M]; //存放求完平均值之后再将结果扩大100倍
 
 RCC_cfg();
 EXTI_cfg(); 
 IO_cfg();
 NVIC_cfg();
 USART_cfg();
 
 DMA_cfg();
 ADC_cfg();  
 
  while(1)
      
   while(USART_GetFlagStatus(USART1,USART_FLAG_TXE)==RESET);//等待传输完成否则第一位数据容易失    
   filter();
    for(i=0;i
     {
       value[i]= GetVolt(After_filter[i]);
    
       printf("value[%d]:\t%d.%dv\n",i,value[i]/100,value[i]0) ;
     Delay(0x80000);
     }

   }

}

//
void IO_cfg()
  
    //八路AD输入
     
       GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
    GPIO_Init(GPIOB, &GPIO_InitStructure);
   
     
       GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
    GPIO_Init(GPIOB, &GPIO_InitStructure);
   
      
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
    GPIO_Init(GPIOC, &GPIO_InitStructure);
     
    
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
    GPIO_Init(GPIOC, &GPIO_InitStructure);

      
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
    GPIO_Init(GPIOC, &GPIO_InitStructure);

      
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
    GPIO_Init(GPIOC, &GPIO_InitStructure);

      
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
    GPIO_Init(GPIOC, &GPIO_InitStructure);

      
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
    GPIO_Init(GPIOC, &GPIO_InitStructure);
      
}

 // 配置串口参数  
 
void USART_cfg()
{
 USART_InitTypeDef USART_InitStructure;

 USART_InitStructure.USART_BaudRate = 115200;
 USART_InitStructure.USART_WordLength = USART_WordLength_8b;
 USART_InitStructure.USART_StopBits = USART_StopBits_1;
    USART_InitStructure.USART_Parity = USART_Parity_No;
    USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
    USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;

 STM_EVAL_COMInit(COM1, &USART_InitStructure);

}


void SerialPutChar(uint8_t c)
{
  USART_SendData(EVAL_COM1, c);
  while (USART_GetFlagStatus(EVAL_COM1, USART_FLAG_TXE) == RESET)
  {
  }
}

void SerialReceivechar(uint8_t c)
{
 
 USART_ReceiveData(EVAL_COM1);

 while( USART_GetFlagStatus(EVAL_COM1, USART_FLAG_RXNE) != RESET)
 {
 }
}

void ADC_cfg(void)
{
 ADC_InitTypeDef ADC_InitStructure;

   ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; // 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数据右对齐 ;
   //八个通道转换时下面赋值为8(相当于开启通道的数目)
   ADC_InitStructure.ADC_NbrOfChannel = M; // 顺序进行规则转换的ADC通道的数目 8 ;
   ADC_Init(ADC1, &ADC_InitStructure);
 

 //下面设置指定的ADC规则通道,一共需要设置8个

 // 设置指定ADC的规则组通道8-15,设置它们的转化顺序和采样时间为55.5周期;
   ADC_RegularChannelConfig(ADC1, ADC_Channel_8,  1, ADC_SampleTime_55Cycles5);
   ADC_RegularChannelConfig(ADC1, ADC_Channel_9,  2, ADC_SampleTime_55Cycles5);
   ADC_RegularChannelConfig(ADC1, ADC_Channel_10, 3, ADC_SampleTime_55Cycles5);
   ADC_RegularChannelConfig(ADC1, ADC_Channel_11, 4, ADC_SampleTime_55Cycles5);
   ADC_RegularChannelConfig(ADC1, ADC_Channel_12, 5, ADC_SampleTime_55Cycles5);
   ADC_RegularChannelConfig(ADC1, ADC_Channel_13, 6, ADC_SampleTime_55Cycles5);
   ADC_RegularChannelConfig(ADC1, ADC_Channel_14, 7, ADC_SampleTime_55Cycles5);
   ADC_RegularChannelConfig(ADC1, ADC_Channel_15, 8, ADC_SampleTime_55Cycles5);
 
   ADC_DMACmd(ADC1, ENABLE); // 使能指定的ADC的DMA请求 ;
   ADC_Cmd(ADC1, ENABLE); // 使能指定的ADC ;

   ADC_ResetCalibration(ADC1); // 重置指定的ADC的校准寄存器 ;
   while(ADC_GetResetCalibrationStatus(ADC1)); // 获取ADC重置校准寄存器的状态 ;

   ADC_StartCalibration(ADC1);  // 开始指定ADC的校准程序 ;
   while(ADC_GetCalibrationStatus(ADC1)); // 获取指定ADC的校准状态 ;
   ADC_SoftwareStartConvCmd(ADC1, ENABLE); // 使能指定的ADC的软件转换启动功能 ;
}

void DMA_cfg(void)
{
 DMA_InitTypeDef DMA_InitStructure;

 DMA_DeInit(DMA1_Channel1);
 DMA_InitStructure.DMA_PeripheralBaseAddr = ADC1_DR_Address;
 
 
 DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)&AD_Value; //上面这句很显然是DMA要连接在Memory

                                                            //中变量的地址,
                                                //AD_Value是我自己在memory中定义的一个变量数组;

 DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC; //这里设置的是单向传输,如果需要双向传输:
                                                    //把DMA_DIR_PeripheralSRC改成

                                                   //DMA_DIR_PeripheralDST即可。

 //转换通道为M,每组转换N次
 DMA_InitStructure.DMA_BufferSize = M*N;//上面的这句是设置DMA在传输时缓冲区的长度,前面有定义过了

                                      //buffer的起始地址:ADC1_DR_Address ,为了安全性和可靠性,一

                                      //般需要给buffer定义
                                //一个储存片区,这个参数的单位有三种类型:Byte、HalfWord、word,
                          //我设置的2个half-word(见下面的设置);32位的MCU中1个half-word占16 bits。
             
 DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;

//上面的这句是设置DMA的外设递增模式,如果DMA
              //选用的通道(CHx)有多个外设连接,需要使用外设递增模式:
                 //DMA_PeripheralInc_Enable;我的例子里DMA只与ADC1建立了联系,
                 //所以选用DMA_PeripheralInc_Disable
 DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;//上面的这句是设置DMA的内存递增模式,DMA访

                                              //问多个内存参数时,
                                  //需要使用DMA_MemoryInc_Enable,当DMA只访问一个内存参数时,
                                    //可设置成:DMA_MemoryInc_Disable。
 
 DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;

                                              //定义外设数据宽度为16位

                                              //上面的这句是设置DMA在访问时每次操作的数据长度。

                                      //有三种数据长度类型,前面已经讲过了,这里不在叙述。
  DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;//与上面雷同。在此不再说明。

 

 DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;

                                  //上面的这句是设置DMA的传输模式:连续不断的循环式,
                     //若只想访问一次后就不要访问了(或按指令操作来反问,也就是想要它访问
               //的时候就访问,不要它访问的时候就停止),可以设置成通用模式:DMA_Mode_Normal

 DMA_InitStructure.DMA_Priority = DMA_Priority_High;//上面的这句是设置DMA的优先级别:可以分为4级:

                                                    //VeryHigh,High,Medium,Low.
 
 DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;//上面的这句是设置DMA的2个memory中的变量互相访问的

 

 DMA_Init(DMA1_Channel1,&DMA_InitStructure);//前面那些都是对DMA结构体成员的设置,在次再统一对DMA整

                                            //个模块做一次初始化,
                                           //使得DMA各成员与上面的参数一致。
 
 DMA_Cmd(DMA1_Channel1,ENABLE);

///////////////////////////////////
u16 GetVolt(u16 advalue)  
{
   return (u16)(advalue * 330 / 4096);   //求的结果扩大了100倍,方便下面求出小数
}
 
//求AD转换结果的平均值函数
void filter(void)
{
 int i;     
 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;
      }
 
       

 


PUTCHAR_PROTOTYPE
{
 
 
  USART_SendData(EVAL_COM1, (uint8_t) ch);

 
  while (USART_GetFlagStatus(EVAL_COM1, USART_FLAG_TC) == RESET)
  {
  }

  return ch;
}

#ifdef  USE_FULL_ASSERT


void assert_failed(uint8_t* file, uint32_t line)
{
 

 
  while (1)
  {
  }
}

#endif  

关键字:STM32  AD转换  DMA  传输调试 引用地址:STM32 八路AD转换用DMA传输调试成功,DMA传输不错位

上一篇:STM32 通用定时器 介绍
下一篇:STM32的AD用DMA方式时的请注意初始化顺序

推荐阅读最新更新时间:2024-03-16 14:45

STM32串口通信数据乱码的相关问题
STM32串口通信以及温度采集搞定,其中主要遇到STM32系列单片机时钟树的问题,串口通信遇到串口调试助手能够接收到数据但出现乱码现象,开始一直以为是串口配置和程序代码问题,因为是第一次上电在线调试STM32板子,后面主要查串口波特率配置和收发函数程序段,如下图: 波特率设置成115200没问题,试着降低波特率改成9600和4800但问题依旧没有解决,紧接着如下处理: 将重定向函数注释,单独写串口发送字节和字符串函数,依旧失败。最后锁定到系统时钟配置上,由于手上的STM32开发板改用了12M的晶振,根据单片机时钟树的理解和解读,一般采用外部时钟HSE,系统时钟配置成72M,8*9=72,,12*6=72,对于 SYSCLK
[单片机]
<font color='red'>STM32</font>串口通信数据乱码的相关问题
基于STM32的IAP升级程序分享
因为之前写个STM32的IAP升级程序,所以我总结了做IAP升级的三个主要的难点: 1、如何设置中断向量,也就是说中断向量的重定向 2、如何配置程序的起始地址 3、如何从IAP跳转到APP程序 4、使用库函数要注意的地方(防止被坑) 说文章的时候我已经完成了一个最简单的IAP升级程序,可以通过串口接收bin文件写入到flash里面,然后再运行。 1、如何设置中断向量,也就是说中断向量的重定向 STM8不像STM32那样有个一寄存器管理着中断向量的地址,所以STM32的中断可以任意设置(符合要求的情况之下),但是STM8的中断向量表是固定的 在0X8000地址,不能修改,所以BOOT区不能开中断,否则会和APP区的中断打架,但是
[单片机]
基于<font color='red'>STM32</font>的IAP升级程序分享
stm32---SPI与外部FLASH
stm32外部flash是EN25QXX 芯片 使用 STM32F1 的 SPI 和外部 FLASH(EN25QXX)进行通信,实现数据的读写 操作。本章要实现的功能是:首先检测外部 FLASH 是否正常,然后使用 K__UP 和 K_DOWN 键控制 FLASH 的写入和读取,并将 数据显示在 串口助手上,同时控制 D1 指示灯不断闪烁,提示系统正常 运行 W25Q128 将 16M 的容量分为 256 个块( Block),每个块大小为 64K 字 节,每个块又分为 16 个扇区( Sector),每个扇区 4K 个字节。 W25Q128 的最小擦除单位为一个扇区,也就是每次必须擦除 4K 个字节。这样我们需要给 W2
[单片机]
STM32 基础系列教程 1- CubeMX+GPIO
前言 学习stm32 GPIO 的使用,设置某一GPIO引脚为输出功能,将对应引脚拉高或拉低输出,同时学会初步认识STM32最新的HAL库的使用, 用代码实现控制GPIO引脚输出产生周期出1s 占空比为50%的PWM波。 示例详解 基于硬件平台: STM32F10C8T6最小系统板, MCU 的型号是 STM32F103c8t6, 使用stm32cubemx 工具自动产生的配置工程,使用KEIL5编译代码。 本示例所用的最小系统板原理图: 新建STM32 CUBEMX 工程, 双击桌面STM32CubeMX工具 在CubeMX中菜单中点 File à New Project … 在新弹出的界面
[单片机]
<font color='red'>STM32</font> 基础系列教程 1- CubeMX+GPIO
usmart_dev.init(SystemCoreClock/1000000); stm32 战舰的usmart 程序
usmart_dev.init(SystemCoreClock/1000000); //初始化USMART 对初始化程序的解析。对应 stm32 战舰的usmart 程序来分析 usmart_dev是_m_usmart_dev结构体的变量, usmart_dev结构体变量的初始化如下 struct _m_usmart_dev usmart_dev= { usmart_nametab, usmart_init, usmart_cmd_rec, usmart_exe, usmart_scan, sizeof(usmart_nametab)/sizeof(struct _m_usmart_nametab),//函数数量
[单片机]
无线传感器网络底层平台的深层研究
引言 目前,对WSN(Wireless Sensor Netwoek)的研究主要集中在协议栈、定位算法、能耗管理以及体系结构设计上,而针对无线传感网络操作系统的研究却相对较少,尤其是对其底层平台的研究更少,所以针对无线传感网络操作系统底层平台的研究有十分广阔的空间。本论文针对意法半导体STM32系列MCU和TI公司的CC2520无线模块进行介绍。主要描述了操作系统底层平台的构建,以及硬件驱动程序的实现。本论文的底层硬件抽象层是针对CC2520无线射频模块的,包括了平台构建、相关寄存器,以及外围接口等各个部分。而硬件驱动程序主要是为完成无线通信所需的硬件驱动设计,这主要包括:相应异步事件的中断机制;和PC通信的USART驱动;连
[单片机]
无线传感器网络底层平台的深层研究
STM32学习之串口
第一步:把串口用的引脚设置。接收的为 GPIO_Mode_IN_FLOATING; //浮空输入 发送的为GPIO_Mode_AF_PP; // 复用推挽输出 第二部设置 void NVIC_Configuration(void) NVIC_InitTypeDef NVIC_InitStructure; NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0); NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQChannel ; // 全局中断、、 NVIC_InitStructure.NVIC_IRQChannelPreemptio
[单片机]
STM32开发者社区:从这里开启你的STM32之旅!小白和PRO都友好
当面对STM32Cube生态系统这样一个庞大而丰富的开发世界时,工程师难免会产生疑问,从哪里开始才好? ST的许多合作伙伴和客户都希望有更多的产品能够利用STM32Cube开发环境。开发人员很享受开发环境的图形用户界面和工具的易用性,如STM32CubeMX、免费的STM32CubeIDE以及许多软件包、驱动程序和中间件,这些都有助于更快地将产品推向市场。随着越来越多的企业选择ST的产品,越来越多的工程师在ST的生态系统中迈出了第一步。为了降低开发人员的进入门槛,ST推出了STM32开发者社区。开发者社区如何为开发团队提供帮助,ST如何将STM32生态系统整合在一起?让我们详细聊聊。 STM32开发者社区 为开
[单片机]
<font color='red'>STM32</font>开发者社区:从这里开启你的<font color='red'>STM32</font>之旅!小白和PRO都友好
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

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

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

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