Linux下的旋钮驱动

 先看一次正旋和反旋波形上差异

 

可以看出两个波形慢了一个相位,如图中,判断正反旋的一种方法如下
绿色信号线下降开始监听黄色的信号线的状态,如果是下降沿,判断为正旋,如果是上升沿,判断为反旋,绿色信号线上升结束一次监听 

      正旋输出

    

      反旋输出

 

#include<linux/module.h>
#include<linux/gpio.h>
#include<linux/input.h>
#include <linux/interrupt.h>
#include<linux/platform_device.h>
#define knod1_pin  76  //gpio引脚号
#define knod2_pin  124  //gpio引脚号
int knod1_irq;
int knod2_irq;
int knod_start;
struct input_dev *input_dev;

static irqreturn_t knod1_irq_hander(int irq, void *dev_id)
{
	int state;
	state = gpio_get_value(knod1_pin);
	if (!state){  //下降沿
		knod_start=1;
	}else{		 //下降沿
		knod_start=0;
	}

	return IRQ_HANDLED;
}

static irqreturn_t knod2_irq_hander(int irq, void *dev_id)
{
	int state;
	state = gpio_get_value(knod2_pin);
	if (knod_start&&state) {  //下降沿
		input_report_key(input_dev, KEY_VOLUMEUP, 1);
		input_sync(input_dev);
		input_report_key(input_dev, KEY_VOLUMEUP, 0);
		input_sync(input_dev);
	}
	if (knod_start&&(!state)){
		input_report_key(input_dev, KEY_VOLUMEDOWN, 1);
		input_sync(input_dev);
		input_report_key(input_dev, KEY_VOLUMEDOWN, 0);
		input_sync(input_dev);
	}
	return IRQ_HANDLED;
}

void request_knod_irq(void)
{
	int ret;
	ret = gpio_request(knod1_pin, "knod1_pin");
	if (ret < 0){
		printk("request pin %d err!\n", knod1_pin);
		return;
	}
	ret = gpio_request(knod2_pin, "knod2_pin");
	if (ret < 0){
        printk("request pin %d err!", knod1_pin);
		return;
	}
	knod1_irq = gpio_to_irq(knod1_pin);
	knod2_irq = gpio_to_irq(knod2_pin);
	ret =request_irq(knod1_irq, knod1_irq_hander ,IRQ_TYPE_EDGE_BOTH, "knod1_irq",(void *)knod1_pin);
	if (ret < 0){
		printk("request irq %d err!\n", knod1_pin);
		return;
	}
	ret =request_irq(knod2_irq, knod2_irq_hander ,IRQ_TYPE_EDGE_BOTH, "knod2_irq", (void *)knod2_pin);
	if (ret < 0){
		printk("request irq %d err!\n", knod2_pin);
		return;
	}
}

int register_knod_input_device(void)
{
	int ret;
	input_dev = input_allocate_device();
	if (!input_dev){
		printk("failed to allocate input device err!\n");
		return -1;
	}
	input_dev->name = "knod";
	set_bit(EV_KEY, input_dev->evbit);
	set_bit(KEY_VOLUMEUP, input_dev->keybit);
	set_bit(KEY_VOLUMEDOWN, input_dev->keybit);
	ret = input_register_device(input_dev);
	if (ret){
		printk("faild to register input device!\n");
		input_free_device(input_dev);
		return -1;
	}
	return 0;
}

static void knod_release(struct device *dev)
{
	return;
}

static int knod_probe(struct platform_device *pdev)
{
	printk("%s\n", __FUNCTION__);
	pdev->dev.release = knod_release,
	request_knod_irq();
	register_knod_input_device();
	return 0;
}

static int knod_remove(struct platform_device *pdev)
{
	free_irq(knod1_irq, (void *)knod1_pin);
	free_irq(knod2_irq, (void *)knod2_pin);
	gpio_free(knod1_pin);
	gpio_free(knod2_pin);
	input_unregister_device(input_dev);
	return 0;
}

static struct platform_driver knod_driver = {
	.driver = {	
				.name = "knod",
                .owner =THIS_MODULE,
              },
	.probe  = knod_probe,
	.remove = knod_remove,
};

static struct platform_device knod_device = {
	.name = "knod",
};

static int knod_init(void)
{
	printk(KERN_INFO "knod_init!\n");
	platform_device_register(&knod_device);
	return platform_driver_register(&knod_driver);
}

static void knod_exit(void)
{
	printk(KERN_INFO "knod_exit!\n");
	platform_driver_unregister(&knod_driver);
	platform_device_unregister(&knod_device);
} 

module_init(knod_init);
module_exit(knod_exit);
MODULE_LICENSE("GPL v2");

 

### 旋钮驱动按键 Encoder 的工作原理与实现 #### 编码器基本结构 旋转编码器是一种能够将机械运动转换为电信号的装置,通常由一组开关触点组成。其核心功能在于检测位置变化并将其转化为脉冲信号供处理器解析[^2]。 #### 工作原理概述 旋转编码器主要依赖两个通道 A 和 B 输出相位差为90度的方波信号。这种设计使得可以通过监测两路信号之间的相对关系来判断转动方向以及计算转过的角度或步数。具体来说: - 当顺时针旋转时,A 相先于 B 相发生变化; - 反之逆时针旋转,则B相领先于A相发生改变[^1]。 此外,部分型号还配备了一个额外的按钮作为中心按下确认操作的功能扩展。 #### 软件驱动实现方法 对于嵌入式 Linux 系统下的编码器支持,可以采用轮询机制或者更高效的中断处理模式: ##### 中断方式 利用 GPIO 引脚配置成边沿触发类型的硬件中断源,在每次状态切换(上升沿/下降沿)都会引发一次 CPU 响应程序执行特定的任务逻辑流程如下所示: ```c static irqreturn_t rotary_encoder_irq_handler(int irq,void *dev_id){ static uint8_t last_state=LOW; uint8_t current_a,current_b; // 获取当前AB端口的状态 current_a=gpio_get_value(GPIO_A); current_b=gpio_get_value(GPIO_B); if(current_a !=last_state ){ if (current_b==HIGH){ counter++; }else{ counter--; } } last_state=current_a; printk(KERN_INFO "Counter Value:%d\n",counter); return IRQ_HANDLED; } ``` 上述代码片段展示了如何基于Linux Kernel框架编写针对某一指定GPIO编号所关联起来外部物理实体对象——即此处提到的Rotary Encoder 设备单元进行事件捕捉响应过程中的关键环节之一。 同时需要注意初始化阶段要完成必要的资源申请绑定动作比如注册对应的IRQ handler函数指针变量等常规步骤省略未展示出来以便聚焦重点内容阐述清晰简洁明了易于理解掌握实际应用场景需求灵活调整优化性能表现达到预期目标效果最佳实践分享给大家共同进步成长! ---
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值