【应用C】C语言实现基于中断方式的旋转编码器方向识别(编码器没有空闲状态)


  编码器类型:方向依靠AB两相的电平变化前后顺序而定,编码器没有所谓的空闲状态,先看编程识别方向的串口输出演示视频:

编码器(无空闲状态)编程识别方向的串口输出演示

01 - 编码器旋转波形

  这种编码本质上是一个开关,旋钮转动时带动两个IO口的电平变化,没有所谓的空闲状态,唯一判断旋钮转动的,是AB相电平变化的前后顺序,比如下图,黄线为A相,蓝线为B相,按照编码器的规格书说明:
  若电平的变化,A相比B相快(无论是上升沿还是下降沿),则为顺时针旋转
在这里插入图片描述
  相反,电平的变化,B相比A相快(无论是上升沿还是下降沿),则逆时针旋转
在这里插入图片描述
  其中,前后顺序的空隙时间是有最小规定的,可以参考规格书,一般视旋转的速度而定,但最小为几ms,可以作为滤波值的参考。

02 - 编程思路

  要对这种编码进行编程得到正确且响应速度快的软件,必须使用外部中断以及ms级别的定时器作滤波(如果电路带滤波,可以不需要),编程思路如下:
  A相和B相都需要各自拥有一套独立的电平记录器、滤波器和超时器,人为置AB相有空闲状态(程序变量为指定的不可能从IO口读到的值,比如100、200,IO口只能读到0和1),用外部中断边沿触发,若A相触发外部中断,先滤波,滤波结束后得到A相电平X保存到记录器中,检查B相的电平记录器Y,然后有以下2种情况:
  (1)、若X == Y,则说明B相比A相先触发,逆时针旋转,向接口输出结果。
  (2)、若X != Y,说明B相还没触发,于是A相保持电平记录器的值,等待B相。

  这里只是正常情况,因为我们人为置AB相有空闲状态,所以情况较为复杂,变为如下:
  (1)、若B相的电平记录器Y为空闲状态
    a、A相需要等待B相,但不是无限等待,要设置一个超时时间Timeout,在Timeout之前B相还没到达,则清空A相电平记录器,说明这次触发异常。
  (2)、若B相的电平记录器Y不为空闲状态
    a、若X==Y,则说明B相比A相先触发,逆时针旋转,向接口输出结果。
    b、若X!=Y,说明B相还没触发,于是A相保持电平记录器的值,等待B相,设置超时时间Timeout,操作如上

03 - 源代码

Encoder.h

#ifndef _Encoder_H_
#define _Encoder_H_
typedef enum {
	DIR_IDLE,
	DIR_INC,
	DIR_DEC,
} Dir_t;

extern void Encoder_Init(void);
extern void Encoder_A_ISR(void);
extern void Encoder_B_ISR(void);
extern void Encoder_Service_1ms(void);
extern void Encoder_EnableInterrupt(void);
extern void Encoder_DisableInterrupt(void);

extern Dir_t Encoder_GetDir(void);

Encoder.c

#include "Encoder.h"
#define ENCODER_A_RAW_VALUE					100
#define ENCODER_B_RAW_VALUE					200
#define ENCODER_FILTER_CNT_3MS				3
#define ENCODER_TIMEOUT_200MS				200

#define Encoder_Clear_Timeout()				{Encoder_A_Timeout = 0; Encoder_B_Timeout = 0;}
#define Encoder_Clear_FilterValue()			{Encoder_A_FilterValue = ENCODER_A_RAW_VALUE; Encoder_B_FilterValue = ENCODER_B_RAW_VALUE;}
#define Encoder_Set_DirInc()				{Encoder_Dir = DIR_INC;}
#define Encoder_Set_DirDec()				{Encoder_Dir = DIR_DEC;}
#define Encoder_Set_A_Timeout()				{Encoder_A_Timeout = ENCODER_TIMEOUT_200MS;}
#define Encoder_Set_B_Timeout()				{Encoder_B_Timeout = ENCODER_TIMEOUT_200MS;}

static Dir_t Encoder_Dir;

static volatile uint8_t Encoder_A_FilterCnt;
static volatile uint8_t Encoder_B_FilterCnt;

static uint8_t Encoder_A_Timeout;
static uint8_t Encoder_B_Timeout;

static uint8_t Encoder_A_FilterValue;
static uint8_t Encoder_B_FilterValue;

void Encoder_Init(void)
{
	Encoder_Dir = DIR_IDLE;
	Encoder_A_FilterCnt = 0;
	Encoder_B_FilterCnt = 0;

	Encoder_Clear_Timeout();
	Encoder_Clear_FilterValue();
}

void Encoder_EnableInterrupt(void)
{
	Encoder_Init();
	ExternalInterrupt0_Enable();
	ExternalInterrupt2_Enable();
}

void Encoder_DisableInterrupt(void)
{
	ExternalInterrupt0_Disable();
	ExternalInterrupt2_Disable();
	Encoder_Init();
}


void Encoder_A_Service_ISR(void)
{
	if (Encoder_A_FilterCnt == 0) {
		Encoder_A_FilterCnt = ENCODER_FILTER_CNT_3MS;
	}
}


void Encoder_B_Service_ISR(void)
{
	if (Encoder_B_FilterCnt == 0) {
		Encoder_B_FilterCnt = ENCODER_FILTER_CNT_3MS;
	}
}

void Encoder_Service_1ms(void)
{
	//Filter handler
	if (Encoder_A_FilterCnt) {
		if(--Encoder_A_FilterCnt == 0) {
			Encoder_Set_A_Timeout();
			Encoder_A_FilterValue = IO_Encoder_A;
			//B's Falling or Rising edge arrives earlier than A
			if (Encoder_B_FilterValue != ENCODER_B_RAW_VALUE) {
				if (Encoder_A_FilterValue == Encoder_B_FilterValue) {
					Encoder_Set_DirInc();
				}
				Encoder_Clear_FilterValue();
				Encoder_Clear_Timeout();
			}
		}
	}

	if (Encoder_B_FilterCnt) {
		if(--Encoder_B_FilterCnt == 0) {
			Encoder_Set_B_Timeout();
			Encoder_B_FilterValue = IO_Encoder_B;
			//A's Falling or Rising edge arrives earlier than B
			if (Encoder_A_FilterValue != ENCODER_A_RAW_VALUE) {
				if (Encoder_B_FilterValue == Encoder_A_FilterValue) {
					Encoder_Set_DirDec();
				}
				Encoder_Clear_FilterValue();
				Encoder_Clear_Timeout();
			}
		}
	}

	//Timeout handler
	if (Encoder_A_Timeout) {
		if (--Encoder_A_Timeout == 0) {
			Encoder_Clear_FilterValue();
		}
	}

	if (Encoder_B_Timeout) {
		if (--Encoder_B_Timeout == 0) {
			Encoder_Clear_FilterValue();
		}
	}
}

Dir_t Encoder_GetDir(void)
{
	Dir_t Dir_Buff = Encoder_Dir;
	if (Dir_Buff != DIR_IDLE) {
		Encoder_Dir = DIR_IDLE;
	}

	return Dir_Buff;
}
  • 1
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
C语言中,旋转编码器中断实现可以通过以下步骤: 1. 配置旋转编码器的引脚和中断服务程序(ISR)。 2. 在主程序中开启中断服务程序。 3. 在中断服务程序中读取编码器状态,根据旋转方向更新计数器的值。 以下是一个简单的旋转编码器中断实现的示例代码: ``` #include <avr/io.h> #include <avr/interrupt.h> volatile int encoderCount = 0; ISR(PCINT0_vect) { static uint8_t prevEncoderState = 0; uint8_t encoderState = PINB & 0x03; if (encoderState != prevEncoderState) { if (prevEncoderState == 0) { if (encoderState == 1) { encoderCount++; } else { encoderCount--; } } else if (prevEncoderState == 1) { if (encoderState == 3) { encoderCount++; } else { encoderCount--; } } else if (prevEncoderState == 3) { if (encoderState == 2) { encoderCount++; } else { encoderCount--; } } else { // prevEncoderState == 2 if (encoderState == 0) { encoderCount++; } else { encoderCount--; } } prevEncoderState = encoderState; } } int main(void) { DDRB &= ~0x03; // Set encoder pins as inputs PCICR |= 0x01; // Enable PCINT0 interrupt PCMSK0 |= 0x03; // Enable interrupts on encoder pins sei(); // Enable global interrupts while (1) { // Main program loop } } ``` 在这个例子中,我们假设旋转编码器的两个引脚分别连接到 ATmega328P 微控制器的 PB0 和 PB1 引脚。我们使用 PCINT0 中断来监听这两个引脚的状态变化,并在中断服务程序中更新计数器的值。具体而言,我们在 ISR 中读取当前编码器状态,根据状态的变化判断旋转方向,并更新计数器的值。需要注意的是,我们使用了一个静态变量来保存前一个编码器状态,以便于下一次中断服务程序的处理。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值