当前位置:首页 > 单片机 > 单片机
[导读]之前已经简单论述过,根据我个人菜鸟的了解与认识,对之前的知识进行整理回顾:DMA:我的理解就是一个通道,或者是一座桥梁。在静态内存到静态内存,或者外设到静态内存间的一个通讯的通道。建立这个通道的好处是:可以

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

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

{

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波形了。

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

在嵌入式开发中,STM32的时钟系统因其灵活性和复杂性成为开发者关注的焦点。然而,看似简单的时钟配置背后,隐藏着诸多易被忽视的陷阱,轻则导致系统不稳定,重则引发硬件损坏。本文从时钟源选择、PLL配置、总线时钟分配等关键环...

关键字: STM32 时钟系统

在嵌入式系统开发中,STM32系列微控制器的内部温度传感器因其低成本、高集成度特性,广泛应用于设备自检、环境监测等场景。然而,受芯片工艺差异和电源噪声影响,其原始数据存在±1.5℃的固有误差。本文从硬件配置、校准算法、软...

关键字: STM32 温度传感器

在能源效率与智能化需求双重驱动下,AC-DC转换器的数字控制技术正经历从传统模拟方案向全数字架构的深刻变革。基于STM32微控制器的PFM(脉冲频率调制)+PWM(脉冲宽度调制)混合调制策略,结合动态电压调整(Dynam...

关键字: AC-DC STM32

当前智能家居产品需求不断增长 ,在这一背景下 ,对现有浇花装置缺陷进行了改进 ,设计出基于STM32单片机的全 自动家用浇花机器人。该设计主要由机械结构和控制系统构成 ,机械结构通过麦克纳姆轮底盘与喷洒装置的结合实现机器...

关键字: STM32 麦克纳姆轮 安全可靠 通过性强

用c++编程似乎是让你的Arduino项目起步的障碍吗?您想要一种更直观的微控制器编程方式吗?那你需要了解一下Visuino!这个图形化编程平台将复杂电子项目的创建变成了拖动和连接块的简单任务。在本文中,我们将带您完成使...

关键字: Visuino Arduino ESP32 STM32

基于STM32与LoRa技术的无线传感网络凭借其低功耗、广覆盖、抗干扰等特性,成为环境监测、工业自动化等场景的核心解决方案。然而,如何在复杂电磁环境中实现高效休眠调度与动态信道优化,成为提升网络能效与可靠性的关键挑战。本...

关键字: STM32 LoRa

在实时控制系统、高速通信协议处理及高精度数据采集等对时间敏感的应用场景中,中断响应延迟的优化直接决定了系统的可靠性与性能上限。STM32系列微控制器凭借其灵活的嵌套向量中断控制器(NVIC)、多通道直接内存访问(DMA)...

关键字: STM32 DMA

数字电源技术向高功率密度、高效率与高动态响应方向加速演进,STM32微控制器凭借其基于DSP库的算法加速能力与对LLC谐振变换器的精准控制架构,成为优化电源动态性能的核心平台。相较于传统模拟控制或通用型数字控制器,STM...

关键字: STM32 数字电源

STM32微控制器凭借其针对电机控制场景的深度优化,成为高精度、高可靠性驱动系统的核心选择。相较于通用型MCU,STM32在电机控制领域的核心优势集中体现在FOC(磁场定向控制)算法的硬件加速引擎与PWM死区时间的动态补...

关键字: STM32 电机控制

无线充电技术加速渗透消费电子与汽车电子领域,基于Qi协议的无线充电发射端开发成为智能设备能量补给的核心课题。传统模拟控制方案存在响应滞后、参数调整困难等问题,而基于STM32的数字PID控制结合FOD(Foreign O...

关键字: STM32 无线充电
关闭