EC11旋转编码器、stm32f103驱动程序

本文围绕EC11旋转编码器与STM32开发展开。介绍了EC11手册要点,包括旋转速度、RC滤波等;阐述硬件电路,如加RC滤波电路及电容电阻参数;说明了驱动程序关键点,如IO口设置、中断触发方式等;还提及输出现象,最后分享汽车行业单片机开发对开关输入监测的成熟办法。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1、EC11手册的要点

注意:旋转的速度、RC滤波
在这里插入图片描述在这里插入图片描述
手册中推荐的电路(已含有RC滤波):
在这里插入图片描述
输出波形特点:
在这里插入图片描述

2、硬件电路
加上RC滤波电路
做法是两个端点都采用10pF电容接地,10KΩ电阻接VCC。
实测100pF电容也行。
在这里插入图片描述
用示波器看看波形有无噪声
另外,看看不同旋转速度时的延时要求(具体见下面的中断服务函数)
顺逆时针方向的波形:
第一张图里黄色波形上升沿触发外部中断时,只有大约0-2ms的时间就会错过另一波形的准确电平。
如果中断服务函数里最前部不加延时,判断真滴是会非常不准确。。。
在这里插入图片描述
在这里插入图片描述
3、驱动程序关键点
两个端点接的IO口设置:均采用下拉输入(不接信号时是低电平,用来检测是否有高电平信号输入)
使用外部中断来检测EC11端点电平变化:
中断触发方式:上升沿触发(这也是上面IO口设置成下拉输出入的原因)
中断服务函数:触发中断的端点为高电平时,判断此时另一端点电平状态是高还是低,以此来判断旋转方向是顺时针还是 逆时针。
针对旋转速度大小,可以调节中断服务函数里的延时。

最后,旋转EC11时别手残,转就好好转,我怀疑是我滤波电容没按照手册选。。。

下面贴主要源码:
(从正点原子mini板源码基础上改的)

【knob.h】
#define knob1_clk PAin(6)
#define knob1_dt PAin(1)

【knob.c】
void knob_init(void)
{
	GPIO_InitTypeDef GPIO_InitStructure;
 	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
 	
	GPIO_InitStructure.GPIO_Pin  = GPIO_Pin_6;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD; //ÊäÈ룬ÏÂÀ­
 	GPIO_Init(GPIOA, &GPIO_InitStructure);
 
	GPIO_InitStructure.GPIO_Pin  = GPIO_Pin_1;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD;  
	GPIO_Init(GPIOA, &GPIO_InitStructure);
	
} 

【exit.c】
//外部中断初始化函数
void EXTIX_Init(void)
{
 	EXTI_InitTypeDef EXTI_InitStructure;
 	NVIC_InitTypeDef NVIC_InitStructure;

  	RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO,ENABLE);//外部中断,需要使能AFIO时钟

	knob_init();//初始化按键对应io模式 A6

    //GPIOA6 中断线以及中断初始化配置
  	GPIO_EXTILineConfig(GPIO_PortSourceGPIOA,GPIO_PinSource6);
  	EXTI_InitStructure.EXTI_Line=EXTI_Line6;
  	EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;	
  	EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Rising;//上升沿触发
  	EXTI_InitStructure.EXTI_LineCmd = ENABLE;
  	EXTI_Init(&EXTI_InitStructure);	//根据EXTI_InitStruct中指定的参数初始化外设EXTI寄存器

		
		NVIC_InitStructure.NVIC_IRQChannel = EXTI9_5_IRQn; //使能按键所在的外部中断通道
  	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0x02;	//抢占优先级2, 
  	NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0x01;		//子优先级1
  	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;	//使能外部中断通道
  	NVIC_Init(&NVIC_InitStructure); 
 

}

//中断服务函数
void EXTI9_5_IRQHandler(void)
{
	delay_ms(1);	//*************很重要*******************
	if(knob1_clk==1)	
		{
			if(knob1_dt == 1)
			{
				printf("knob: +1 \r\n");//顺时针
				//delay_ms(10);
		   	}else{
							printf("knob: -1 \r\n");//逆时针
				 }
	  }		
 	 EXTI_ClearITPendingBit(EXTI_Line6);    //清除LINE6上的中断标志位 
}

4、输出现象
先顺时针旋转,后逆时针旋转的现象:
在这里插入图片描述

【完】

20221222 更新:
后来入了汽车行业,发现对于开关输入监测有更成熟稳定的办法。
汽车行业的单片机开发,一般是有一个时间片轮询的自编小OS 或 freertos这种的改成时间片轮询方式运行;
在此基础上:
1个1ms执行1次的函数A监测沿变、滤波、记录电平;
1个5ms执行1次的函数B取得滤波后的电平判断出旋转方向。

【完】

### STM32F103 HAL库与EC11旋转编码器接口 #### 配置环境准备 为了使STM32F103能够成功连接并操作EC11旋转编码器,需先准备好必要的开发工具链。这包括但不限于选用合适的芯片型号STM32F103RCT6[^1]、安装STM32CubeMX用于初始化外设配置以及MDK-Keil作为集成开发环境来编写和调试程序。 #### 初始化设置 利用STM32CubeMX完成基本的硬件资源分配工作非常重要。对于EC11旋转编码器而言,其两个信号线应分别被设定为外部中断触发源之一,以便检测旋钮转动方向;而中间按钮则可简单地作为一个普通的GPIO输入端口处理,并启用内部上拉电阻以简化电路设计[^3]。 #### 编码器模式配置 针对STM32系列微控制器,在使用TIMx定时器时可以通过特定寄存器配置进入正交解码(即四倍频)模式,从而更精确地捕捉到增量型编码器产生的脉冲序列变化情况。具体来说就是将某个通用定时器比如TIM3的工作方式调整至Encoder Mode下运行[^2]。 #### 获取计数值 当一切就绪之后就可以通过调用`__HAL_TIM_GET_COUNTER()`宏函数轻松获取当前时刻相对于起始位置所经历过的总步数了。此方法返回的是一个无符号整数类型的变量enc,代表自上次清零以来累计增加或减少了多少个单位距离[^5]。 ```c // 假定已经完成了上述所有准备工作... uint32_t enc; /* 读取TIM3定时器中的计数器值 */ enc = (uint32_t)(__HAL_TIM_GET_COUNTER(&htim3)); ```
评论 23
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值