KST-STM32学习之PWM实现的各种途径(软件硬件+通用高级定时器)

发布者:巳午未最新更新时间:2018-12-14 来源: eefocus关键字:KST-STM32  PWM  软件硬件  通用高级定时器 手机看文章 扫描二维码
随时随地手机看文章

一、STM32的通用定时器介绍


STM32F103ZE 拥有 TIM2、TIM3、TIM4 和 TIM5 共 4 个通用定时器。


STM32F103C8拥有TIM2/TIM3/TIM4这三个通用定时器


通用定时器除了具备基本定时器的向上计数器功能外,还可以向下、向上/向下计数。


此外还具备独立通道,能够实现输入捕获、输出比较、PWM 输出、单脉冲输出的功能。


二、输出比较产生PWM


1、产生PWM的几种方式


①、硬件实现。比如STM32 自带的 PWM 输出功能。


注意了,硬件实现的时候,外部接线需要接到通用定时器的外部通道上!


②、通用定时器的输出比较实现软件的PWM,这个软件产生的PWM要比定时器模拟产生的PWM精确许多。


③、单纯的定时器计数实现PWM。


2、STM32的输出比较相关介绍


每个通用定时器拥有4路捕获/比较通道。


每路通道都有一个捕获/比较寄存器(TIMx_CCRx)用于装载初值。


同时该寄存器包含两个寄存器,一个是供用户写入比较值的,另一个是和计数器比较的当前捕获/比较寄存器。


捕获/比较模式寄存器 x(TIMx_CCMRx)中有输出比较预装载使能位(OCxPE ),开启后只有当更新时间到来时, TIMx_CCRx寄存器的比较值,才会传入到当前捕获/比较寄存器。否则写入的比较值将立即生效。


当计数器和捕获/比较模式寄存器 x(TIMx_CCMRx)一样时,会发生什么呢?


①、假如此时设置了事件产生寄存器(TIMx_EGR)中的 CCxG 位,会产生一个捕获/比较事件。


②、设置了相应的中断使能位TIMx_DIER 寄存器中的 CCxIE 位,则会产生一个捕获/比较中断。


3、输出比较产生PWM代码实现


timer.c



#include "timer.h"


/* TIM4中断优先级配置函数 */

void NVIC_TIM4Enable(void)

{

    NVIC_InitTypeDef NVIC_initstructure;


    NVIC_initstructure.NVIC_IRQChannel = TIM4_IRQn;           //选择TIM4中断通道

    NVIC_initstructure.NVIC_IRQChannelCmd = ENABLE;           //使能中断通道

    NVIC_initstructure.NVIC_IRQChannelPreemptionPriority = 0; //设定抢占优先级为0

    NVIC_initstructure.NVIC_IRQChannelSubPriority = 0;        //设定响应优先级为0

    NVIC_Init(&NVIC_initstructure);

}


/* TIM4初始化函数*/

void TIM4Init(void)

{

    TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;


    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM4, ENABLE);        //使能TIM4时钟

    TIM_TimeBaseStructure.TIM_Period = 255;                     //自动重装载值设为255,方便产生0-255级RGB颜色等级

    TIM_TimeBaseStructure.TIM_Prescaler = 72*39-1;              //计数周期设为39us,以使RGB刷新达到100Hz无闪烁效果

    TIM_TimeBaseStructure.TIM_ClockDivision = 0;                //基本定时器没有时钟分频功能,此项会被忽略

    TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //设置向上计数模式

    TIM_TimeBaseInit(TIM4, &TIM_TimeBaseStructure);             //初始化TIM4

    TIM_OC1PreloadConfig(TIM4, TIM_OCPreload_Enable);           //使能TIM4 CCR1的预装载寄存器

    TIM_SetCompare1(TIM4, 0);                                   //设定TIM4捕获比较1寄存器值为0

    NVIC_TIM4Enable();                                          //初始化TIM4中断优先级

    TIM_ClearITPendingBit(TIM4, TIM_IT_Update|TIM_IT_CC1|TIM_IT_CC2|TIM_IT_CC3); //清除中断标志位,否则启动中断会先进中断服务函数

    TIM_ITConfig(TIM4, TIM_IT_Update|TIM_IT_CC1|TIM_IT_CC2|TIM_IT_CC3, ENABLE);  //使能TIM4更新中断和捕获/比较1、2、3的中断源

    TIM_Cmd(TIM4, ENABLE);                                      //使能TIM4定时器

}


stm32f10x_it.c


...


/* TIM4中断 */

void TIM4_IRQHandler (void)

{

    /* 采用直接寄存器操作,可以获得比库函数更高的执行效率,节省中断函数执行时间 */

    if ((TIM4->SR & TIM_FLAG_Update) != 0) //检测是否为定时器溢出中断

    {

        TIM4->SR = ~TIM_FLAG_Update; //清除更新中断标志位

        if (TIM4->CCR1 != 0)         //比较值(即占空比)不为0时,点亮小灯,下同

        {

            LED = 1;

        }

    }    

    else if ((TIM4->SR & TIM_IT_CC1) != 0) //检测比较中断产生时熄灭小灯,下同

    {

        TIM4->SR = ~TIM_IT_CC1; //清除捕获/比较通道1中断标志位

        LED = 0;

    }

}


此处只是简单的验证,所以在主函数通过设定比较值来改变占空比。


TIM_SetCompare1(TIM4, 5); //这里的设置的范围是0~255(定时器设置的时候分了256级!)


//当然也可以直接采用寄存器的方式 TIM4->CCR1 = 5;


还需要注意的是通用定时器的分频值,预计装载值。具体解释可以看


三、硬件PWM


硬件实现的时候,外部接线需要接到通用定时器的外部通道上!当然也不可以,不在对应的外部通道上!(前提得能重映射!)


基本定时器不能输出PWM,通用和高级定时器才可以。


而通用和高级的配置又有所不同,一个一个来看…


1、高级定时器的硬件PWM输出


对应TIM1_CH1 - PA8 ; TIM_CH4 - PA11


void TIM1Init()

{

GPIO_InitTypeDef GPIO_InitStructure;

TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;

TIM_OCInitTypeDef  TIM_OCInitStructure;


RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM1, ENABLE);

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA , ENABLE);  //使能GPIO外设时钟使能


//设置该引脚为复用输出功能,输出TIM1 CH1 CH4的PWM脉冲波形

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8|GPIO_Pin_11; //TIM_CH1 //TIM_CH4

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;  //复用推挽输出

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

GPIO_Init(GPIOA, &GPIO_InitStructure);


TIM_TimeBaseStructure.TIM_Period = 255; //设置在下一个更新事件装入活动的自动重装载寄存器周期的值

TIM_TimeBaseStructure.TIM_Prescaler = 72*39-1; //设置用来作为TIMx时钟频率除数的预分频值  不分频

TIM_TimeBaseStructure.TIM_ClockDivision = 0; //设置时钟分割:TDTS = Tck_tim

TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;  //TIM向上计数模式

TIM_TimeBaseInit(TIM1, &TIM_TimeBaseStructure); //根据TIM_TimeBaseInitStruct中指定的参数初始化TIMx的时间基数单位


//TIM_OCMode;TIM_OCPolarity-对应输出的占空比CCR。 TIM_OCMode=1;TIM_OCPolarity=High或者TIM_OCMode=2;TIM_OCPolarity=Low为正常

TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; //选择定时器模式:TIM脉冲宽度调制模式1

TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //比较输出使能

TIM_OCInitStructure.TIM_Pulse = 1;                            //设置待装入捕获比较寄存器的脉冲值;也就是刚上电的占空比

TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;     //输出极性:TIM输出比较极性高

TIM_OC1Init(TIM1, &TIM_OCInitStructure);  //根据TIM_OCInitStruct中指定的参数初始化外设TIMx

TIM_OC4Init(TIM1, &TIM_OCInitStructure);  //根据TIM_OCInitStruct中指定的参数初始化外设TIMx


TIM_CtrlPWMOutputs(TIM1,ENABLE); //MOE 主输出使能

TIM_OC1PreloadConfig(TIM1, TIM_OCPreload_Enable);  //CH1预装载使能  

TIM_OC4PreloadConfig(TIM1, TIM_OCPreload_Enable);  //CH4预装载使能  

TIM_ARRPreloadConfig(TIM1, ENABLE); //使能TIMx在ARR上的预装载寄存器

TIM_Cmd(TIM1, ENABLE);  //使能TIM1

}


2、通用定时器的硬件PWM输出


对应TIM3_CH2 - PA7


void TIM3Init(void)

{

GPIO_InitTypeDef GPIO_InitStructure;

TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;

TIM_OCInitTypeDef TIM_OCInitStructure;


RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE);  //使能定时器 3 时钟

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); //使能 GPIOA时钟


//设置该引脚为复用输出功能,输出 TIM3 CH2 的 PWM 脉冲波形  GPIOA.7

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_7; //TIM_CH2

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //复用推挽输出

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

GPIO_Init(GPIOA, &GPIO_InitStructure); //初始化 GPIO


//初始化 TIM3

TIM_TimeBaseStructure.TIM_Period = 255; //设置在自动重装载周期值

TIM_TimeBaseStructure.TIM_Prescaler = 72*39-1; //设置预分频值

TIM_TimeBaseStructure.TIM_ClockDivision = 0; //设置时钟分割:TDTS = Tck_tim

TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //TIM 向上计数模式

TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure); //初始化 TIMx


//初始化 TIM3 Channel2 PWM 模式

TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; //选择 PWM 模式 1

TIM_OCInitStructure.TIM_Pulse = 255;

TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //比较输出使能

TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; //输出极性高

TIM_OC2Init(TIM3, &TIM_OCInitStructure); //初始化外设 TIM3 OC2


TIM_OC2PreloadConfig(TIM3, TIM_OCPreload_Enable); //使能预装载寄存器

TIM_Cmd(TIM3, ENABLE); //使能 TIM3

}

--------------------- 

关键字:KST-STM32  PWM  软件硬件  通用高级定时器 引用地址:KST-STM32学习之PWM实现的各种途径(软件硬件+通用高级定时器)

上一篇:KST-STM32学习之项目实战RGB舞台灯光
下一篇:KST-STM32学习之位带操作

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

【话说定时器系列】之十:PWM输入模式测量脉宽及占空比实验
STM32定时器 是 ST MCU 内部最基础且常用的外设,实际应用尤为普遍。去年,电堂推出了 《STM32 TIMER基础及常规应用介绍》 ,为大家梳理了 STM32 TIMER 的庞大内容,涵盖 TIMER 的基本应用原理、常规应用等。现在将课程内容整理为文章,针对STM32定时器有基本了解的用户,分享具体的应用实现环节及常见问题解决。 1. 实验内容 使用STM32定时器PWM输入模式测量脉宽及占空比。 上一节介绍了利用定时器输入捕获功能,通过定时器的单个通道来实现对1路外来信号脉冲宽度及占空比进行测量,并在测量过程中统计和计算用于测量的定时器自身的溢出事件。这里将介绍另外一种测量脉宽及占空比的方式。利用定时器PW
[单片机]
STM32F429定时器4生成PWM,转化为DAC
STM32F429时钟树 原文地址 定时器PWM 从STM32F4的内部时钟树可知, (1)高级定时器timer1, timer8以及通用定时器timer9, timer10, timer11的时钟来源是APB2总线 (2)通用定时器timer2-timer5,通用定时器timer12-timer14以及基本定时器timer6,timer7的时钟来源是APB1总线 (3)当APB1和APB2分频数为1的时候,TIM1、TIM8-TIM11的时钟为APB2的时钟,TIM2-TIM7、TIM12-TIM14的时钟为APB1的时钟; (4)而如果APB1和APB2分频数不为1,那么TIM1、TIM8-TIM11的时钟为APB2的时钟
[单片机]
外设一个一个学_PWM
PWM脉冲宽度调制(PWM)! 一般用途:电机、机器人。系统时钟(当然一般是RTC)、蜂鸣器、手机屏幕明暗调节: 1、书写流程。 根据芯片手册、 关于使用PWM驱动蜂鸣器的步骤。 ①:设置PWM的模式为PWMTOUT1 GPDCON 0XE0300080 在4~7位 写入 0x2 ②:PWM_ON PWM_OFF /*作为控制命令参数*/ ③:设置预分频函数:SET_PRE SET_CNT TCNTM=2TCMPB 占空比 ④:关于寄存器的修改 TCFG0 0XEA000000 TCFG1 4~7 TCON 0XEA000008 8~11 0x2 0010 0x9 1001 驱动如下: #if
[单片机]
stm32 PWM各通道配置引脚
对pwm一直有个疑惑,直到看到这个说明, TIM1_ETR PA12 PE7 TIM1_CH1 PA8 PE9 TIM1_CH2 PA9 PE11 TIM1_CH3 PA10 PE13 TIM1_CH4 PA11 PE14 TIM1_BKIN PB12 PA6 PE15 TIM1_CH1N PB13 PA7 PE8 TIM1_CH2N PB14 PB0 PE10 TIM1_CH3N PB15 PB1 PE12 TIM2_CH1_ETR PA0 PA15 PA0 PA15 TIM2_CH2 PA1 PB3 PA1 PB3 TIM2_CH3 PA2 PB10 TIM2_CH4 PA3
[单片机]
STC12C5A60S2单片机PWM程序
两个头文件 main.c #include REG51.H #include intrins.h #define U8 unsigned char #define U16 unsigned int void DelayMs(U8 ms); void PWM_clock(U8 clock); void PWM_start(U8 module,U8 mode); ////////////////////// 延时子程序///////////////////////////// void DelayMs(U8 ms) //在11.0592M晶振下,stc10f系列(单周期指令)的ms级延时 { U16 i; while(ms--)
[单片机]
专访赛灵思副总裁:让软件开发者更容易地使用硬件
文章来源:APP Devoloper Magazine 从5G和无人驾驶汽车到科学研究,都需要硬件能够可靠、快速和高效地执行独特的任务,并能够适应不断变化的标准和规范。 在本文中,Xilinx的AI和软件副总裁Ramine Roane讨论了如何通过统一的软件平台使新的和现有的芯片技术帮助广大开发变得越来越易用,并且使技术可以进步得更快。 问:是什么驱动着FPGA在新的垂直领域和细分市场中越来越广泛的采用? Roane:简而言之,数据和计算需求继续呈指数级增长,而CPU性能却停滞不前。 CPU频率难以持续提升,促使CPU制造商转移至多核架构,从本质上将缩放问题从芯片层迁移到软件层。但是,阿姆达尔定律严重限制了多线程软件的加速效率
[嵌入式]
单相PWM整流器能量双向传输的实现技术
引言 PWM 整流器已不是一般传统意义上的AC/DC转换器。由于电能的双向传输,当PWM整流器从电网吸取电能时,其运行于整流工作状态;而当PWM整流器向电网传输电能时,其运行于有源逆变工作状态。作为电网主要“污染”源的整流器首先受到了学术界的关注,并开展了大量研究工作。其主要思路就是将PWM技术引入整流器的控制当中,使整流器网侧电流正弦化,且于单位功率因数运行。能量可双向传输的PWM整流器不仅体现出AC/DC特性(整流),而且还可呈现DC /AC特性(有源逆变),因而确切地说,这类PWM整流器是一种新型的可逆PWM变流器。由于PWM整流器实现了网侧电流正弦化,且运行于单位功率因数, 甚至能量可双向传输,因而真正实现了“绿色电能转换”
[电源管理]
单相<font color='red'>PWM</font>整流器能量双向传输的实现技术
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
热门活动
换一批
更多
设计资源 培训 开发板 精华推荐

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

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

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