历史上的今天

今天是:2024年09月27日(星期五)

正在发生

2021年09月27日 | stm32之定时器运用———呼吸灯

发布者:RadiantGaze 来源: eefocus关键字:stm32  呼吸灯 手机看文章 扫描二维码
随时随地手机看文章

呼吸灯原理

1.在模拟电路中,呼吸灯的实现可以通过一个呈现正弦的电压控制,这个电压是连续变化的,所以肉眼看上去就是逐渐变暗,逐渐变亮。


2.而在数字电路中如何实现这种效果呢?就需要通过pwm,也就是脉冲宽度调制,将模拟量转换为数字量。只要能够用连续电压控制的东西都是可以通过pwm方式来驱动,效果是一样的。


3.

这里写图片描述

上面一块区域的面积等于对应下来的矩形的面积,当然,取得块的间隔越小(即pwm的周期越小),效果越好。这时,如果周期定了,就可以通过改变占空比来实现面积的改变,从而模拟出上面那张图的电压的连续变化。


注意:pwm波的高度是一定的,所以只能通过改变宽度(占空比)来实现面积的改变


4.说明:观察这张图,会发现下面的pwm波是中心与上面的对齐(即pwm中心为高电平),然后左右扩展,每个波的周期还是一样。当然这时可以的。但更多的是运用左对齐(起始为高电平),然后向右扩展直到面积到达要求。


5.stm32实现pwm输出的原理:设点一个值为a,然后在设置一个重装值b,b>a.开始计数,当计数值小于a时,输出高电平,当计数值大于a时,输出低电平,直到计数到b,到b后又重复来一遍。所以改变这个a就可以改变占空比、


6.PWM 的输出其实就是对外输出脉宽可调(即占空比调节)的方波信号,信号频率是由自动重装寄存器 ARR 的值决定,占空比由比较寄存器 CCR 的值决定。其示意图如图 19.1.2 所示:

这里写图片描述

从图 19.1.2 中可以看到,PWM 输出频率是不变的,改变的是 CCR 寄存器内的值,此值的改变将导致 PWM 输出信号占空比的改变。占空比其实就是一个周期内高电平时间与周期的比值。PWM 输出比较模式总共有 8 种,具体由寄存器 CCMRx 的位 OCxM[2:0]配置。我们这里只讲解最常用的两种 PWM 输出模式:PWM1 和 PWM2,PWM1 和 PWM2 这两种模式用法差不多,区别之处就是输出电平的极性不同。如图 19.1.3 所示:

这里写图片描述

pwm输出配置步骤

其实 PWM 输出和上一章一样也是通用定时器的一个功能,因此还是要用到定时器的相关配置函数


1.因为pwm是由定时器输出的,既然用到定时器,就先要使能定时器的时钟:


RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM14,ENABLE);//我们用的为tim14定时器


2.因为用到io作为输出,所以要打开io口的时钟:


RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOF,ENABLE);//注意:io口的外设均是挂在AHB1总线上的


3.通过看手册,TIM14 的 CH1 通道对应的管脚是 PF9,而pf9有很多复用功能,所以要选择pf9的输出模式:通过函数:


void GPIO_PinAFConfig(GPIO_TypeDef* GPIOx, uint16_t GPIO_PinSource,

uint8_t GPIO_AF);//前两个参数不说了。第三个参数为复用为哪种功能,这里我们使用的是 TIM14 功能,所以参数为 GPIO_AF_TIM14


所以函数为:GPIO_PinAFConfig(GPIOF,GPIO_PinSource9,GPIO_AF_TIM14);//AF就是复用的意思


4.配置io口,同之前led时一样,只不过参数有些变化:


      GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AF;//PF9 管脚模式配置为复用输出

      GPIO_InitStructure.GPIO_Pin=GPIO_Pin_9;

      GPIO_InitStructure.GPIO_Speed=GPIO_Speed_100MHz;//速度不变

      GPIO_InitStructure.GPIO_OType=GPIO_OType_PP;//推完输出不变

      GPIO_InitStructure.GPIO_PuPd=GPIO_PuPd_UP;//上啦输出不变

      GPIO_Init(GPIOF,&GPIO_InitStructure);


5.初始化定时器参数,包含自动重装值,分频系数,计数方式等.同前面使用定时器中断


      TIM_TimeBaseInitStructure.TIM_Period = pre;//预装值,这里依然通过参数传递进来

      TIM_TimeBaseInitStructure.TIM_Prescaler = psc;//预分频系数

      TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1;//固定不变

      TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up;//向上计数

      TIM_TimeBaseInit(TIM14,&TIM_TimeBaseInitStructure);


6.定时器基本的参数配置完了,但是还没设置它为pwm输出模式:

用到的函数为:


void TIM_OCxInit(TIM_TypeDef* TIMx,TIM_OCInitTypeDef* TIM_OCInitStruct);//


注意:我们知道每个通用定时器有多达 4 路 PWM 输出通道(对于 TIM9-TIM14 最多有 2 路),所以TIM_OCxInit 函数名中的 x 值可以为 1/2/3/4。函数的第一个参数相信大家一看就清楚,是用来选择定时器的。第二个参数是一个结构体指针变量:


typedef struct

{

uint16_t TIM_OCMode; //比较输出模式

uint16_t TIM_OutputState; //比较输出使能

uint16_t TIM_OutputNState; //比较互补输出使能

uint32_t TIM_Pulse; //脉冲宽度

uint16_t TIM_OCPolarity; //输出极性

uint16_t TIM_OCNPolarity; //互补比较输出极性

uint16_t TIM_OCIdleState; //空闲状态下比较输出状态

uint16_t TIM_OCNIdleState; //空闲状态下比较输出状态

}


这里我们比较常用的 PWM 模式所需的成员变量:

TIM_OCMode:比较输出模式选择,总共有 8 种,最常用的是 PWM1 和 PWM2。

TIM_OutputState:比较输出使能,用来使能 PWM 输出到 IO 口。

TIM_OCPolarity:输出极性,用来设定输出通道电平的极性,是高电平还是低电平。

结 构 体 内 其 他 的 成 员 变 量 TIM_OutputNState , TIM_OCNPolarity ,

TIM_OCIdleState 和 TIM_OCNIdleState 是高级定时器才用到的。


所以配置完为:


TIM_OCInitStructure.TIM_OCMode=TIM_OCMode_PWM1;     

TIM_OCInitStructure.TIM_OCPolarity=TIM_OCPolarity_Low;

TIM_OCInitStructure.TIM_OutputState=TIM_OutputState_Enable;

TIM_OC1Init(TIM14,&TIM_OCInitStructure);


7.开启定时器


TIM_Cmd(TIM14,ENABLE);


8.修改 TIMx_CCRx 的值控制占空比(这一步写在主函数中,因为要实时去改变占空比).


其实经过前面几个步骤的配置,PWM 已经开始输出了,只是占空比和频率是固定的,例如本章要实现呼吸灯效果,那么就需要调节 TIM14 通道 1 的占空比,通过修改 TIM14_CCR1 值控制。调节占空比函数是:


void TIM_SetCompare1(TIM_TypeDef* TIMx, uint32_t Compare1);//对 于 其 他 通 道 , 分 别 有 对 应 的 函 数 名 , 函 数 格 式 是 TIM_SetComparex(x=1/2/3/4)。

1

分析:第一个参数不说了,第二个参数是计数值。。注意:这个计数值一定要小于前面设定定时器时总的预装载值(TIM_TimeBaseInitStructure.TIM_Period = pre);


代码:


pwm.c


#include "pwm.h"


void TIM14_PWM_Init(u16 pre,u16 psc)

{

      GPIO_InitTypeDef GPIO_InitStructure;

      TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure;

      TIM_OCInitTypeDef TIM_OCInitStructure;



      RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM14,ENABLE);//´ٍ؟ھ¶¨ت±ئ÷µؤت±ضس

          RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOF,ENABLE);//زٍخھسأµ½¶ث؟ع£¬ثùزشزھت¹ؤـ¶ث؟عت¼ت±ضس£¬ءيحâioµؤحâة趼تا¹ز½سشعAHB1×ـدكةدµؤ


          GPIO_PinAFConfig(GPIOF,GPIO_PinSource9,GPIO_AF_TIM14);


      GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AF;

          GPIO_InitStructure.GPIO_Pin=GPIO_Pin_9;

      GPIO_InitStructure.GPIO_Speed=GPIO_Speed_100MHz;//ثظ¶ب

      GPIO_InitStructure.GPIO_OType=GPIO_OType_PP;//حئحئحىتن³ِ

      GPIO_InitStructure.GPIO_PuPd=GPIO_PuPd_UP;//ةدہ­

      GPIO_Init(GPIOF,&GPIO_InitStructure);//شع³ُت¼»¯ز»دآ


          TIM_TimeBaseInitStructure.TIM_Period = pre;//ةèضأ¶¨ت±ئ÷µؤضـئع£¬ز²¾حت£×¢زâ£؛صâہïخھ´«µؤ²خت‎

      TIM_TimeBaseInitStructure.TIM_Prescaler = psc;//¶¨ت±ئ÷ش¤·ضئµدµت‎

          TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1;//ح¨³£²»ذق¸ؤثû£¬¹ج¶¨µؤ£¬ز»°مآج²¨µؤت±؛ٍسأ

      TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up;//دٍةد¼ئت‎£¬´سءم؟ھت¼¼ئت‎£¬¼ئت‎µ½ضط×°ضµ

      TIM_TimeBaseInit(TIM14,&TIM_TimeBaseInitStructure);//³ُت¼»¯ز»دآ


          TIM_OCInitStructure.TIM_OCMode=TIM_OCMode_PWM1;

        TIM_OCInitStructure.TIM_OCPolarity=TIM_OCPolarity_Low;

          TIM_OCInitStructure.TIM_OutputState=TIM_OutputState_Enable;

        TIM_OC1Init(TIM14,&TIM_OCInitStructure);


        TIM_Cmd(TIM14,ENABLE);


}


mian.c


int main()

{

    u8 fx = 0;

    u32 i = 0;//ำรภดภผำ

    RCC_HSE_Config(8,336,2,7);

    Beep_Init();

    NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);

    SysTick_Init(168);

    key_init();

    LED_Init();

    TIM14_PWM_Init(500-1,84-1);//2000hz,ึฦฺถจมห,0.5MS


    //while(1)

    //{

        //  if(fx==0)

            //{

                //  i++;

                //  if(i==300)

                //  {

                    //      fx=1;

                //  }

            //}


            //else

            //{

            //      i--;

                //if(i==0)

            //  {

                //      fx=0;

                //}

            //}

         // TIM_SetCompare1(TIM14,i);

            //delay_ms(10);

    //}



    /********the second code*****/

        while(1)

        {

           while(i<=300)

           {

              TIM_SetCompare1(TIM14,i);

              delay_ms(10);

              i++;

           }


           while(i!=0)

           {

              TIM_SetCompare1(TIM14,i);

              delay_ms(10);

              i--;

           }

      }

}


补充:

1.主函数中注释代码是官方的代码,下面是我自己原创的代码。官方代码是引用了一个变量来判断方向。


2.观察主函数的代码,我们设定的定时器为0.5ms,意思就是pwm波的周期为0.5ms,而每次执行完TIM_SetCompare1(TIM14,i);波形(占空比)就会改变,而后面写的delay_ms(10);是为了维持这一个波形一段时间,反映在模拟信号上就是电压变化的很平缓,很慢,自然亮度的变化也就很缓慢自然。自然,这个延迟时间知道要大于你定时时间好几倍吧-.-


扩展

试试用pwm输出来控制板子上的蜂鸣器来实现控制他的声音大小


提示:看电路图发现beep连在pf9上,而pf9本来就是tim13的复用口。所以只需在这个代码基础上修改定时器的标号为tim13即可

关键字:stm32  呼吸灯 引用地址:stm32之定时器运用———呼吸灯

上一篇:基于STM32F103 PWM呼吸灯设计
下一篇:STM32F103 呼吸灯的实现

推荐阅读

筹备已经的第92届中国电子展,将于10月31日-11月2日在上海新国际博览中心隆重登场,展会重点展示电子元器件、集成电路、测试测量、智能制造及电子设备、汽车电子、物联网等领域。届时,国内外知名厂商、行业大咖、专家学者将汇聚一堂,共同探讨行业发展新方向。作为万众瞩目的中国电子行业盛会将会呈现怎样的精彩,让我们拭目以待! 创新电子元器件才能走...
在现代经济发展的新形势下,“政务公开、推进电子政务、加快政府信息化建设”已成为政府信息化的重要发展方向。为满足人民来访(联合)接待中心电子政务机房的建设需要,实现可扩容、绿色环保机房的构建,台达技术团队经过对现场环境的详细勘察,遵循技术先进性、高可靠性及耐用性、与其他系统之间的兼容性、可扩展性、适用性的原则,为用户量身定制了一整...
是德科技公司(NYSE:KEYS)日前发布了一款原生云的分布式网络性能和安全性测试软件——CyPerf。该软件可以帮助机构明确用户体验特征以及验证混合网络、安全设备和服务的性能极限,使他们更自信地推出新产品。是德科技是一家领先的技术公司,致力于帮助企业、服务提供商和政府客户加速创新,创造一个安全互联的世界。 许多新技术,譬如 SD-WAN、多云体系结...
东莞松山湖,这个隐秘地方又跑出了一只独角兽。投资界获悉,知名机器人公司海柔创新不久前宣布完成两轮超2亿美元融资。其中,C轮融资由五源资本领投,红杉中国、源码资本、VMS鼎珮集团、华登国际、势能资本跟投;D轮融资由今日资本领投,红杉中国、五源资本、源码资本、联想之星、零一创投跟投。截至目前,海柔创新已经完成七轮融资,并在一年内完成B+、C...

史海拾趣

小广播
设计资源 培训 开发板 精华推荐

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

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

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

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