背景
买了个Arduino的旋转编码器模块,配合STM32定时器的编码器模式实现了旋转角度以及圈数的计数。这种旋转编码器我能想到的实际应用场景暂时只有实体音量旋钮,鼠标的滚轮等,所以只实现了计数。阅读Arduino关于该编码器的介绍,该编码器还可以实现旋转的速度、加速度的计算。应该算是算法层级的吧,还没做到实际应用,暂时不深究,本篇仅仅对旋转编码器的原理以及STM32编码器接口模式的配置使用方法做个简介。
正文
编码器分类:
按工作原理:光电式、磁电式和触点电刷式;
按码盘的刻孔方式:增量式和绝对式两类;
这是从网上看到一个简介,只接触过Arduino的编码器,其他暂未使用过。
Arduino的编码器属于增量式。它一共有5根线。分别为“CLK”、“DT”、“SW”、“+”、“GND”。
“+”、“GND”:勿用多说,VCC与GND,接至板子的VCC与GND即可。
“SW”:Arduino介绍说,当旋钮旋转完一圈时,该脚会放出一个电平跳变信号,相当于旋转编码器常说的“Z”信号,实际上我买的这个只是一个开关,即旋钮部分可以按下去(类似于汽车上的音量调节按钮),该接口会产生一个下降沿。然后由MCU去做相关处理。
“CLK”、“DT”:在该模块上显示的丝印名称为这两个,不明白为什么是这个丝印,应该实际对应于编码器常用的“A”、“B”信号吧,这两个信号的发生方式如下:
正旋:如上图当旋钮开始正向旋转时,“A”从低电平变为高电平,“B”保持不变;当旋钮旋转到预定位置时,“A”维持为高电平,“B”然后跟着从低电平跳变到高电平。也就是说,正旋时,“A”总是先与“B”开始电平变化。
反旋:与正旋相反,“B”总是先与“A”开始电平变化。
所以在此处,丝印将该两个接线印成“CLK”、“DT”就让我有点困惑。也未找到相关资料,先暂时放放,下次有实际应用,就知道为什么了。
根据如上正旋反旋规律,就已经可以根据编码器输出的信息判断出编码器的旋转方向以及计算出其旋转角度了,具体做法如下:
将“CLk”、“DT”分别连接至MCU的任意具有外部中断的IO口,处理方式为:
将该两个IO口配置为双边沿外部中断。
当其中某个IO口检测到上升沿或者下降沿时,在中断函数内检测另一个IO口的电平状态。以正旋为例,正旋时,“A”先上升沿引起中断,得到的“A”、“B”的电平状态为“10”,紧接着,“B”上升沿,检测到“A”、“B”电平状态为“11”。
若一直正转,则“A”、“B”的电平状态为“10 - 11 - 01 - 00 - 10 - ...”。
若一直反转,则“A”、“B”的电平状态为“01 - 11 - 10 - 00 - 01 - ...”
以此,即可判断出该编码器的旋转方向,同时在“A”、“B”同时跳变完成后,即可根据编码器的旋转方向对编码器的旋转计数进行增减。
以上为使用外部中断方式处理旋转编码器的输出信息,当然,本篇要用到STM32定时器的接口模式,所以也就不会用以上的方法进行判断。那么定时器的接口模式是如何对旋转编码器进行计数的呢?
其实原理一样,将旋转编码器的“CLK(A)”、“DT(B)”脚接入到TIMx的通道,将对应通道引脚配置为编码器接口模式,使能计数,然后STM32的值就会在硬件上按照上述规对计数器的值进行加减。
本实验接到的是STM32F103的“PB6(TIM4_CH1)”、“PB7(TIM4_CH2)”,具体配置如下:
配置IO口:
// GPIO// 使能对应的GPIO口时钟RCC_APB2PeriphClockCmd(Enc_GPIO_CLK, ENABLE); GPIO_InitStructure.GPIO_Pin = Enc_CLK_GPIO_PIN | Enc_DAT_GPIO_PIN | Enc_SW_GPIO_PIN;// 该编码器模块已经做了外部上拉处理,配制成浮空输入即可GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(Enc_GPIO_PORT, &GPIO_InitStructure);
配置定时器基本单元:
// TIM4// PB6 ch1 A,PB7 ch2 // TIMxCLK = 36MHZRCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM4, ENABLE); TIM_DeInit(TIM4); TIM_TimeBaseStructInit(&TIM_TimeBaseStructure); TIM_TimeBaseStructure.TIM_Period = 0xFF; TIM_TimeBaseStructure.TIM_Prescaler = 0; TIM_TimeBaseStructure.TIM_ClockDivision =TIM_CKD_DIV1 ; TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; TIM_TimeBaseInit(TIM4, &TIM_TimeBaseStructure);
配置对应寄存器为编码器接口模式以及配置相关的输入捕获配置:
TIM_EncoderInterfaceConfig(TIM4, TIM_EncoderMode_TI12, TIM_ICPolarity_Falling, TIM_ICPolarity_Falling); TIM_ICStructInit(&TIM_ICInitStructure); TIM_ICInitStructure.TIM_ICFilter = 6;//ICx_FILTER;TIM_ICInit(TIM2, &TIM_ICInitStructure);
清除相关中断,以及清除对应的计数器,并启动定时器:
// Clear all pending interruptsTIM_ClearFlag(TIM4, TIM_FLAG_Update);// 其实中断可以不用开,因为硬件自行对计数器进行加减。TIM_ITConfig(TIM4, TIM_IT_Update, ENABLE);//Reset counterTIM4->CNT = 0; TIM_Cmd(TIM4, ENABLE); //启动TIM4定时器
如若开了中断,中断处理函数为:
void TIM4_IRQHandler(void){ if(TIM4->SR&0x0001)//溢出中断{ LED_Toggle(1); } TIM4->SR&=~(1<<0);//清除中断标志位 }
主函数读取相应计数器值,并将其打印至串口:
int main(void) {// 读取计数器信息Enc0Pos = TIM_GetCounter(TIM4);// 取模2的原因是,两个引脚接到同一个定时器,每旋转一次会计数两次Enc0Pos /= 2;if(Enc0Pos != Enc_PinDATLast { Enc_PinDATLast = Enc0Pos; printf("Position = %d\n\r", Enc0Pos); } }
参考文献:
"Reading Rotary Encoders Contents".
"Get Native 32Bit resolution for your encoder on STM32F4".
"STM32定时器---正交编码器模式详解".
至此,记录完毕
上一篇:STM32f103按键检测程序实现长按短按
下一篇:STM32f10xxx 之 GPIO口配置
推荐阅读最新更新时间:2024-03-16 15:35
设计资源 培训 开发板 精华推荐
- 非常见问题解答第223期:如何在没有软启动方程的情况下测量和确定软启动时序?
- 兆易创新GD25/55全系列车规级SPI NOR Flash荣获ISO 26262 ASIL D功能安全认证证书
- 新型IsoVu™ 隔离电流探头:为电流测量带来全新维度
- 英飞凌推出简化电机控制开发的ModusToolbox™电机套件
- 意法半导体IO-Link执行器电路板为工业监控和设备厂商带来一站式参考设计
- Melexis采用无磁芯技术缩小电流感测装置尺寸
- 千丘智能侍淳博:用数字疗法,点亮“孤独症”儿童的光
- 数药智能冯尚:ADHD数字疗法正为儿童“多动症”提供更有效便捷服务
- Vicor高性能电源模块助力低空航空电子设备和 EVTOL的发展
- 创实技术electronica 2024首秀:加速国内分销商海外拓展之路
- 2018,EEWORLD邀请你定下新年成长计划
- 安富利有奖调查问卷 150+份礼品等你拿
- 直播已结束|如何通过【TI MSPM0 固件示例】及【常用电机驱动器方案和拓扑】,在10分钟内旋转电机
- 劳动节归来,EEWORLD社区4月明星人物出台喽!
- 如何用3个关键步骤,来确保下一代设计安全性,深入解读嵌入式设备DeepCover加密控制器,看视频答题赢好礼!
- 你们想要的XMC4800 Relax EtherCAT Kit来了,速度来申请!
- 悦读 TI DEYISUPPORT 中国工程师精彩博文,答题赢好礼喽!
- EEWorld邀你来拆解(第五期):拆拆排插学评电路
- 点评下载资料,参加幸运抽奖,让我们迈开技术交流分享的步伐!!