STM32 使用DMA+DAC+TIMER 输出正弦波

之前已经简单论述过,根据我个人菜鸟的了解与认识,对之前的知识进行整理回顾:

DMA:我的理解就是一个通道,或者是一座桥梁。在静态内存到静态内存,或者外设到静态内存间的一个通讯的通道。建立这个通道的好处是:可以抛开CPU,不占用CPU的资源,直接使用这块内存的内容,速度也会加快。

DAC:STM32F103中有两个DAC,可以同时使用。DAC的作用就是将数字量转化为模拟量(电压),在这就不作太多的讲解。

TIMER:定时器。不作讲解。

那么对于使用DMA+DAC+TIMER产生正弦波的原理或过程,我有这样一个简单的理解:

  先将一个可以生成正弦波的数据表保存在静态内存中,然后在DAC以及这块内存中间使用DMA建立一个通道,经过以上步骤之后,DAC模块就可以通过DAM通道拿取静态内存中可以生成正弦波的数据,拿取数据,然后经过数模准换,在引脚进行输出就可以得到正弦波了。那么当然,这个速度是非常快的,如果没有一定的延时,那么得到的估计就是一个变化很快的模拟量。所以这个时候就需要使用定时器TIMER了。DAC在初始化的时候,可以设置成使用定时器触发,这就意味着,当定时器溢满的时候,就会触发DAC工作。这样一来,就可以通过改变定时器的定时时间来改变正弦波的周期了。

以上是我的一个简单的了解,应该会有很多不严谨不正确的地方,毕竟是一个新手菜鸟,以上见解也是方便自己学习,本人也会根据不断学习进行补充营养的。下面贴出一个例子进行分析:

1、初始化波形表以及输出的引脚

/********正弦波输出表***********/
void SineWave_Data( u16 cycle ,u16 *D)
{
	u16 i;
	for( i=0;i<cycle;i++)
	{
		D[i]=(u16)((Um*sin(( 1.0*i/(cycle-1))*2*PI)+Um)*4095/3.3);
	}
}

/******************正弦波形表***********************/
#ifdef  Sine_WaveOutput_Enable
     u16 SineWave_Value[256];		//用函数封装
#endif

/******DAC寄存器地址声明*******/
#define DAC_DHR12R1    (u32)&(DAC->DHR12R1)   //DAC通道1输出地址
#define DAC_DHR12R2    (u32)&(DAC->DHR12R2)   //DAC通道2输出地址

/****************初始化引脚******************/
void SineWave_GPIO_Config(void)
{
	GPIO_InitTypeDef GPIO_InitStructure;
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);  //开时钟
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;       //推挽输出模式
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;	//输出速率
        GPIO_InitStructure.GPIO_Pin =  GPIO_Pin_5 ; //选择引脚
	GPIO_SetBits(GPIOA,GPIO_Pin_5)	;	//拉高输出
	GPIO_Init(GPIOA, &GPIO_InitStructure);		//初始化
}

 2、初始化DAC

/******************DAC初始化ˉ*************************/
void SineWave_DAC_Config( void)
{
    DAC_InitTypeDef            DAC_InitStructure;
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_DAC, ENABLE);//开DAC时钟

  /**************DAC结构初始化*******************/
    DAC_StructInit(&DAC_InitStructure);
    DAC_InitStructure.DAC_WaveGeneration = DAC_WaveGeneration_None;//不产生波形
    DAC_InitStructure.DAC_OutputBuffer = DAC_OutputBuffer_Disable; //不使能输出缓存
    DAC_InitStructure.DAC_Trigger = DAC_Trigger_T2_TRGO;//DAC触发为定时器2触发
    DAC_Init(DAC_Channel_1, &DAC_InitStructure);//初始化
    DAC_Cmd(DAC_Channel_1, ENABLE);	   //使能DAC的通道1
    DAC_DMACmd(DAC_Channel_1, ENABLE); //使能DAC通道1的DMA
}

  3、定时器配置

/*********定时器初始化************/
void SineWave_TIM_Config(u32 Wave1_Fre)
{
    TIM_TimeBaseInitTypeDef    TIM_TimeBaseStructure;
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);//开时钟
    TIM_TimeBaseStructInit(&TIM_TimeBaseStructure);
    TIM_TimeBaseStructure.TIM_Prescaler = 0x0;     //不预分频
    TIM_TimeBaseStructure.TIM_ClockDivision = 0x0; //不分频  TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;//向上计数
    TIM_TimeBaseStructure.TIM_Period = Wave1_Fre;//设置输出频率
    TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);
    TIM_SelectOutputTrigger(TIM2, TIM_TRGOSource_Update);//设置TIME输出触发为更新模式
}

 4、DMA配置 

/*********DMA配置***********/
void SineWave_DMA_Config(void)
{
	DMA_InitTypeDef            DMA_InitStructure;
	RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA2, ENABLE);//开启DMA2时钟

	DMA_StructInit( &DMA_InitStructure);		//DMA结构体初始化
	DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralDST;//从寄存器读数据
	DMA_InitStructure.DMA_BufferSize = 256;//寄存器大小
	DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;//外设地址不递增
	DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;	//内存地址递增
	DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;//宽度为半字
	DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;//宽度为半字
	DMA_InitStructure.DMA_Priority = DMA_Priority_VeryHigh;/优先级非常高
	DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;//关闭内存到内存模式
	DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;//循环发送模式

     DMA_InitStructure.DMA_PeripheralBaseAddr = DAC_DHR12R1;//外设地址为DAC通道1的地址
     DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)SineWave_Value;//波形数据表内存地址

     DMA_Init(DMA2_Channel3, &DMA_InitStructure);//初始化
     DMA_Cmd(DMA2_Channel3, ENABLE); //使能DMA通道3    

}

 5、正弦波初始化

 

void SineWave_Init(u16 Wave1_Fre)
{
  u16 f1=(u16)(72000000/sizeof(SineWave_Value)*2/Wave1_Fre);//计算频率
   SineWave_Data( 256 ,SineWave_Value);		//生成输出正弦波的波形表
  SineWave_GPIO_Config();			  //初始化io
  SineWave_TIM_Config(f1);			  //初始化定时器
  SineWave_DAC_Config();			  //配置DAC
  SineWave_DMA_Config();			  //配置DMA
  TIM_Cmd(TIM2, ENABLE);			 //开启定时器
}

  经过以上的简单配置,就可以使得32板输出sin波形了。

时间: 2024-07-29 17:56:33

STM32 使用DMA+DAC+TIMER 输出正弦波的相关文章

STM32 基DMA的DAC波形发生器

DAC是STM32系列的一个基本外设,可以将数字信号转化成模拟信号,这次我将使用DAC来输出一个特定波形. 首先确定工作方法,由于我目前在做的简易示波器在输出波形的同时还需要显示输入信号,所以不能占用太多CPU时间,于是就选用了基于DMA的ADC. 使用DMA只需告诉DMA外设它要怎么搬移数据就可以处理其他事. 首先定义一下 #define DAC_DHR12R1    (u32)&(DAC->DHR12R1)   //DAC DATA buff 作为DMA的外设数据地址 首先是初始化输出管

STM32外设DMA使用总结

STM32外设DMA使用总结: 1.根据需要选择DAM模式: (1)循环模式-DMA_Mode =  DMA_Mode_Circular (2)正常模式-DMA_Mode =  DMA_Mode_Normal 2.对于DMA1的Chanel3,对应外设为USART3的RX 试想:如果串口接收中断和DAM中断同时打开,CPU如何相应? (1)中断优先级不同:这好说,支持嵌套中断(NVIC)的Cortex-M3自然优先服务中断优先级高的 (2)中断优先级相同:处理原则,先来先处理:若同时到来,中断号

STM32之DMA+ADC

借用小甲鱼的经典:各位互联网的广大网友们.大家早上中午晚上好..(打下小广告,因为小甲鱼的视频真的很不错).每次看小甲鱼的视频自学都是比较轻松愉快的..我在想,如果小甲鱼出STM32的视频,我会一集不漏的听的.哈.好了..学习到了STM32的DMA模块..琢磨了一下中文参考手册,官方是这样描述的: 直接存储器存取(DMA)用来提供在外设和存储器之间或者存储器和存储器之间的高速数据传输.无须CPU干预,数据可以通过DMA快速地移动,这就节省了CPU的资源来做其他操作. 是的,无需CPU干预,可以想

STM32之DMA

一.DMA简介 1.DMA简介 DMA(Direct Memory Access:直接内存存取)是一种可以大大减轻CPU工作量的数据转移方式. CPU有转移数据.计算.控制程序转移等很多功能,但其实转移数据(尤其是转移大量数据)是可以不需要CPU参与.比如希望外设A的数据拷贝到外设B,只要给两种外设提供一条数据通路,再加上一些控制转移的部件就可以完成数据的拷贝. DMA就是基于以上设想设计的,它的作用就是解决大量数据转移过度消耗CPU资源的问题.有了DMA使CPU更专注于更加实用的操作--计算.

STM32学习笔记——DAC

1.概述STM32DAC 2.STM32F407库函数总结 3.输出模拟电压配置过程 --------------------------------------------------------------------------------------------------------------------- -----------------------------------------------------------------------------------------

STM32 UART DMA实现未知数据长度接收

串口通信是经常使用到的功能,在STM32中UART具有DMA功能,并且收发都可以使用DMA,使用DMA发送基本上大家不会遇到什么问题,因为发送的时候会告知DMA发送的数据长度,DMA按照发送的长度直接发送就OK了,但是使用DMA接收时候就不同了,因为有时候数据接收并不是每一次都是定长的,但是DMA只在接收数据长度和设定数据长度相同的时候才可以触发中断,告诉MCU数据接收完毕,针对这个问题,解决方法如下,有一点复杂,但是很管用. UART在传输一个字节的时候,首先拉低,传输起始位,然后在是LSB

STM32 ADC DMA 中断模式多通道读取ADC转换值

因为做一个电机控制项目,其中需要用到用PWM触发ADC,读取ADC的转换值. 这里有两个关键问题: 第一,如何使PWM触发ADC转换? 第二,如何在ADC多通道扫描模式下,读取各个channel的adc转换值? 第三,如何使用DMA读取多通道ADC转换值? 第一个问题,首先需要将ADC设置成外部触发转换,第二需要选择触发是发生在Timer的上升沿还是下降沿.程序如下: adc.ADC_Resolution = ADC_Resolution_12b; adc.ADC_ScanConvMode =E

STM32使用TIM闪烁LED——输出比较方式

STM32定时器输出比较器可以直接操作对应的GPIO,在计数器值等于比较寄存器的值时,对应的GPIO可以有以下四种动作 无动作 激活 取消激活 翻转 激活电平由输出极性寄存器决定 将LED置于定时器输出比较对应的GPIO上,对它的操作将非常方便,完全由硬件完成,不消耗CPU时间,无需中断,每个定时器都对应有4个输出比较,可以轻松实现4个LED的流水灯效果 宏定义部分如下 #define USER_TIM_PSC 36000-1 #define USER_TIM_PERIOD 1000-1 #de

stm32 spi dma tft lcd

http://habrahabr.ru/post/139384 http://electronics.stackexchange.com/questions/100685/full-duplex-slave-spi-dma-and-interrupts-on-stm32f103 https://my.st.com/public/STe2ecommunities/mcu/Lists/cortex_mx_stm32/DispForm.aspx?ID=51426&Source=/public/STe2