在linux内核gpio驱动中使用消息队列解耦原有的功能函数

时间: 2024-05-11 18:20:07 浏览: 17
在Linux内核GPIO驱动中使用消息队列可以解耦原有的功能函数,实现异步处理GPIO中断事件。下面是一个简单的示例代码: ```c #include <linux/gpio.h> #include <linux/interrupt.h> #include <linux/module.h> #include <linux/mutex.h> #include <linux/slab.h> #define MAX_QUEUE_LEN 32 struct gpio_event { int gpio; bool value; struct list_head list; }; static struct gpio_event *events; static struct list_head event_list; static DEFINE_MUTEX(event_lock); static DECLARE_WAIT_QUEUE_HEAD(event_waitq); static struct task_struct *event_task; static struct task_struct *queue_task; static int gpio_event_thread(void *data) { while (!kthread_should_stop()) { wait_event_interruptible(event_waitq, !list_empty(&event_list)); mutex_lock(&event_lock); while (!list_empty(&event_list)) { struct gpio_event *event = list_first_entry(&event_list, struct gpio_event, list); list_del(&event->list); mutex_unlock(&event_lock); // 处理 GPIO 中断事件 printk(KERN_INFO "GPIO %d value %d\n", event->gpio, event->value); kfree(event); mutex_lock(&event_lock); } mutex_unlock(&event_lock); } return 0; } static irqreturn_t gpio_event_handler(int irq, void *dev_id) { unsigned int gpio = (unsigned int)dev_id; bool value = gpio_get_value(gpio); struct gpio_event *event = kmalloc(sizeof(*event), GFP_KERNEL); if (!event) { printk(KERN_ERR "Failed to allocate gpio event\n"); return IRQ_HANDLED; } event->gpio = gpio; event->value = value; mutex_lock(&event_lock); if (list_empty(&event_list)) { wake_up_interruptible(&event_waitq); } if (list_is_singular(&event_list)) { if (event_task) { wake_up_process(event_task); } } list_add_tail(&event->list, &event_list); if (list_is_full(&event_list)) { if (queue_task) { wake_up_process(queue_task); } } mutex_unlock(&event_lock); return IRQ_HANDLED; } static int gpio_event_queue_thread(void *data) { while (!kthread_should_stop()) { mutex_lock(&event_lock); if (list_is_full(&event_list)) { struct gpio_event *event = list_first_entry(&event_list, struct gpio_event, list); list_del(&event->list); // 处理 GPIO 中断事件 printk(KERN_INFO "Discarded GPIO %d value %d\n", event->gpio, event->value); kfree(event); } mutex_unlock(&event_lock); set_current_state(TASK_INTERRUPTIBLE); schedule(); } return 0; } static int __init gpio_event_init(void) { int ret; int i; printk(KERN_INFO "GPIO event module loaded\n"); events = kmalloc_array(MAX_QUEUE_LEN, sizeof(*events), GFP_KERNEL); if (!events) { printk(KERN_ERR "Failed to allocate gpio event array\n"); return -ENOMEM; } INIT_LIST_HEAD(&event_list); event_task = kthread_run(gpio_event_thread, NULL, "gpio_event"); if (!event_task) { printk(KERN_ERR "Failed to create gpio event thread\n"); ret = -ENOMEM; goto err_free_events; } queue_task = kthread_run(gpio_event_queue_thread, NULL, "gpio_queue"); if (!queue_task) { printk(KERN_ERR "Failed to create gpio queue thread\n"); ret = -ENOMEM; goto err_stop_event_thread; } for (i = 0; i < MAX_QUEUE_LEN; i++) { events[i].gpio = -1; } for (i = 0; i < NUM_GPIO_CHIPS; i++) { struct gpio_chip *chip = &gpio_chip[i]; if (chip->base <= chip->ngpio) { int j; for (j = 0; j < chip->ngpio; j++) { int gpio = chip->base + j; if (gpio_is_valid(gpio)) { ret = gpio_request_one(gpio, GPIOF_IN, "gpio_event"); if (ret) { printk(KERN_ERR "Failed to request gpio %d\n", gpio); goto err_free_gpio; } ret = request_irq(gpio_to_irq(gpio), gpio_event_handler, IRQF_TRIGGER_RISING | IRQF_TRIGGER_FALLING, "gpio_event", (void *)(unsigned long)gpio); if (ret) { printk(KERN_ERR "Failed to request irq for gpio %d\n", gpio); gpio_free(gpio); goto err_free_gpio; } } } } } return 0; err_free_gpio: for (i = 0; i < NUM_GPIO_CHIPS; i++) { struct gpio_chip *chip = &gpio_chip[i]; if (chip->base <= chip->ngpio) { int j; for (j = 0; j < chip->ngpio; j++) { int gpio = chip->base + j; if (gpio_is_valid(gpio)) { free_irq(gpio_to_irq(gpio), (void *)(unsigned long)gpio); gpio_free(gpio); } } } } kthread_stop(queue_task); err_stop_event_thread: kthread_stop(event_task); err_free_events: kfree(events); return ret; } static void __exit gpio_event_exit(void) { int i; printk(KERN_INFO "GPIO event module unloaded\n"); for (i = 0; i < NUM_GPIO_CHIPS; i++) { struct gpio_chip *chip = &gpio_chip[i]; if (chip->base <= chip->ngpio) { int j; for (j = 0; j < chip->ngpio; j++) { int gpio = chip->base + j; if (gpio_is_valid(gpio)) { free_irq(gpio_to_irq(gpio), (void *)(unsigned long)gpio); gpio_free(gpio); } } } } kthread_stop(queue_task); kthread_stop(event_task); kfree(events); } module_init(gpio_event_init); module_exit(gpio_event_exit); MODULE_LICENSE("GPL"); ``` 在上面的示例代码中,我们使用了一个事件队列 `event_list`,用于存储 GPIO 中断事件。当一个 GPIO 中断事件发生时,我们将事件信息保存到队列中,并唤醒等待事件的线程 `event_task`。如果队列已满,我们将丢弃最早的事件,并唤醒等待队列空闲的线程 `queue_task`。线程 `event_task` 会从队列中取出事件并处理,线程 `queue_task` 则负责丢弃队列中最早的事件。这样,我们就实现了 GPIO 中断事件的异步处理,并且可以通过消息队列解耦原有的功能函数。

相关推荐

最新推荐

recommend-type

linux内核4.7版本设备驱动介绍

linux内核4.7版本设备驱动介绍,包含如下模块:class、pci、platform、pinctrl、kset、kobject、bus、device、device_driver、i2c、tty、regmap、misc、spin_lock、inode、mutex、cdev、gpio_keys、usb
recommend-type

linux GPIO中断使用说明 V1.02.pdf

Linux GPIO 中断是指在 Linux 操作系统中使用通用输入/输出(GPIO) pins 来实现中断功能。GPIO 中断是指当 GPIO 引脚的电平状态发生变化时,产生中断请求,触发相应的中断服务程序。 Title 1: GPIO 中断原理 GPIO...
recommend-type

LEDS-GPIO驱动中遇到的问题总结

LEDs-GPIO驱动中遇到的问题总结 本文总结了LEDS-GPIO驱动中遇到的问题,涵盖了unlocked_ioctl与ioctl的区别、miscdevice函数、GPX三个控制寄存器等知识点。 1. #ifdef 语句 #ifdef 语句可以翻译为:如果宏定义了...
recommend-type

Zynq正确的使用GPIO

在 Zynq 的开发中,有两种 GPIO,一种是 Zynq 自带的外设(MIO/EMIO),存在于 PS 中,第二种是 PL 中加入的 AXI_GPIO IP 核。这两种 GPIO 的使用方法和实现机制不同,本文将详细介绍如何正确地使用这两种 GPIO。 ...
recommend-type

使用GPIO模拟实现IIC Slave的方法及source code_1.docx

在本文档中,我们将介绍如何使用GPIO模拟实现IIC Slave。IIC Slave是I2C总线上的从机设备,它可以接收来自主机的命令和数据。为了实现IIC Slave,我们需要了解I2C信号的特点,包括START、STOP、ACK、NACK和接收数据...
recommend-type

基于嵌入式ARMLinux的播放器的设计与实现 word格式.doc

本文主要探讨了基于嵌入式ARM-Linux的播放器的设计与实现。在当前PC时代,随着嵌入式技术的快速发展,对高效、便携的多媒体设备的需求日益增长。作者首先深入剖析了ARM体系结构,特别是针对ARM9微处理器的特性,探讨了如何构建适用于嵌入式系统的嵌入式Linux操作系统。这个过程包括设置交叉编译环境,优化引导装载程序,成功移植了嵌入式Linux内核,并创建了适合S3C2410开发板的根文件系统。 在考虑到嵌入式系统硬件资源有限的特点,通常的PC机图形用户界面(GUI)无法直接应用。因此,作者选择了轻量级的Minigui作为研究对象,对其实体架构进行了研究,并将其移植到S3C2410开发板上,实现了嵌入式图形用户界面,使得系统具有简洁而易用的操作界面,提升了用户体验。 文章的核心部分是将通用媒体播放器Mplayer移植到S3C2410开发板上。针对嵌入式环境中的音频输出问题,作者针对性地解决了Mplayer播放音频时可能出现的不稳定性,实现了音乐和视频的无缝播放,打造了一个完整的嵌入式多媒体播放解决方案。 论文最后部分对整个项目进行了总结,强调了在嵌入式ARM-Linux平台上设计播放器所取得的成果,同时也指出了一些待改进和完善的方面,如系统性能优化、兼容性提升以及可能的扩展功能等。关键词包括嵌入式ARM-Linux、S3C2410芯片、Mplayer多媒体播放器、图形用户界面(GUI)以及Minigui等,这些都反映出本文研究的重点和领域。 通过这篇论文,读者不仅能了解到嵌入式系统与Linux平台结合的具体实践,还能学到如何在资源受限的环境中设计和优化多媒体播放器,为嵌入式技术在多媒体应用领域的进一步发展提供了有价值的经验和参考。
recommend-type

管理建模和仿真的文件

管理Boualem Benatallah引用此版本:布阿利姆·贝纳塔拉。管理建模和仿真。约瑟夫-傅立叶大学-格勒诺布尔第一大学,1996年。法语。NNT:电话:00345357HAL ID:电话:00345357https://theses.hal.science/tel-003453572008年12月9日提交HAL是一个多学科的开放存取档案馆,用于存放和传播科学研究论文,无论它们是否被公开。论文可以来自法国或国外的教学和研究机构,也可以来自公共或私人研究中心。L’archive ouverte pluridisciplinaire
recommend-type

Python字符串为空判断的动手实践:通过示例掌握技巧

![Python字符串为空判断的动手实践:通过示例掌握技巧](https://img-blog.csdnimg.cn/72f88d4fc1164d6c8b9c29d8ab5ed75c.png?x-oss-process=image/watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBASGFyYm9yIExhdQ==,size_20,color_FFFFFF,t_70,g_se,x_16) # 1. Python字符串为空判断的基础理论 字符串为空判断是Python编程中一项基本且重要的任务。它涉及检查字符串是否为空(不包含任何字符),这在
recommend-type

box-sizing: border-box;作用是?

`box-sizing: border-box;` 是 CSS 中的一个样式属性,它改变了元素的盒模型行为。默认情况下,浏览器会计算元素内容区域(content)、内边距(padding)和边框(border)的总尺寸,也就是所谓的"标准盒模型"。而当设置为 `box-sizing: border-box;` 后,元素的总宽度和高度会包括内容、内边距和边框的总空间,这样就使得开发者更容易控制元素的实际布局大小。 具体来说,这意味着: 1. 内容区域的宽度和高度不会因为添加内边距或边框而自动扩展。 2. 边框和内边距会从元素的总尺寸中减去,而不是从内容区域开始计算。
recommend-type

经典:大学答辩通过_基于ARM微处理器的嵌入式指纹识别系统设计.pdf

本文主要探讨的是"经典:大学答辩通过_基于ARM微处理器的嵌入式指纹识别系统设计.pdf",该研究专注于嵌入式指纹识别技术在实际应用中的设计和实现。嵌入式指纹识别系统因其独特的优势——无需外部设备支持,便能独立完成指纹识别任务,正逐渐成为现代安全领域的重要组成部分。 在技术背景部分,文章指出指纹的独特性(图案、断点和交叉点的独一无二性)使其在生物特征认证中具有很高的可靠性。指纹识别技术发展迅速,不仅应用于小型设备如手机或门禁系统,也扩展到大型数据库系统,如连接个人电脑的桌面应用。然而,桌面应用受限于必须连接到计算机的条件,嵌入式系统的出现则提供了更为灵活和便捷的解决方案。 为了实现嵌入式指纹识别,研究者首先构建了一个专门的开发平台。硬件方面,详细讨论了电源电路、复位电路以及JTAG调试接口电路的设计和实现,这些都是确保系统稳定运行的基础。在软件层面,重点研究了如何在ARM芯片上移植嵌入式操作系统uC/OS-II,这是一种实时操作系统,能够有效地处理指纹识别系统的实时任务。此外,还涉及到了嵌入式TCP/IP协议栈的开发,这是实现系统间通信的关键,使得系统能够将采集的指纹数据传输到远程服务器进行比对。 关键词包括:指纹识别、嵌入式系统、实时操作系统uC/OS-II、TCP/IP协议栈。这些关键词表明了论文的核心内容和研究焦点,即围绕着如何在嵌入式环境中高效、准确地实现指纹识别功能,以及与外部网络的无缝连接。 这篇论文不仅深入解析了嵌入式指纹识别系统的硬件架构和软件策略,而且还展示了如何通过结合嵌入式技术和先进操作系统来提升系统的性能和安全性,为未来嵌入式指纹识别技术的实际应用提供了有价值的研究成果。