
![]() - J6 W' u3 F- i 有人使用STM32定时器的PWM功能,遇到点小疑问,这里一起看看。他先将STM32某定时器计数周期设为0xff,单向递增计数模式,OC比较值设为0x7F。在某时刻将新的计数周期0x7F与比较值0x3F加载到影子寄存器。当正在进行的一个周期结束后,经示波器测量确实可以看到其下一个周期发生变化,但其周期明显与预设值对应不上!再次经过一个周期,定时器才会按照预设值稳定输出。2 b3 z9 ]6 c4 \! J* `' U% S# D 以上是咨询者不算很清晰的描述【其实咨询TIMER问题最好配上时序图】,但可以了解到他要做的事情就是在ARR=0xff,ccr=0x7F的PWM输出状态下,于某时刻赋予ARR和CCR新值以改变输出波形。 ![]() 事情不算复杂,疑点就是为什么需要2个周期延时后才能有基于新配置的稳定输出。【他这里说的2个周期显然不清楚到底指的前后哪个周期值】 # j* J0 r: r1 b$ ^: C+ m: j1 K 这个问题主要涉及到定时器寄存器的预装功能。有些时候我们需要保证输出波形完整性、连续性,开启ARR及CCR的预装功能就比较合适,使用CubeMx配置时做些勾选即可。开启预装后,修改ARR及CCR的值,生效时间点最长可能延后1个旧计数周期。不过这里要注意,我们修改ARR及CCR的值若不是特别需求,建议在一个计数周期内完成修改,不要一个值在更新事件之前完成赋值,另一个则发生在该更新事件之后。 + }* P8 H) A1 k- } 我也将上面提醒告知咨询者,他反馈问题依然没能解决,准确说是疑惑依然没能解除。他这里的确只是提出疑惑,并未提出具体需求。另外,他还将问题的复现过程做了如下补充:9 B' C( L* y( z6 |3 `1 C" N 3 N; @% t6 i5 a 第一步:& ]) \5 e# k* I9 r 1 - 关闭预装载(ARR/CCR) 2 - 设定ARR=CCR=0xFF, r; p6 b; O+ s9 t& ?, } 3 - 开启输出比较功能(连接到示波器查看波形)) x+ ~/ S* R; I 8 k ?: w4 T' |: N* h ; k1 j7 B+ _( |2 {0 p! l3 }7 p! B 第二步: 1 - 设置示波器触发(延迟一段时间拉高某个IO引脚以触发示波器采集) 2 - 立即清空CNT! O& c9 R& t7 R" f$ x 3 - 使能预装载 4 - 设定ARR=0X7F,CCR=0X3F 现象:本周期结束后其计数周期确实会相应改变,但改变之后的前两个周期明显与设定值不符,即经过2个周期后才可稳定输出。 / }1 Z0 y' _# E! d, y& A. V6 g ( q! z# c7 I! j( m( L. ]5 K 复现过程表述得比较清晰了,他也再次明确了疑惑点。同时还强调说网上很多人碰到类似问题或疑惑。他在STM32L4系列和STM32H7系列上都遇到了同样问题。既然这样,现在我们使用STM32L4系列的TIM1来进行验证,参照他的操作步骤来组织代码,看看来龙去脉。 - j. V( J2 O+ \: O6 F+ s: m 我是这样测试验证的。4 J# }, G" o, t& h- \# x) S 6 t* y6 U/ h: o2 F' V+ K 在关闭ARR/CCR的预装前提下给二者分别赋值0xff和0x7f,启动CH1的PWM输出并使能该通道的输出比较中断。在第一个脉冲的比较中断里按照咨询者的做法修改ARR/CCR值。为了便于观察效果,我也开启了更新中断,通过更新中断记录脉冲个数,输出几个脉冲后就将定时器及输出都关闭掉。【选择PWM模式1,极性为高有效】 / X( d/ ]9 A) r% k9 M 编写代码、编译除错后,运行得到如下结果:【黄色波形结尾的4个脉冲是基于新参数的输出。绿色波形的上升沿指示修改数据的时间点,这里辅助显示下,重点在黄色波形。】 6 Q4 s4 s5 u0 k- t ![]() 9 F' g5 F8 b% `+ O6 l$ g : b( u; a$ L3 j4 E- @ 上面输出波形应该说跟客户反馈的是一致的。我们来一起看看。在第一个脉冲的比较中断处,也是第一个脉冲的正中央【红色箭头所指位置】处做参数修改。修改步骤按照咨询者提供的来实现。) Q0 ^+ m& I3 R* t" m1 ~! m 7 V3 ?/ ~. i' Y8 T, U; p # S2 X7 t5 g: M) F" Z6 U' L ![]() T$ P; X5 c# ]+ ~3 C 即按照上面的3小步操作【第1小步是咨询者设置示波器的动作,不用理它】。1 f) ~$ k+ u8 D# X. u/ ? * `3 W6 v3 o1 }# k+ m 9 r5 J7 R+ Z8 ?9 ^/ A: R 这里是开启预装功能后才修改ARR/CCR值,这两个新值目前只能暂居预装寄存器,实际起作用的ARR/CCR值仍是之前的0xff和0x7f。而且,还在此处对计数器做了清零,即从此刻起PWM输出脉冲重头来。这样从上图红色箭头到蓝色箭头之间的波形依然是基于ARR=0xff,CCR=0x7f参数运行的。定时器于蓝色箭头所指位置处发生更新事件,新数据【0x7f,0x3f】生效起作用。 6 A+ M8 S1 Y# ^" K' {, F/ x* {, z & V3 D1 @6 k5 Q# f9 A; [- T 显然从修改时刻算起到数据生效 刚好延时1个旧周期,结合到这里,恰好是新周期的2倍。修改数据后整整花了2个周期的时间才生效就是这么来的。 - `- U X2 [% e6 a' D$ x9 k 如果在上面操作步骤的基础上拿掉对计数器清零操作,其它不变,输出结果又不一样了。见下面截图: - D5 B& Z" [- R L" p! H! z ![]() 同样,红色箭头所指位置为修改数据的时刻。由于此时没有对计数器清零,计数器按部就班计数,PWM输出按预定配置输出,继续运行半个旧周期后计数器溢出产生更新事件于蓝色箭头所指位置。蓝色箭头所指位置以后PWM输出按新参数运行。不难看出,这次新数据的生效从修改时刻算起仅延时半个旧周期,相当于后续的1个新周期。 " h# {- c& x. a! l: a- ?/ ]; M 这里强调下,我这里测试时选择的刚好是旧周期的中间点,所以延时生效为半个旧周期。具体应用时的延时跟我们修改参数所选的时间点有关,一般来讲,它最长不会超过1个旧周期,最短极限为0。; s% @# v8 Y2 {% J e8 F0 o; i- O ; O2 r( c+ B8 {5 |; |0 `, e( A 当然,如果说我们不关心修改数据前后波形的完整性,直接关闭预装功能也行,或者手动产生更新事件也可以。总之,我们根据实际应用需求来定。6 n# P+ p; {2 |! K2 H& i - i" |5 Y u, R) D: e 聊到这里,关于咨询者的疑惑基本解释得差不多了。9 C$ c) R! d- L+ h7 B8 Y $ C0 Y C: k, c* s: e 或许眼尖的人看到上面第一种操作输出的PWM波形里面有个非常细窄的尖脉冲,它是怎么回事呢?示波器问题?非也!【见下图椭圆形框住的细长线】 . ]% x3 s: U o / e# Q7 X T% v$ p3 N4 k ; z9 a( ~; t: w/ I) k ![]() 此处尖脉冲产生的原因是------当发生比较中断时我们才去做修改参数动作,在做计数器清零操作前,计数器的值已经大于设置的CCR值【0x7f】,按照当前PWM输出模式及极性选择,输出则变为低电平了。但是,在中断里我们很快又将计数器值做了个清零,此时计数器值又小于CCR值【0x7f】了。同样,按照当前PWM配置,输出又变回高电平,最终就产生了这么个一下一上的尖脉冲。# @8 |$ a3 N3 ] , @' \) e. I6 S$ o- v - u2 J5 R) E8 _- m m 转载自: [color=var(--weui-FG-2)]Miler 如有侵权请联系删除 |
基于定时器捕获测量脉宽的应用示例
狂欢三】STM32C031使用TIM定时器DMA方式实现WS2812彩灯输出(三)
【狂欢三】STM32C031使用TIM定时器DMA方式实现PWM输出(二)
【狂欢三】STM32C031使用TIM定时器PWM输出
stm32使用定时器触发dma传输,启动dma没反应的几种情况的解决方法
定时器剩余通道是否可以做PWM输出呢?
基于STM32双定时器+ADC+DMA实战经验分享
基于STM32的定时器触发ADC时可能遇到的情形
【NUCLEO-U545RE-Q评测】5. 基本计时器
基于STM32的定时器不按设定超时产生中断