基于STM32的编码器详解# K2 Y! C4 e6 W6 n
基本原理:
# V* a [' m" `+ j7 G 光电编码器由三个部分组成:发光二极管,码盘以及码盘背面的光传感器。
% l( s; T! k, f1 ^! H+ s 这个码盘安装在旋转轴上,上面均匀地排列着透光和不透光的扇形区域。当码盘转动时,不透光的部分能够挡住光线,而透光区则允许光线透过,那么码盘背面的光传感器就会周期性地收到光信号,从而输出一列方波。
9 Z J* b. a- L2 t) |4 Z8 l; M) Q$ C! d. s% E$ j6 @
, q( H4 x! g1 H" J 我们知道,码盘转动一周时,光传感器输出的脉冲个数是一定的,通过检测一定时间内收到的脉冲个数,就可以知道在这段时间内码盘转动了多少圈,进而换算为速度。例如,一个码盘转动一周时会输出100个脉冲,在0.1s内我们收到了500个脉冲,这意味着0.1s内码盘转动了5周,即码盘的转速为50r/s。3 l7 k7 ]8 B/ z% c, z
( \" v$ X4 f+ Z- V$ Y2 K6 l
4 P( L, o2 D, K& x; l 但是,还有一个问题。设想,如果编码器只输出一列方波(假设为A),我们该怎样判断码盘是正转还是反转?因为无论是正转还是反转,都会产生同样的方波,而它们对速度的贡献显然是相反的!
6 ^6 q2 [# E+ K7 ]
* D% @' ~/ ]! w
* I V; c* q( `7 z5 P" M7 L 接下来我们看一看这个问题该怎样解决:9 |0 m r6 f# T) ^
/ |. O0 z% N# H+ H& \7 q
# ?5 @, B2 K9 J- D' k4 o 上面我们已经说过,码盘上均匀地刻着透光和不透光的扇形区域,我们在这一圈扇形区域内再均匀地刻上一圈透光和不透光的扇形区域,不同的是,外圈和内圈的区域是“交错”的。也就是说,当外圈处于不透光区域时,内圈对应的一半为透光区域,一半为不透光区域;当外圈处于透光区域时,内圈对应的一半为不透光区域,一半为透光区域。. E1 D$ x: E( s1 E- @; O7 M
, ~6 L6 G/ q. M4 w/ ~" ~9 p3 N
$ q# O7 V. U) }. x! E) r2 ]
于是,当码盘转动时编码器会输出两列相位差为90°方波,波形如下:$ ^2 e& ]( {. t; X
& |% }# {% i$ m! ~1 }5 q3 o
% \$ t6 Q! o; A8 Z- g
6 `* q K5 E2 D# l" E
{! h# X7 \% c6 f1 q
$ p- V C; a9 v9 \ 下面左图为正转时的波形,右图为反转时的波形:) E& L$ T! i3 M7 Q8 I6 j
, S M. s; W' G) T
7 d7 f4 Z2 l- Q( [
/ L, _# Y: Z( T* U" K! O2 o- G: s- E7 y
8 h3 K' f( s( E% D' e& e' i3 y1 w' W5 W
: g3 Z& _# E2 o0 Z+ b9 t$ u
当码盘正转时,在方波A的上升沿,方波B恒为低电平;当码盘反转时,在方波A的上升沿,方波B恒为高电平。通过判断A处于上升沿时,B的电平状态就可以方便地知道码盘旋转的方向。
- m+ g/ t) f* {
U0 ~1 A( M* w X0 k6 I
$ K" |! V: L" I8 ~ 速度计算:9 [: f# \- W. e. A0 n/ h v
, ~3 g: O( g8 j( \% S( e
& B$ F9 s, \4 j 不同型号的编码器,码盘旋转一周输出的脉冲个数是不同的,我使用的编码器码盘旋转一周输出的脉冲个数为90,小车轮子的直径为75mm 。
+ V1 h; w8 |# V/ O5 N
" s& W p5 H \; v% O# z9 c, A+ w+ R" K+ C
假如我们在时间T内统计到的有效脉冲数目为S(正转脉冲数+1,反转脉冲数-1),小车轮子的直径为D,圆周率为pi(约为3.14),那么小车的速度换算公式如下:+ J( ^/ o6 [- C& f$ l
7 i; e, t$ e! F5 {
' p" R( d2 t9 v n3 B( i V = pi*S*D / (90*T)
N/ Y' v: J1 N. m/ [* y" E) g
) Y' V& O) c/ G# m' W b8 \4 c( i. g# y
例如,我们每隔1 s统计一次有效脉冲数目,某一次得到的有效脉冲数目为500,那么小车此时的速度为:; ?! x/ f1 \5 V7 z, c3 m4 O6 d
+ C9 {1 D9 @" P+ ~
( @9 M( X2 m0 m( WV = 3.14*500*75*10^(-3)/(90*1)6 v7 J" `. j F4 N/ A5 b
% n2 o. F2 T* i
$ _8 ^: o) V* V& x7 Z( I
≈ 1.31 m/s
9 [% F7 O' H7 z: a
9 S1 a) d6 b& D) D- B5 u; J5 c# H% S/ f' v; n( |& w
我们可以看出,问题的关键在于,如何统计一定时间内的脉冲数目?目前有两种思路:
2 l5 I: }7 b, d; b& T! L8 }# R
E/ w3 g; t. B6 C b0 v" r) Q5 q3 b. t
①利用中断检测A的上升沿,触发中断时判断B的电平,来决定计数值加还是减;
6 S5 }9 d$ w+ A: K9 Z( r: H6 J
# @2 i+ \+ n0 k) z' z' N
②将定时器设置为编码器模式,直接读取计数值和方向。
6 J% U g. S; h+ C& e8 Q0 \
, s' Y# z; k$ U. Z6 u( }: Z8 F4 g$ l, t2 Y7 P
下面分别介绍这两种方法的实现:
5 x( Y1 c* d8 v; }( Z8 X8 B" P1 y* m! F5 E T& K2 F
9 U# ~5 c9 k n- a' I' ^
代码实现, i- J7 K# o5 K9 s
利用中断检测A的上升沿,触发中断时判断B的电平,来决定计数值加还是减
) Q4 M$ T! _9 N+ j2 o( t: J. J; ~3 o
9 O* |# z2 _1 c9 }. B
1 Z, ?% d/ i; L7 k(1)打开CubeMX,设置相关管脚,我这里使用的是PC2和PC3来 接A、B相9 ~7 Z3 D2 I" Q* d, \
, s4 k6 c) z1 `: v$ s+ G6 S( Z
$ O) C: C) J `$ O v
0 T f9 y& U% l# N7 F* g+ v7 @8 ^
8 Z9 V6 s; D7 w: e) v% i
4 \9 b! b2 h7 Q# C4 J& ~& i! e设置A、B相接口5 k# c1 n! L$ Y4 l/ B
( [* { p3 |, O5 e. A& b T0 {5 U/ f& g$ M# b H% Y7 ? N$ E/ H
+ W8 U" }) c/ H5 I
$ g8 o/ i4 m7 B% e# C
( G* J3 h( G& E8 s; w将PC2设置为上升沿触发
! q2 {' A+ y: P1 \$ X9 L8 {
8 |7 J( l& ?/ X' I/ T) Q, w( K# w7 h; b$ o7 B0 g
(2)为方便观察,打开UART串口,从串口读取采集到的数据" N& t1 k" f; F2 ^4 {' K* o
9 N1 G" n6 }" G
5 y& v6 ~/ H1 M& _$ S. _: n) x
9 \0 B0 e+ U% A: s4 ]0 n
- w3 ^3 I/ R3 p$ X& N; j
6 a# z% F- C' \
打开UART1
9 u, X9 g$ i" t- _3 n
* {9 X7 f# N& c/ Z$ |
; \2 M+ C! U: f& Z
/ I8 D7 f. `2 f4 ]0 o# U
! r9 Y2 g* {: J- H& m& _, C( N+ @, _! L3 v$ c6 @) Q
UART串口设置. y4 m1 [, S2 p, _
* H( `" v+ C) F: L f. r
+ v+ G" M6 |6 J& E! {) u) T( x( V
(3)打开定时器2,定时器响应时通过UART串口把计数值打印出来
. J) Y5 |/ }3 L* [8 h4 }* S' T$ ?1 o( E# N
: t6 O' q1 y& G
0 T" E0 c+ s4 j8 _# Y; b' V& R
3 ]( U) T4 ` L2 E( |6 E1 Z. w k* o& F6 g7 ^4 t& _4 ]
配置TIM2% F6 ~' a5 `% t# E' k
% ^- F( B: d. `; j5 N4 U* I) b% Q1 ]# u+ p& N7 Z7 M" f
7 V9 ^4 N. n; W6 P0 d8 s
( X4 x5 s3 ~% T5 v$ p
a* m$ A: _6 g& E) z
计时200ms
9 E O. ?$ C/ i& v2 @8 Y1 f3 L6 s5 W, F+ M# D
+ w6 _0 o& E/ e(4)导入Cube工程4 c- |9 t, s% i' m6 B( k1 S2 K
- ' N1 k3 Y- B8 f; M3 K K* c
- int COUNT = 0; //用来计数# T0 S2 q: \# s
- char str_buff[64];
$ v) \5 d K8 W7 O( B - char point[64];
( \8 t+ R$ C+ w6 u! m! Q - float speed = 0; //速度. T! K1 B; {- A
- int flag = 0; //flag=0时计数,flag=1时清零
复制代码 定义相关变量$ I7 T3 {: }4 j( R. c0 B
- 2 h' G5 a& L; r6 S( F
- void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
( q# s! _8 b. J4 ?$ I- Y - {
# v0 }! l$ a' `9 G' E - if (GPIO_Pin == GPIO_PIN_2)/ F |; C; m) V1 f4 M5 h
- {8 ?3 i( p' D' A% D- l. H- A
- if (flag == 1)
) W! o4 G# V& i9 K; K4 e - COUNT = 0;
, b2 s1 f I* @ o# @( z) E! m - else
/ A& v; R, r% \$ _! f - { k& f7 g3 J1 @* y! ^3 I' b& k
- if (HAL_GPIO_ReadPin(GPIOC, GPIO_PIN_3) == 0)" ?6 i- P E2 E- x* P8 e) c
- COUNT++;( u( V5 E! |7 t3 \ d9 [) a t
- else6 Z9 S' }. g, i$ n* v
- {2 \" n. E6 V- V
- COUNT--;
2 Y1 w, o- g# A* B$ T# q9 g/ W - } i* t: l6 f$ P$ |7 D3 @: K
- }; b; c7 n8 t; ]: r( ~7 m7 ?
- }
2 u/ E$ x6 `, ~% Y - }
复制代码 重写中断回调函数1 B: }& [( J3 F0 B7 U9 G
5 O0 v& p$ g! ]5 `- void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)
( P/ s/ V: i& h. B2 t" W - {, Y4 u) G7 ~* F) O
- if (htim->Instance == TIM2)
( s4 W3 X: B3 J$ W - {2 B; d/ E5 l- u
- if (flag == 0)
- _7 }5 {! K8 N V& } - {
3 ~0 S: F# B) y) ~- y r4 [ - speed = (float)COUNT * 3.14 * 0.14 / (90 * 0.2)*100/34; //cm/s6 A8 U; N4 R9 w6 o- L
- sprintf(str_buff, "speed:%dcm/s\r\n",(int)speed);
7 y. f4 H7 I3 z$ V. i/ | - HAL_UART_Transmit_IT(&huart1, (uint8_t*)str_buff, sizeof(str_buff));& t* }: p) l5 \, w" _
- flag = 1;. ^* K& s2 j+ e9 t3 P. [% P' i
- } a, ?2 s& A3 E/ q4 \; Y0 a2 _
- else
( V) H' L5 n3 {- T - {* [: L; Q9 |" l6 G
- flag = 0;+ l6 `" s7 y' V' q
- }
4 b. n: a2 [) j$ s - }
4 [& p. ]+ d* H - }
复制代码 定时器响应后通过UART串口发送数据, U4 H4 i/ s- v5 l5 H
- 5 c% J9 c, ]2 }9 i. O
- HAL_TIM_Base_Start_IT(&htim2);
复制代码 在主函数中打开定时器20 }! j9 }, h/ ~& ^& s) T- j
; w% E( H- f8 E% Z* r
/ B6 F7 t- [% C+ I5 }0 c实际测试2 D" ~/ s' D: B" Q( {6 o
测试效果如图:( i2 r; d; h d# Y( c' u+ _
" v0 P1 a8 z# ?. J3 q p
+ ~% k: ~7 v9 T4 _' o: L
2 {5 o$ i/ G" a3 i: |+ D3 ?
- n1 {2 Z$ e i% h
实际效果4 N! u9 s* L; J% ?( h, e. _
( I* A- y" [- V
& D5 V! T3 i5 q* `# i
2. 使用定时器的编码器模式
+ X$ g# v1 _+ A, \
$ X! K6 b+ l6 R, L8 _- ~/ }6 @# y& c* Q4 s
(1)设置TIM4为编码器模式
5 |7 T8 W! _- Z; [+ {& ?& [3 p7 o3 ?" y
- E* p& J8 u; x$ r" Z' w7 S/ W
. d) c- \$ Q9 W" P8 P" H
& [/ N6 w8 T8 Y4 O" t. e- x: W! b' c
设置Combined Channels
. V& A* d5 E/ b+ K T4 _8 N Z5 O: K. M, ]& O6 P
9 V7 N% b5 L$ h/ ?7 _7 ]( I(2)配置编码模式2 k' G) [& R& _" _( v0 x7 G
. z8 ?! j; }7 r/ s" Q$ |. }
5 D6 Q4 A' h8 U" y* A& F% E g
9 |2 |* U- l* P7 q+ R2 C1 Z3 k' PPrescaler:分频系数; x9 Z; w, z$ `
v& D. X* s1 X8 V
5 \4 g* e' c6 p/ g4 t9 i0 gCounter Mode:计数模式,设置为UP代表码盘正转时计数值增加,反 转时计数值减小& J: Q% G8 m3 o7 k) E3 G* t1 N9 R
I+ `# R3 N/ ^
- W% |8 W" E9 g( i
Counter Period:当编码器计数值等于该值时,计数值会清零,一般避 免编码器清零设置为65535(最大值)
8 R+ [( V# [# R0 h9 Z3 c& H6 [
9 v" ~, H( j# x4 s; y
' q4 u4 X: E/ I: mEncoder Mode:计数模式,编码器计数有三种模式可选! B2 x! u6 Q+ z7 Y6 [
3 Z- S d" W, n: i2 Y9 i3 L7 u9 \7 ~2 C1 q. ]
①T1* v" W0 y, s1 E
. f3 s% N% d# F* E: E/ W
* d Z E$ C* s+ I2 v9 ~0 B
只在上升沿计数,例如在一定时间内A产生了100个脉冲,那么编码器计数值为200(A、B产生脉冲数相等),由于分频系数的存在,实际调用函数得到的计数值为200 / ( Prescaler + 1 );% b9 c+ a. k: \ Z0 D/ {9 `
5 f0 r+ A7 D4 b) I( W0 D
0 X! Q- [; T+ P②T22 }) ^9 u& E4 ]8 H$ T
' S0 [5 t. ~# A5 v
4 |8 C: d8 `! _只在下降沿计数,计数值与T1相等( Prescaler相等的情况下);
6 b, t- ?, P7 ^
9 U5 \ n/ @; n8 R9 n; e7 K* s
: m: Q- [/ o4 V, t) [7 T③TI and T2
7 [% |. q& n: n8 I6 Y1 N0 J/ c/ Q/ a
8 z1 V, _, s4 }0 y9 D( H1 T, t
在上升沿、下降沿都计时,计数值为T1的两倍( Prescaler相等的情况下);+ M1 B" o* h4 o$ M9 j2 a
% ^7 M* o: \' w/ J$ Q7 S* x( L+ _% H5 k/ _0 }7 X
(3)导入CubeMx工程2 b" {1 O5 q$ O0 \5 x& D. j% t& {
- 0 F) R6 ^; z3 ^# N6 b! e$ e
- uint8_t GetData; //编码器计数$ U& n, | ^% Y6 B2 a$ M* G7 c
- uint8_t data[64];$ ^7 @& G# H* Q* p5 x
- uint8_t Direction; //编码器方向
复制代码 定义相关变量
0 [( A' u5 Y8 E) Q3 X/ c4 a" U7 q6 M- HAL_TIM_Encoder_Start(&htim4, TIM_CHANNEL_ALL);
复制代码 打开定时器模式" {. i$ K" O8 d$ u4 W
% c, B5 ~, z/ C: _- while (1)+ Q7 C- X' @# b' c7 S
- { Q/ ?% c7 u! w7 b' D( h! q
-
! I4 @, h) A! ]9 H9 V* s6 q4 L/ t% v - GetData = __HAL_TIM_GET_COUNTER(&htim4);
7 Y1 ?* E( P2 K( _ - Direction = __HAL_TIM_IS_TIM_COUNTING_DOWN(&htim4); 8 d+ V9 b1 E6 k/ M6 G5 q' t
- sprintf((char *)data, "fre:%4d dir:%d",GetData*2,Direction);
8 X9 ^+ U! ?. _, g4 s% W1 t2 G1 _ - 3 R( p% l3 _7 h3 P; @9 ~ R( Q
- OLED_Clear();2 b% V0 b' i" M H
- OLED_ShowString(0,0,data);& {1 o7 E. c( n( ]4 I
- __HAL_TIM_SET_COUNTER(&htim4,0);
D! m# ~8 k5 V - HAL_Delay(500);. r. }/ j" \. _; ~! Q
- /* USER CODE END WHILE */. K/ X0 |3 Q8 u4 F
0 f: n2 Y( |! y3 S- /* USER CODE BEGIN 3 */
5 E, R2 m+ e9 S8 l- L$ P% S8 Y - }
复制代码 读取相关值并通过OLED显示出来1 w) }2 I- c, w3 E
- U4 K. h' K9 u" W1 Y/ Q2 M( ?! U/ H2 @3 d( V8 W
实际效果
0 h6 [9 v( ~ x6 l( p$ T- | 用函数发生器模拟两列方波,配置编码器时我采用T1 and T2模式,分频系数为3,所以1s内采集到的脉冲个数为( 50 * 4 ) / ( 3 + 1 ) = 50,延时时间为500ms,所以OLED显示的值为 50 / 2 * 2 = 50 。
% s) d0 r0 Y1 \7 ~ K( ~9 X/ _
d+ i! @0 m7 R
$ K1 s. R4 g: {/ p& ~! ^3 n* {! ?: x! f" E* @1 \3 B: x0 c
# g) ?- O; G' b. T
实际效果. `" L& z i; q8 O$ s8 d& P1 K6 @
6 `9 h8 E' d' L) V( p; n
' i3 h8 z; ], j! y3 h: E- C! b: b
. q, W* F7 Z: \" O
3 W0 a4 N, K s |