文章目录
一、编码器原理
二、为什么要用编码器
三、STM32编码器配置相关
四、STM32实战代码
五、一些注意
参考
一、编码器原理
如果两个信号相位差为90度,则这两个信号称为正交。由于两个信号相差90度,因此可以根据两个信号哪个先哪个后来判断方向、根据每个信号脉冲数量的多少及整个编码轮的周长就可以算出当前行走的距离、如果再加上定时器的话还可以计算出速度。
二、为什么要用编码器
从上图可以看出,由于TI,T2一前一后有个90度的相位差,所以当出现这个相位差时就表示轮子旋转了一个角度。但有人会问了:既然都是脉冲,为什么不用普通IO中断?实际上如果是轮子一直正常旋转当然没有问题。仔细观察上图,如果出现了毛刺呢?这就是需要我们在软件中编写算法进行改正。于是,我们就会想到如果有个硬件能够处理这种情况那不是挺好吗?
对应的硬件的编码器就来了~
我们看到STM32的硬件编码器还是很智能的,当T1,T2脉冲是连续产生的时候计数器加一或减一一次,而当某个接口产生了毛刺或抖动,则计数器计数不变,也就是说该接口能够容许抖动。
在STM32中,编码器使用的是定时器接口,通过数据手册可知,定时器1,2,3,4,5和8有编码器的功能,而其他没有。同时只有CH1和CH2是进行编码器模式的~
三、STM32编码器配置相关
编码器输入信号TI1,TI2经过输入滤波,边沿检测产生TI1FP1,TI2FP2接到编码器模块,通过配置编码器的工作模式,即可以对编码器进行正向/反向计数。
比如如果用的是定时器2,则对应的引脚是在PA0和PA1上。
通常为了提高精度我们会选择在上升沿和下降沿都进行计数!
还有一个非常重要的图这里也记录下
其中让人费解的应该是在第二列的相对信号的电平,这里就来详细谈一下吧。
其实也不难理解哈,我们上面也说了通常为了提高精度会在A、B两相的上升沿和下降沿都进行计数,那么对应在一个周期就可以计数四次,计数次数的增加也就意味着精度的提高!
编码器模式下,如果此时处于正转,那么这四次计数应该都是加的。同理,如果是反转,那么这四次计数都是减的。那么问题来了,如何判断正反转呢?
不就是在相对电平的基础上嘛!!!
仔细对照图中,在正转或者反转的情况下,A相对B的电平高低以及上表中的计数方向便可了然于心!!!
选择编码器接口模式的方法是:如果计数器只在TI2的边沿计数,则置TIMx_SMCR寄存器中的
SMS=001;如果只在TI1边沿计数,则置SMS=010;如果计数器同时在TI1和TI2边沿计数,则
置SMS=011。
通过设置TIMx_CCER寄存器中的CC1P和CC2P位,可以选择TI1和TI2极性;如果需要,还可以
对输入滤波器编程。
两个输入TI1和TI2被用来作为增量编码器的接口。假定计数器已经启动(TIMx_CR1
寄存器中的CEN=’1’),计数器由每次在TI1FP1或TI2FP2上的有效跳变驱动。
配置范例:

CC1S=’01’ (TIMx_CCMR1寄存器, IC1FP1映射到TI1) CC2S=’01’ (TIMx_CCMR2寄存器, IC2FP2映射到TI2) CC1P=’0’ (TIMx_CCER寄存器, IC1FP1不反相, IC1FP1=TI1) CC2P=’0’ (TIMx_CCER寄存器, IC2FP2不反相, IC2FP2=TI2) SMS=’011’ (TIMx_SMCR寄存器,所有的输入均在上升沿和下降沿有效). CEN=’1’ (TIMx_CR1寄存器,计数器使能)
四、STM32实战代码
————————————————

1 /*TIM2初始化为编码器接口*/ 2 void Encoder_Init_TIM2(void) 3 { 4 TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; 5 TIM_ICInitTypeDef TIM_ICInitStructure; 6 GPIO_InitTypeDef GPIO_InitStructure; 7 8 RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);//使能定时器4的时钟 9 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);//使能PA端口时钟 10 11 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0|GPIO_Pin_1; //端口配置 12 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; //浮空输入 13 GPIO_Init(GPIOA, &GPIO_InitStructure); //根据设定参数初始化GPIOA 14 15 TIM_TimeBaseStructInit(&TIM_TimeBaseStructure); 16 TIM_TimeBaseStructure.TIM_Prescaler = 0x0; // 预分频器 17 TIM_TimeBaseStructure.TIM_Period = ENCODER_TIM_PERIOD; //设定计数器自动重装值 18 TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;//选择时钟分频:不分频 19 TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;////TIM向上计数 20 TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure); 21 TIM_EncoderInterfaceConfig(TIM2, TIM_EncoderMode_TI12, TIM_ICPolarity_Rising, TIM_ICPolarity_Rising);//使用编码器模式3 22 TIM_ICStructInit(&TIM_ICInitStructure); 23 TIM_ICInitStructure.TIM_ICFilter = 10; 24 TIM_ICInit(TIM2, &TIM_ICInitStructure); 25 TIM_ClearFlag(TIM2, TIM_FLAG_Update);//清除TIM的更新标志位 26 TIM_ITConfig(TIM2, TIM_IT_Update, ENABLE); 27 //Reset counter 28 TIM_SetCounter(TIM2,0); 29 TIM_Cmd(TIM2, ENABLE); 30 } 31 32 /*TIM4初始化为编码器接口*/ 33 void Encoder_Init_TIM4(void) 34 { 35 TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; 36 TIM_ICInitTypeDef TIM_ICInitStructure; 37 GPIO_InitTypeDef GPIO_InitStructure; 38 RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM4, ENABLE);//使能定时器4的时钟 39 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);//使能PB端口时钟 40 41 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6|GPIO_Pin_7; //端口配置 42 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; //浮空输入 43 GPIO_Init(GPIOB, &GPIO_InitStructure); //根据设定参数初始化GPIOB 44 45 TIM_TimeBaseStructInit(&TIM_TimeBaseStructure); 46 TIM_TimeBaseStructure.TIM_Prescaler = 0x0; // 预分频器 47 TIM_TimeBaseStructure.TIM_Period = ENCODER_TIM_PERIOD; //设定计数器自动重装值 48 TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;//选择时钟分频:不分频 49 TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;////TIM向上计数 50 TIM_TimeBaseInit(TIM4, &TIM_TimeBaseStructure); 51 TIM_EncoderInterfaceConfig(TIM4, TIM_EncoderMode_TI12, TIM_ICPolarity_Rising, TIM_ICPolarity_Rising);//使用编码器模式3 52 TIM_ICStructInit(&TIM_ICInitStructure); 53 TIM_ICInitStructure.TIM_ICFilter = 10; 54 TIM_ICInit(TIM4, &TIM_ICInitStructure); 55 TIM_ClearFlag(TIM4, TIM_FLAG_Update);//清除TIM的更新标志位 56 TIM_ITConfig(TIM4, TIM_IT_Update, ENABLE); 57 //Reset counter 58 TIM_SetCounter(TIM4,0); 59 TIM_Cmd(TIM4, ENABLE); 60 } 61 62 /*单位时间编码器计数 输入定时器 输出速度值*/ 63 int Read_Encoder(u8 TIMX) 64 { 65 int Encoder_TIM; 66 switch(TIMX) 67 { 68 case 2: Encoder_TIM= (short)TIM2 -> CNT; TIM2 -> CNT=0;break; 69 case 3: Encoder_TIM= (short)TIM3 -> CNT; TIM3 -> CNT=0;break; 70 case 4: Encoder_TIM= (short)TIM4 -> CNT; TIM4 -> CNT=0;break; 71 default: Encoder_TIM=0; 72 } 73 return Encoder_TIM; 74 } 75 76 void TIM4_IRQHandler(void) 77 { 78 if(TIM4->SR&0X0001)//溢出中断 79 { 80 } 81 TIM4->SR&=~(1<<0);//清除中断标志位 82 } 83 84 void TIM2_IRQHandler(void) 85 { 86 if(TIM2->SR&0X0001)//溢出中断 87 { 88 } 89 TIM2->SR&=~(1<<0);//清除中断标志位 90 }