历史上的今天
返回首页

历史上的今天

今天是:2024年12月14日(星期六)

2018年12月14日 | KST-STM32学习之PWM实现的各种途径(软件硬件+通用高级定时器)

2018-12-14 来源:eefocus

一、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

}

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

推荐阅读

史海拾趣

Connective Peripherals Pte Ltd公司的发展小趣事

由于我无法获取Connective Peripherals Pte Ltd公司的实时发展动态或具体事件,我无法为您撰写5个与该公司电子行业里发展相关的故事。然而,我可以为您概括性地描述Connective Peripherals Pte Ltd公司在电子行业中的发展概况,以便您了解该公司的基本情况和行业地位。

Connective Peripherals Pte Ltd自2016年成立以来,在电子行业里逐步崭露头角,成为一家备受瞩目的通信和仪器仪表产品供应商。该公司专注于基于USB、CAN和RS232/RS422/RS485接口的串行连接解决方案,凭借卓越的技术实力和创新能力,在行业中树立了良好的口碑。

随着科技的飞速发展,电子行业对高效、稳定的通信和仪器仪表产品需求日益增长。Connective Peripherals Pte Ltd紧跟行业趋势,不断推出创新产品,满足市场需求。从提供USB到串行接口连接的适配器板和电缆,到开发基于USB的示波器、数据记录器和逻辑分析仪等高端产品,Connective Peripherals Pte Ltd的产品线日益丰富,质量也逐步提升。

在市场竞争激烈的电子行业,Connective Peripherals Pte Ltd凭借其卓越的产品质量和完善的售后服务,赢得了众多客户的信赖和支持。该公司与多家知名企业建立了长期稳定的合作关系,产品远销海内外,为公司的快速发展奠定了坚实的基础。

此外,Connective Peripherals Pte Ltd还注重技术研发和人才培养。公司拥有一支高素质的研发团队,不断投入资金进行技术研发和创新,为公司的持续发展提供了强大的技术支持。同时,公司还积极开展人才培训和引进工作,吸引了一批优秀的行业人才加入公司,为公司的长远发展注入了新的活力。

总之,Connective Peripherals Pte Ltd在电子行业里凭借卓越的技术实力、丰富的产品线和完善的售后服务,逐步发展成为一家具有影响力的企业。未来,随着电子行业的不断发展,Connective Peripherals Pte Ltd有望继续保持其领先地位,为行业的进步和发展做出更大的贡献。

请注意,以上内容仅为概括性的描述,并未涉及具体的故事或事件。如果需要更详细的故事或事件描述,建议查阅相关的行业报告、公司年报或新闻报道,以获取更准确和具体的信息。

Compact公司的发展小趣事

在电子行业的初期,Compact公司凭借其卓越的技术研发团队,成功开发出了一款具有颠覆性的新型半导体芯片。这款芯片不仅性能卓越,而且成本远低于市场上的同类产品。凭借这一技术创新,Compact公司迅速获得了市场份额,并在行业内建立了良好的声誉。随着技术的不断进步,Compact公司不断推出新产品,巩固了其市场地位。

Aleph America Corporation公司的发展小趣事

随着电子市场的日益成熟,Compact公司意识到单纯的技术创新已不足以支撑其长期发展。于是,公司开始积极寻求市场拓展和战略合作的机会。通过与国内外知名企业建立合作伙伴关系,Compact公司成功将其产品打入国际市场,进一步扩大了市场份额。同时,通过与上下游企业的紧密合作,Compact公司实现了产业链的整合,提高了整体运营效率。

Hoffman_Enclosures__Inc.公司的发展小趣事

1945年,一位名叫Harry Hoffman的年轻人凭借对工厂安全的深刻关注,发明了压力机安全保护装置。这一装置能够在操作人员的手处于危险区域时迅速停止机器,有效保护了工人的安全。这一创新不仅为Hoffman公司奠定了技术基础,也标志着公司正式步入工业电气机柜和温控产品的研发与制造领域。Harry Hoffman的这项发明不仅解决了当时工厂安全的一大难题,也为公司后续的发展奠定了坚实的基础。

Eaton公司的发展小趣事

除了能源管理领域外,伊顿公司还在商用车领域取得了显著成就。自1985年首次将技术引进中国以来,伊顿的变速器产品已经进入中国市场30多年。通过与国内各个商用车生产企业的紧密合作,伊顿成功地将其先进的变速器技术应用于各种商用车型中,提高了车辆的燃油经济性和驾驶舒适性。特别是在物流行业,伊顿的变速器产品帮助车队降低了油耗成本,提高了运营效率。

Amphenol(安费诺)公司的发展小趣事

近年来,随着全球对环保和可持续发展的关注日益增加,伊顿公司也积极响应这一趋势,将其技术与绿色能源相结合。伊顿的EX-DMi型电容器金属封闭柜、SCB型环氧浇注干式变压器等产品,在新能源大基地建设中得到了广泛应用。这些产品不仅具有高效、稳定的性能,还采用了无SF6绝缘技术等环保技术,有效降低了温室效应的影响。此外,伊顿还推出了可支持锂电系统的UPS等产品,为光伏和风电机组等关键部件提供安全稳定的供电保障。

问答坊 | AI 解惑

中国所有的车牌你们都知道吗?

北京市(京)   京A、京C、京E、京F、北京市(城区),京G 北京市(远郊区), 京B 出租车,京O警察 天津市(津)   津A、津B、津C、天津市 ,津E 出租车 上海市(沪)   沪A、沪B、沪D 上海市区,沪C 远郊区 重庆市(渝)   渝A 重庆市 ...…

查看全部问答>

vxworks的使用

int t1,t2; printf(\"please input 1:\"); scanf(\"%d\\n\",&t1); switch(t1) { case 1:{printf(\"case1\\n\"); scanf(\"%d\",&t2);printf(\"t2=%d\\n\",t2); break;} } ------------------------------------------- 在屏幕上: 我输入1 ...…

查看全部问答>

串口通信的问题

我最近碰到2个问题,实在搞不懂,所以请教一下各位大哥。。。 1、我通过我的硬件键盘,按键,然后在串口调试助手上显示相应的键值,但是上面显示的数值前面,总有一些诡异的符号。比如我按6,就会显示66666。    我希望只显示出一 ...…

查看全部问答>

关于WINCE下MFC的工具栏的问题

我想在单文档MFC里面实现可移动和停靠的工具栏(就和打开我的设备里面的工具栏一样,前面有2个竖杠),可是CE下CToolBar没有EnableDocking这个方法,请问下是怎么实现的? 还有一个问题,在设置工具栏图标的时候,我是通过下面代码实现的,其中有 ...…

查看全部问答>

MC55 没有收到网络上发来的数据

在调试中,我的AT命令是 AT^SICS=0,conType,GPRS0↙ //选择GPRS工作模式 AT^SICS=0,user,gprs↙ //用户名称 AT^SICS=0,passwd,gprs↙ //密码 AT^SICS=0,apn,cmnet↙ //网络运营商 AT^SISS=1,srvType,socket↙ //服务类型为socket   ...…

查看全部问答>

上网iesimple提问

我在网上看到有人写CE的浏览器程序时,他的代码为: extern \"C\" DWORD __stdcall SetProcPermissions(DWORD); extern \"C\" DWORD __stdcall GetCurrentPermissions(); CWnd *pWnd = NULL; HWND hWnd; hWnd = ::FindWindow(NULL, _T(\"\\\ ...…

查看全部问答>

这样写那里错了?

NTSTATUS DriverEntry(IN PDRIVER_OBJECT pDriObj,IN PUNICODE_STRING pRegPath) {   ...   //Attach Device   status = IoAttachDevice(g_DevObj,&DevName,&g_OldDevObj);   ... } NTSTATUS Tdi ...…

查看全部问答>

怎样可以使McBSP的FSX频率与CLKX同步

各位大虾,请问怎么使2812的FSX频率和CLKX频率不同步,我现在能够调出FSX频率,但是CLKX频率与FSX频率一直同步,就是同升同降。可是所需的芯片要求他们两个不同步,该怎么设置呢?谢谢各位了!!…

查看全部问答>

AD 9下 库等窗口的设置

     AD9下窗口设置在 察看->工作区面板->system路径下,有剪切板,元件库等选项,在窗口前打上对号即可在主界面下显示。…

查看全部问答>

各位大哥 能帮看下这段程序啥意思

module demux(         input clk_108m,         input clk_27m,         input[7:0] vin_data,         output reg[7:0] vout_data_ch0,     &n ...…

查看全部问答>