ST给的TIM例子都是实现一个固定频率的当时输出,如果想在每次定时中断后改变定时器值,那么需要动态的修改定时器周期配置。
0 d; ~" _" G8 K) O) ?实测中发现一有坑,给大家共享。
' r- L7 F5 w) v- void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)
6 {6 ` Y, A5 s0 u - {2 o( Q! P: [" R! a( r5 ~& N
- /* USER CODE BEGIN Callback 0 */! W6 q0 y! w4 S( S: e/ p" n
! g- M5 w2 m. w* x0 p2 D& m- /* USER CODE END Callback 0 */3 M7 O% X( g' J+ d( l$ v
- if (htim->Instance == TIM22) {
8 D0 m/ D! p# M9 ^. o3 j - HAL_IncTick();7 }* d4 x. N- P/ B ~! Z1 y
- }
. N" t: z6 C& Z3 k* }4 o% V - /* USER CODE BEGIN Callback 1 */
* ^! Y! T' i8 P9 V }6 w - if (htim->Instance == TIM2) {
* L4 n+ J5 ?, z2 n - IrdaTransfer(pDataNEC);) S0 T1 t9 [; x' [6 m! B+ W
- }
' c% I& x; E& H8 p - /* USER CODE END Callback 1 */' u+ t! s1 |$ K4 O) } c/ K9 a8 x
- }
/ O: c& m& C% E9 ]( ^' b: r
复制代码 首先在HAL_TIM_PeriodElapsedCallback中添加TIM的Update事件中断回调函数,自己写IrdaTransfer(pDataNEC)实现NEC编码的红外遥控。
$ C. g& N: q" t' K. S& `- uint16_t IrdaTransfer(uint8_t* pDataNEC)
& S" K: }% ^0 `$ X - {) v/ f6 r2 D& d) T8 e5 l5 B: q
- PulseVal=TimeSerial(pDataNEC,PulseSteps);0 Z5 F7 z: ^7 ]. y, j
- PulseSteps++;% U; }( d3 S/ t/ s- D5 D8 s
- if(PulseVal>0)- c v: c- a6 M, e: {3 s
- {$ q6 O& |. y6 u: K# Q9 N" |
- htim2.Init.Period = PulseVal&0x7fff;/ E/ r6 M6 z2 G' b9 F
- HAL_TIM_Base_Init(&htim2);
- q0 @$ X0 l i9 ^" E% S - __HAL_TIM_CLEAR_FLAG(&htim2, TIM_SR_UIF);
" O' k4 X5 O4 x: p8 O - HAL_TIM_Base_Start_IT(&htim2);' O8 G! U0 K) u7 o
- }
6 i8 G9 ]* ?2 q- V) w4 m! W$ w. V' V - else
1 J' \' W3 a8 I3 [ - {
- Y8 N- C4 T/ o6 U2 R. Z; h9 V0 q( c8 { - __HAL_TIM_CLEAR_FLAG(&htim2, TIM_SR_UIF);
1 \) w0 @; c) E+ p2 ^9 s - HAL_TIM_Base_Stop_IT(&htim2);
# B5 M; c; o/ c' B1 r9 L% p - }
/ _6 |, i% S- j* c0 X6 T - if(PulseVal&0x8000)
9 |7 P+ {6 D4 N, H7 [: E - HAL_GPIO_WritePin(IrDA_Output_GPIO_Port, IrDA_Output_Pin, GPIO_PIN_SET);, S& \' j- e, c( b+ n
- else. M1 l8 u- b/ _
- HAL_GPIO_WritePin(IrDA_Output_GPIO_Port, IrDA_Output_Pin, GPIO_PIN_RESET);/ Y/ s1 z) @- g
- return PulseVal;4 ]. k6 v# o( X" U/ t
- }
" @# o: V4 [2 s" p4 m( n3 w
复制代码 注意这里的坑是:4 g4 ~) ?+ a0 Z4 F5 r
htim2.Init.Period = PulseVal&0x7fff;
) @" g2 i( p4 s- n+ H HAL_TIM_Base_Init(&htim2);
) \+ c3 O% S8 K! {" u. D __HAL_TIM_CLEAR_FLAG(&htim2, TIM_SR_UIF);//这是坑,必须在此处加。
$ c% |! H B6 @: g( D: HHAL_TIM_Base_Start_IT(&htim2);
% j* N2 I2 k0 s! Q在初始化定时时长后,必须先清除中断标志,再启动定时器中断,否则会直接进中断而没有在定时结束时进,这个坑调了我一天,不明白为啥初始化定时器就会触发一次UPDATE中断。9 s8 k9 v* F2 c2 h1 w l/ R
以上代码即可实现每次定时后立即更新定时值,实现任意的序列控制。具体的NEC红外序列实现功能我就不放了,留点自己动脑子的地方,填的坑与大家共享。
9 X6 k+ D+ @* \: s3 o
& v" _0 K* X7 j$ H: k& [4 m6 P3 Y0 z$ s; }3 _% B& j8 }
# s# k1 u. w- f' E( t4 \& M4 c
|
_# q1 ~0 ^5 b; I
HAL_TIM_Base_Start_IT(&htim2);% u2 V7 p; {# ~/ w* O( A
在初始化定时时长后,必须先清除中断标志,再启动定时器中断,否则会直接进中断而没有在定时结束时进,这个坑调了我一天,不明白为啥初始化定时器就会触发一次UPDATE中断。0 z* v$ y: m, G/ p/ n$ G
==>不能算是个坑,应该是手册没看清楚。
因为有些寄存器比方PSC/RCR寄存器必须通过更新事件才能更新,我们初始化时只能操作预转载寄存器,所以让我们用户数据生效,就手动产生个更新事件,让预装载寄存器的数据拷贝到影子寄存器【实际寄存器】发挥作用。但这个操作会置位更新中断标志,所以我们在使能更新中断前有必要清除下该标志UIF,否则可能一使能更新中断就跳进更新中断服务程序。