当前位置:   article > 正文

stm32输出正弦波

stm32输出正弦波

本次说明是看野火的开发板学习,做个笔记

 

计算获取正弦波数据表;

2) 根据正弦波数据表的周期内点数和周期计算定时器触发间隔;

3) 初始化 DAC 输出通道,初始化 DAC 工作模式;

4) 配置触发 DAC 用的定时器;

5) 配置 DMA 自动转运正弦波数据表。

配置完成后,即可在 PA4PA5 引脚中检测到信号输出。

首先

生成正弦波数据表

要输出正弦波,实质是要控制 DAC v=sin(t)的正弦函数关系输出电压,其中 v 为电

压输出,t 为时间。

而由于模拟信号连续而数字信号是离散的,所以使用 DAC 产生正弦波时,只能按一

定时间间隔输出正弦曲线上的点,在该时间段内输出相同的电压值,若缩短时间间隔,提

高单个周期内的输出点数,可以得到逼近连续正弦波的图形,见图 39-4,若在外部电路加

上适当的电容滤波,可得到更完美的图形。

有下图可以看出,其输出的波形不太正,由于取得样太少的原因和没有加电容器铝板的原因,

 

抬升 sin 函数的输出为正值:v = sin(t)+1 ,此时,v 的输出范围为[0:2]

2) 扩展输出至 DAC 的全电压范围: v = 3.3*(sin(t)+1)/2 ,此时,v 的输出范围为[0:3.3]

正是 DAC 的电压输出范围,扩展至全电压范围可以充分利用 DAC 的分辨率;

3) 把电压值以 DAC 寄存器的形式表示:Reg_val = 2

12

/3.3 * v = 2

11

*(sin(t)+1),此时,存

储到 DAC 寄存器的值范围为[0:4096]

4) 实践证明,在 sin(t)的单个周期内,取 32 个点进行电压输出已经能较好地还原正弦波

形,所以在 t[0:2π]区间内等间距根据上述 Reg_val 公式运算得到 32 个寄存器值,

即可得到正弦波表;

5) 控制 DAC 输出时,每隔一段相同的时间从上述正弦波表中取出一个新数据进行输出,

即可输出正弦波。改变间隔时间的单位长度,可以改变正弦波曲线的周期。

 

注意GPIO的配置

GPIO 按照要求被配置为模拟输入模式

  1. static void DAC_Config(void)
  2. {
  3. GPIO_InitTypeDef GPIO_InitStructure;
  4. DAC_InitTypeDef DAC_InitStructure;
  5. /* 使能GPIOA时钟 */
  6. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
  7. /* 使能DAC时钟 */
  8. RCC_APB1PeriphClockCmd(RCC_APB1Periph_DAC, ENABLE);
  9. /* DAC的GPIO配置,模拟输入 */
  10. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4 | GPIO_Pin_5;
  11. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
  12. GPIO_Init(GPIOA, &GPIO_InitStructure);
  13. /* 配置DAC 通道1 */
  14. DAC_InitStructure.DAC_Trigger = DAC_Trigger_T2_TRGO; //使用TIM2作为触发源
  15. DAC_InitStructure.DAC_WaveGeneration = DAC_WaveGeneration_None; //不使用波形发生器
  16. DAC_InitStructure.DAC_OutputBuffer = DAC_OutputBuffer_Disable; //不使用DAC输出缓冲
  17. DAC_Init(DAC_Channel_1, &DAC_InitStructure);
  18. /* 配置DAC 通道2 */
  19. DAC_Init(DAC_Channel_2, &DAC_InitStructure);
  20. /* 使能通道1 由PA4输出 */
  21. DAC_Cmd(DAC_Channel_1, ENABLE);
  22. /* 使能通道2 由PA5输出 */
  23. DAC_Cmd(DAC_Channel_2, ENABLE);
  24. /* 使能DAC的DMA请求 */
  25. DAC_DMACmd(DAC_Channel_2, ENABLE);
  26. }

 

 

在 上 述代 码中 , 定义了 由 脚本 得到 的 正弦波 数 据表 Sine12bit 变量 , 一共 为

POINT_NUM32)个点。在 DAC_Mode_Init 函数中,调用了前面介绍的 DAC_Config

DAC_TIM_Config 初始化 DAC 和定时器,然后在 for 循环中把单通道的正弦波数据表

Sine12bit 复制扩展成为双通道的数据 DualSine12bit,扩展后的数据将会直接被 DMA 搬运

DAC DHR12RD 寄存器中。

复制完数据后,DAC_Mode_Init 调用下面的 DAC_DMA_Config 函数初始化 DMA,配

置的重点是要设置好 DHR12RD 寄存器的地址,正弦波数据的内存地址(注意是双通道数

DualSine12bit),DMA 缓存的个数(即单个周期的正弦波点数)以及 DMA 工作在循环

模式。

 

  1. /**
  2. ******************************************************************************
  3. * @file bsp_xxx.c
  4. * @author fire
  5. * @version V1.0
  6. * @date 2013-xx-xx
  7. * @brief adc1 应用bsp / DMA 模式
  8. ******************************************************************************
  9. * @attention
  10. *
  11. * 实验平台:野火STM32 霸道 开发板
  12. * 论坛 :http://www.firebbs.cn
  13. * 淘宝 :http://fire-stm32.taobao.com
  14. *
  15. ******************************************************************************
  16. */
  17. #include "./dac/bsp_dac.h"
  18. //正弦波单个周期的点数
  19. #define POINT_NUM 32
  20. /* 波形数据 ---------------------------------------------------------*/
  21. const uint16_t Sine12bit[POINT_NUM] = {
  22. 2048 , 2460 , 2856 , 3218 , 3532 , 3786 , 3969 , 4072 ,
  23. 4093 , 4031 , 3887 , 3668 , 3382 , 3042 , 2661 , 2255 ,
  24. 1841 , 1435 , 1054 , 714 , 428 , 209 , 65 , 3 ,
  25. 24 , 127 , 310 , 564 , 878 , 1240 , 1636 , 2048
  26. };
  27. uint32_t DualSine12bit[POINT_NUM];
  28. /**
  29. * @brief 使能DAC的时钟,初始化GPIO
  30. * @param 无
  31. * @retval 无
  32. */
  33. static void DAC_Config(void)
  34. {
  35. GPIO_InitTypeDef GPIO_InitStructure;
  36. DAC_InitTypeDef DAC_InitStructure;
  37. /* 使能GPIOA时钟 */
  38. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
  39. /* 使能DAC时钟 */
  40. RCC_APB1PeriphClockCmd(RCC_APB1Periph_DAC, ENABLE);
  41. /* DAC的GPIO配置,模拟输入 */
  42. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4 | GPIO_Pin_5;
  43. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
  44. GPIO_Init(GPIOA, &GPIO_InitStructure);
  45. /* 配置DAC 通道1 */
  46. DAC_InitStructure.DAC_Trigger = DAC_Trigger_T2_TRGO; //使用TIM2作为触发源
  47. DAC_InitStructure.DAC_WaveGeneration = DAC_WaveGeneration_None; //不使用波形发生器
  48. DAC_InitStructure.DAC_OutputBuffer = DAC_OutputBuffer_Disable; //不使用DAC输出缓冲
  49. DAC_Init(DAC_Channel_1, &DAC_InitStructure);
  50. /* 配置DAC 通道2 */
  51. DAC_Init(DAC_Channel_2, &DAC_InitStructure);
  52. /* 使能通道1 由PA4输出 */
  53. DAC_Cmd(DAC_Channel_1, ENABLE);
  54. /* 使能通道2 由PA5输出 */
  55. DAC_Cmd(DAC_Channel_2, ENABLE);
  56. /* 使能DAC的DMA请求 */
  57. DAC_DMACmd(DAC_Channel_2, ENABLE);
  58. }
  59. /**
  60. * @brief 配置TIM
  61. * @param 无
  62. * @retval 无
  63. */
  64. static void DAC_TIM_Config(void)
  65. {
  66. TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
  67. /* 使能TIM2时钟,TIM2CLK 为72M */
  68. RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);
  69. /* TIM2基本定时器配置 */
  70. // TIM_TimeBaseStructInit(&TIM_TimeBaseStructure);
  71. TIM_TimeBaseStructure.TIM_Period = (20-1); //定时周期 20
  72. TIM_TimeBaseStructure.TIM_Prescaler = 0x0; //预分频,不分频 72M / (0+1) = 72M
  73. TIM_TimeBaseStructure.TIM_ClockDivision = 0x0; //时钟分频系数
  74. TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //向上计数模式
  75. TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);
  76. /* 配置TIM2触发源 */
  77. TIM_SelectOutputTrigger(TIM2, TIM_TRGOSource_Update);
  78. /* 使能TIM2 */
  79. TIM_Cmd(TIM2, ENABLE);
  80. }
  81. /**
  82. * @brief 配置DMA
  83. * @param 无
  84. * @retval 无
  85. */
  86. static void DAC_DMA_Config(void)
  87. {
  88. DMA_InitTypeDef DMA_InitStructure;
  89. /* 使能DMA2时钟 */
  90. RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA2, ENABLE);
  91. /* 配置DMA2 */
  92. DMA_InitStructure.DMA_PeripheralBaseAddr = DAC_DHR12RD_ADDRESS; //外设数据地址
  93. DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)&DualSine12bit ; //内存数据地址 DualSine12bit
  94. DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralDST; //数据传输方向内存至外设
  95. DMA_InitStructure.DMA_BufferSize = POINT_NUM; //缓存大小为POINT_NUM字节
  96. DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; //外设数据地址固定
  97. DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; //内存数据地址自增
  98. DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Word; //外设数据以字为单位
  99. DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Word; //内存数据以字为单位
  100. DMA_InitStructure.DMA_Mode = DMA_Mode_Circular; //循环模式
  101. DMA_InitStructure.DMA_Priority = DMA_Priority_High; //高DMA通道优先级
  102. DMA_InitStructure.DMA_M2M = DMA_M2M_Disable; //非内存至内存模式
  103. DMA_Init(DMA2_Channel4, &DMA_InitStructure);
  104. /* 使能DMA2-14通道 */
  105. DMA_Cmd(DMA2_Channel4, ENABLE);
  106. }
  107. /**
  108. * @brief DAC初始化函数
  109. * @param 无
  110. * @retval 无
  111. */
  112. void DAC_Mode_Init(void)
  113. {
  114. uint32_t Idx = 0;
  115. DAC_Config();
  116. DAC_TIM_Config();
  117. /* 填充正弦波形数据,双通道右对齐*/
  118. for (Idx = 0; Idx < POINT_NUM; Idx++)
  119. {
  120. DualSine12bit[Idx] = (Sine12bit[Idx] << 16) + (Sine12bit[Idx]);
  121. }
  122. DAC_DMA_Config();
  123. }

经过这样的配置后,定时器每间隔一定的时间就会触发 DMA 搬运双通道正弦波表的

一个数据到 DAC 双通道寄存器进行转换,每完成一个周期后 DMA 重新开始循环,从而达

到连续输出波形的目的。

 

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/IT小白/article/detail/454637
推荐阅读
相关标签
  

闽ICP备14008679号