编写LED灯的驱动,使用GPIO子系统,里面添加按键的中断处理

news/2025/2/5 0:52:49/

编写LED灯的驱动,使用GPIO子系统,里面添加按键的中断处理
1.应用程序发送指令控制LED亮灭
2.按键1按下,led1电位反转按键2按下,led2电位反转按键3按下,led3电位反转

内核代码

#include <linux/init.h>
#include <linux/module.h>
#include <linux/of.h>
#include <linux/of_gpio.h>
#include <linux/gpio.h>
#include <linux/interrupt.h>
#include <linux/of_irq.h>
struct device_node *dev_irq1;
struct device_node *dev_irq2;
struct device_node *dev_irq3;
struct device_node *dev_led1;
struct device_node *dev_led2;
struct device_node *dev_led3;
struct gpio_desc *gpio1no;
struct gpio_desc *gpio2no;
struct gpio_desc *gpio3no;
struct timer_list mytimer;
unsigned int irqno1,irqno2,irqno3;void mytimer_func(struct timer_list *timer)
{}
irqreturn_t myirq1_handler(int irq,void *dev)
{gpiod_set_value(gpio1no,!gpiod_get_value(gpio1no));return IRQ_HANDLED;
}
irqreturn_t myirq2_handler(int irq,void *dev)
{gpiod_set_value(gpio2no,!gpiod_get_value(gpio1no));return IRQ_HANDLED;
}
irqreturn_t myirq3_handler(int irq,void *dev)
{gpiod_set_value(gpio3no,!gpiod_get_value(gpio1no));return IRQ_HANDLED;
}
int major;
char kbuf[128]={0};unsigned int *vrcc;
int mycdev_open(struct inode *inode, struct file *file)
{printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);return 0;
}
ssize_t mycdev_read(struct file *file, char  *ubuf, size_t size, loff_t *lof)
{printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);unsigned long ret;//向用户空间读取拷贝if(size>sizeof(kbuf))//用户空间期待读取的大小内核满足不了,那就给内核支持的最大大小size=sizeof(kbuf);ret=copy_to_user(ubuf,kbuf,size);if(ret)//拷贝失败{printk("copy_to_user filed\n");return ret;}return 0;
}
ssize_t mycdev_write(struct file *file, const char  *ubuf, size_t size, loff_t *lof)
{unsigned long ret;//从用户空间读取数据if(size>sizeof(kbuf))//用户空间期待读取的大小内核满足不了,那就给内核支持的最大大小size=sizeof(kbuf);ret=copy_from_user(kbuf,ubuf,size);if(ret)//拷贝失败{printk("copy_to_user filed\n");return ret;}switch(kbuf[0]){case '1':if(kbuf[1]=='0')//关灯{gpiod_set_value(gpio1no,1);}else{gpiod_set_value(gpio1no,0);}break;case '2':if(kbuf[1]=='0')//关灯{gpiod_set_value(gpio2no,1);}else{gpiod_set_value(gpio2no,0);}break;case '3':if(kbuf[1]=='0')//关灯{gpiod_set_value(gpio3no,1);}else{gpiod_set_value(gpio3no,0);}break;}return 0;
}
int mycdev_close(struct inode *inode, struct file *file)
{printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);return 0;
}//定义操作方法结构体变量并赋值
struct file_operations fops={.open=mycdev_open,.read=mycdev_read,.write=mycdev_write,.release=mycdev_close,
};static int __init mycdev_init(void)
{major=register_chrdev(0,"mycdev",&fops);if(major<0){printk("字符设备驱动注册失败\n");return major;}printk("字符设备驱动注册成功:major=%d\n",major);int ret1,ret2,ret3;dev_led1=of_find_node_by_path("/leds");if(dev_led1==NULL){printk("设备树解析失败\n");return -EFAULT;}printk("解析设备树成功\n");dev_irq1=of_find_node_by_path("/myirq");if(dev_irq1==NULL){printk("设备树解析失败\n");return -EFAULT;}printk("解析设备树成功\n");//led1gpio1no=gpiod_get_from_of_node(dev_led1,"led1-gpios",0,GPIOD_OUT_LOW,NULL);if(IS_ERR(gpio1no)){printk("gpio1对象申请失败\n");return -PTR_ERR(gpio1no);}printk("解析gpio1编号成功\n");//led2gpio2no=gpiod_get_from_of_node(dev_led1,"led2-gpios",0,GPIOD_OUT_LOW,NULL);if(IS_ERR(gpio2no)){printk("gpio2对象申请失败\n");return -PTR_ERR(gpio2no);}printk("解析gpio编号成功\n");//led3gpio3no=gpiod_get_from_of_node(dev_led1,"led3-gpios",0,GPIOD_OUT_LOW,NULL);if(IS_ERR(gpio3no)){printk("gpio1对象申请失败\n");return -PTR_ERR(gpio3no);}printk("解析gpio编号成功\n");//中断1irqno1=irq_of_parse_and_map(dev_irq1,0);if(!irqno1){printk("软中断解析失败\n");return -ENXIO;}printk("软中断解析成功 irqno1=%d\n",irqno1);ret1=request_irq(irqno1,myirq1_handler,IRQF_TRIGGER_FALLING,"key1",NULL);if(ret1){printk("注册中断失败\n");return ret1;}printk("注册中断成功\n");//中断2irqno2=irq_of_parse_and_map(dev_irq1,1);if(!irqno2){printk("软中断解析失败\n");return -ENXIO;}printk("软中断解析成功 irqno2=%d\n",irqno2);ret2=request_irq(irqno2,myirq2_handler,IRQF_TRIGGER_FALLING,"key1",NULL);if(ret2){printk("注册中断失败\n");return ret2;}printk("注册中断成功\n");//中断3irqno3=irq_of_parse_and_map(dev_irq1,2);if(!irqno3){printk("软中断解析失败\n");return -ENXIO;}printk("软中断解析成功 irqno3=%d\n",irqno3);ret3=request_irq(irqno3,myirq3_handler,IRQF_TRIGGER_FALLING,"key1",NULL);if(ret3){printk("注册中断失败\n");return ret3;}printk("注册中断成功\n");mytimer.expires=jiffies+HZ;timer_setup(&mytimer,mytimer_func,0);add_timer(&mytimer);return 0;
}
static void __exit mycdev_exit(void)
{del_timer(&mytimer);gpiod_set_value(gpio1no,0);gpiod_set_value(gpio2no,0);gpiod_set_value(gpio3no,0);gpiod_put(gpio1no);gpiod_put(gpio2no);gpiod_put(gpio3no);free_irq(irqno1,NULL);free_irq(irqno2,NULL);free_irq(irqno3,NULL);}
module_init(mycdev_init);
module_exit(mycdev_exit);
MODULE_LICENSE("GPL");

应用

#include<stdlib.h>
#include<stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include<unistd.h>
#include<string.h>
#include "/home/ubuntu/DC22111/char/head.h"
#include <sys/ioctl.h>int main(int argc, char const *argv[])
{char buf[128]={0};int fd=open("/dev/mycdev",O_RDWR);if(fd<0){printf("打开设备文件失败\n");exit(-1);}int a,b,c;while(1){//从终端读取printf("请输入1(灯光))>");scanf("%d",&a);if(a==1){printf("请输入功能:1(开灯),2(关灯)");scanf("%d",&b);if(b==1){printf("请输入要操作的灯光1,2,3>");scanf("%d",&c);ioctl(fd,LED_ON,&c);}else{printf("请输入要操作的灯光1,2,3>");scanf("%d",&c);ioctl(fd,LED_OFF,&c);}}}close(fd);return 0;
}


http://www.ppmy.cn/news/997741.html

相关文章

Socket本质、实战演示两个进程建立TCP连接通信的过程

文章目录 Socket是什么引入面试题, 使你更深刻的理解四元组 Socket网络通信大体流程实战演示TCP连接建立过程需要用到的linux 查看网络的一些命令测试的程序一些准备工作启动服务端, 并没有调用accept启动客户端开启服务accept Socket是什么 通俗来说,Socket是套接字,是一种编…

每日一题——旋转数组的最小数字

题目 有一个长度为 n 的非降序数组&#xff0c;比如[1,2,3,4,5]&#xff0c;将它进行旋转&#xff0c;即把一个数组最开始的若干个元素搬到数组的末尾&#xff0c;变成一个旋转数组&#xff0c;比如变成了[3,4,5,1,2]&#xff0c;或者[4,5,1,2,3]这样的。请问&#xff0c;给定这…

YOLOv8+DeepSORT多目标跟踪(行人车辆计数与越界识别)视频教程

课程链接&#xff1a;https://edu.csdn.net/course/detail/38870 本课程使用YOLOv8和DeepSORT对视频中的行人、车辆做多目标跟踪计数与越界识别&#xff0c;开展YOLOv8目标检测和DeepSORT多目标跟踪强强联手的应用。 课程分别在Windows和Ubuntu系统上做项目演示&#xff0c;并…

pycharm制作柱状图

Bar - Bar_rotate_xaxis_label 解决标签名字过长的问题 from pyecharts import options as opts from pyecharts.charts import Barc (Bar().add_xaxis(["高等数学1&#xff0c;2","C语言程序设计","python程序设计","大数据导论",…

Ubuntu安装JDK与IntelliJ IDEA

目录 前言 Ubuntu 安装 JDK 1、更新软件包列表 2、安装OpenJDK 3、验证安装 Ubuntu安装IntelliJ IDEA 1、下载 IntelliJ IDEA 2、解压缩 IntelliJ IDEA 安装包 3、移动 IntelliJ IDEA 到安装目录 4、启动 IntelliJ IDEA 前言 APT&#xff08;Advanced Package Tool&…

MySQL踩坑之sql_mode的用法

目录 定义 报错重现 ​编辑 原因分析 sql_mode值说明 查看当前sql_mode 设置sql_mode 定义 什么是sql_mode?玩了这么久的MySQL语句࿰

记录对 require.js 的理解

目录 一、使用 require.js 主要是为了解决这两个问题二、require.js 的加载三、main.js 一、使用 require.js 主要是为了解决这两个问题 实现 js 文件的异步加载&#xff0c;避免网页失去响应&#xff1b;管理模块之间的依赖性&#xff0c;便于代码的编写和维护。 二、require.…

openGauss学习笔记-27 openGauss 高级数据管理- JOIN

文章目录 openGauss学习笔记-27 openGauss 高级数据管理- JOIN27.1 交叉连接27.2 内连接27.3 左外连接27.4 右外连接27.5 全外连接 openGauss学习笔记-27 openGauss 高级数据管理- JOIN JOIN子句用于把来自两个或多个表的行结合起来&#xff0c;基于这些表之间的共同字段。 在…