手把手带你写一个中断输入设备驱动~
今天来分享一下以前写一个中断输入设备驱动案例,希望对有需要的朋友能有所帮助。
背景介绍
在一个项目中,有这样一个需求:
主控芯片采用ZYNQ,需要采集外部一个脉冲编码输入信号,这个信号是一个脉冲波形,脉冲数量代表测量结果。比如这有可能是一个电机的霍尔信号输出,代表电机的转速,也有可能是一个光栅编码器的脉冲输出,是什么并不重要。
这个电路本身,利用光耦实现了输入测设备信号与采集端的电气隔离。由于PS端该Bank的电平为3.3V,所以光耦的另一侧也是3.3V。
ZYNQ的PS端运行Linux程序,所以在这个场景下,要从应用程序的角度将外部输入信号用起来,就需要实现这样一个设备驱动程序:
创建设备
在ZYNQ下,使用petalinux工具链,当然本文中对于写这个驱动程序本身换成其他的处理器从代码的角度是类似的。
1.先运行一下工具链环境变量脚本:
source /opt/pkg/petalinux/settings.sh
当然也可以不用手动这样运行,设置成linux开发主机开机自动运行,这里就不赘述怎么设置了,网上很多介绍。
2.创建设备
petalinux-create -t modules --name di-drv
这样在现有的工程下,就自动创建设备文件:
./project-spec/meta-user/recipes-modules/di-drv/files/di-drv.c
修改设备树
./project-spec/meta-user/recipes-bsp/device-tree/files/system-user.dtsi
中添加
/include/ "system-conf.dtsi"
/ { amba {pinctrl_di_default: di-default { mux { groups = "gpio0_0_grp"; function = "gpio0"; }; conf { pins = "MIO0"; io-standard = <1>; bias-high-impedance; slew-rate = <0>; }; }; };di {compatible = "di-drv";pinctrl-names = "default";pinctrl-0 = <&pinctrl_di_default>;di-gpios = <&gpio0 0 0>; };
};
本文中,假定使用的IO引脚为PS_MIO0。
驱动代码
修改上面生成的代码di-drv.c
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/init.h>
#include <linux/ide.h>
#include <linux/types.h>
#include <linux/errno.h>
#include <linux/cdev.h>
#include <linux/of.h>
#include <linux/of_address.h>
#include <linux/of_gpio.h>
#include <linux/device.h>
#include <linux/delay.h>
#include <linux/init.h>
#include <linux/gpio.h>
#include <linux/semaphore.h>
#include <linux/timer.h>
#include <linux/of_irq.h>
#include <linux/irq.h>
#include <asm/uaccess.h>
#include <asm/mach/map.h>
#include <asm/io.h>/* 设备节点名称 */
#define DEVICE_NAME "di-drv"
/* 设备号个数 */
#define DEVID_COUNT 1
/* 驱动个数 */
#define DRIVE_COUNT 1
/* 主设备号 */
#define MAJOR_U
/* 次设备号 */
#define MINOR_U 0struct di_dev {/* 字符设备框架 */dev_t devid; //设备号struct cdev cdev; //字符设备struct class *class; //类struct device *device; //设备struct device_node *nd; //设备树的设备节点spinlock_t lock; //自旋锁变量int di_gpio; //DI gpio号__u32 di_pulses;//DI counter unsigned int di_irq; //DI 中断号
};static struct di_dev di_char = {.cdev = {.owner = THIS_MODULE,},
};/* 中断服务函数 */
static irqreturn_t di_handler(int irq, void *dev)
{di_char.di_pulses++;return IRQ_RETVAL(IRQ_HANDLED);
}/* open函数实现, 对应到Linux系统调用函数的open函数 */
static int di_drv_open(struct inode *inode_p, struct file *file_p)
{ printk("di_drv module opened\n"); file_p->private_data = &di_char; return 0;
} /* read函数实现, 对应到Linux系统调用函数的read操作 */
static ssize_t di_drv_read(struct file *file_p, char __user *buf, size_t len, loff_t *loff_t_p)
{ unsigned long flags;int ret;union e_int_conv{__u8 buf[8];__u32 di_raw;}; /* 获取锁 */spin_lock_irqsave(&di_char.lock, flags);union e_int_conv di;di.di_raw.di = di_char.di_pulses;ret = copy_to_user(buf, di.buf, 8);/* 释放锁 */spin_unlock_irqrestore(&di_char.lock, flags);return ret ? ret : 4;
} /* release函数实现, 对应到Linux系统调用函数的close函数 */
static int di_drv_release(struct inode *inode_p, struct file *file_p)
{ printk("di_drv module release\n");return 0;
} /* file_operations结构体声明 */
static struct file_operations di_fops = { .owner = THIS_MODULE, .open = di_drv_open, .read = di_drv_read, .release = di_drv_release,
}; /* 模块加载时会调用的函数 */
static int __init di_drv_init(void)
{u32 ret = 0;/* 初始化自旋锁 */spin_lock_init(&di_char.lock);/** gpio框架 **/ /* 获取设备节点 */di_char.nd = of_find_node_by_path("/di");if(di_char.nd == NULL){printk("di node not foundr\r\n");return -EINVAL;}/* 获取节点中gpio标号 */di_char.di_gpio = of_get_named_gpio(di_char.nd, "di-gpios", 0);if(di_char.di_gpio < 0){printk("Failed to get di-gpios from device tree\r\n");return -EINVAL;}printk("di-gpio num = %d\r\n", di_char.di_gpio);/* 申请gpio标号对应的引脚 */ret = gpio_request(di_char.di_gpio, "di-drv");if(ret != 0){printk("Failed to request di_gpio\r\n");return -EINVAL;}/* 把这个io设置为输入 */ret = gpio_direction_input(di_char.di_gpio);if(ret < 0){printk("Failed to set di_gpio as input\r\n");return -EINVAL;}/* 获取中断号 */di_char.di_irq = gpio_to_irq(di_char.di_gpio);printk("di_irq number is %d \r\n", di_char.di_irq);/* 申请中断 */ret = request_irq(di_char.di_irq,di_handler,IRQF_TRIGGER_FALLING | IRQF_TRIGGER_RISING,"di-drv", NULL);if(ret < 0){printk("di_irq %d request failed\r\n", di_char.di_irq);return -EFAULT;} /* 注册设备号 */alloc_chrdev_region(&di_char.devid, MINOR_U, DEVID_COUNT, DEVICE_NAME);/* 初始化字符设备结构体 */cdev_init(&di_char.cdev, &di_fops);/* 注册字符设备 */cdev_add(&di_char.cdev, di_char.devid, DRIVE_COUNT);/* 创建类 */di_char.class = class_create(THIS_MODULE, DEVICE_NAME);if(IS_ERR(di_char.class)) {return PTR_ERR(di_char.class);}/* 创建设备节点 */di_char.device = device_create( di_char.class, NULL, di_char.devid, NULL, DEVICE_NAME );if(IS_ERR(di_char.device)) {return PTR_ERR(di_char.device);}di_char.di_pulses = 0; return 0;
}/* 卸载模块 */
static void __exit di_drv_exit(void)
{ /* 释放gpio */gpio_free(di_char.di_gpio);/* 释放中断 */free_irq(di_char.di_irq, NULL);/* 注销字符设备 */cdev_del(&di_char.cdev);/* 注销设备号 */unregister_chrdev_region(di_char.devid, DEVID_COUNT);/* 删除设备节点 */device_destroy(di_char.class, di_char.devid);/* 删除类 */class_destroy(di_char.class);printk("DI dev exit ok\n");
} /* 标记加载、卸载函数 */
module_init(di_drv_init);
module_exit(di_drv_exit); /* 驱动描述信息 */
MODULE_AUTHOR("Embinn");
MODULE_ALIAS("DI input");
MODULE_DESCRIPTION("DIGITAL INPUT driver");
MODULE_VERSION("v1.0");
MODULE_LICENSE("GPL");
这是一个字符驱动的实现,在真实项目中,大部分驱动基本已经被芯片厂商给实现了,但是一些特殊项目的自定义需求,往往就需要去实现自己的驱动。
编译部署
运行以下命令:
petalinux-config -c rootfs
进入modules,使能刚刚创建的模块,退出保存。
运行下面的命令进行编译:
petalinux-build
最终在工程目录下,搜索di-drv.ko,就得到这个驱动的内核模块文件了,拷贝到目标板的某个文件夹下,运行下面的命令装载就完成了:
insmod di-drv.ko
这样在/dev下就会发现新增一个di-drv设备。
当然也可以直接将该驱动放进内核里,这就需要在内核代码树里,添加文件了,这个思路之前有分享过。
总结一下
字符设备是做驱动开发比较容易掌握的驱动类型,也是大多数项目中,需要自己动手写的最多的驱动类型,所以还是应该掌握它才能实现不同的项目需求。至于用户空间怎么访问这个设备,这里就不赘述了,一个文件打开操作,再来一个读取操作就完事了。
1.Arm全新物联网解决方案,缩短2年芯片开发周期,再造一个“智能手机市场”
2.树莓派Zero 2 W正式发布!一起了解一下!
3.一本系统性介绍RISC-V开发技术的新书出版了~
4.多核CPU和单核的区别~
5.2021慕尼黑华南电子展延期通知
6.原来搞单片机也可以面向对象!
免责声明:本文系网络转载,版权归原作者所有。如涉及作品版权问题,请与我们联系,我们将根据您提供的版权证明材料确认版权并支付稿酬或者删除内容。
手把手带你写一个中断输入设备驱动~相关推荐
- 手把手带你写一个中断输入设备驱动
今天群里有人问,要开始驱动开发的话从什么开始比较好. 我说,应该开始去摸索触摸屏驱动,现在我想了下,触摸屏驱动可能会难了些,但是从一个GPIO开始,我觉得一定是一件很容易的事情. 所以这篇文章就来了. ...
- 手把手带你写一个JavaScript类型判断小工具
业务写了很多,依然不是前端大神,我相信这是很多'入坑'前端开发同学的迷茫之处,个人觉得前端职业发展是有路径可寻的,前期写业务是一个积累过程,后期提炼总结,比如编程思想,父子类的原型继承,还是对象之间的 ...
- 【NLP】Pyhon+讯飞开放平台:手把手带你写一个智能语音播报系统
手把手带你写一个智能语音播报系统. 微信扫码登陆讯飞开放平台:https://www.xfyun.cn/ 完成个人认证. 在控制台创建应用,注意应用名称全库查重,很容易跟别人重复. 可查看到pytho ...
- 手把手带你写一个 Vue3 的自定义指令
背景 众所周知,Vue.js 的核心思想是数据驱动 + 组件化,通常我们开发页面的过程就是在编写一些组件,并且通过修改数据的方式来驱动组件的重新渲染.在这个过程中,我们不需要去手动操作 DOM. 然而 ...
- 手把手带你写一个中高级程序员必会的分布式RPC框架
一.概述 什么是RPC? 远程服务调用 官方:一种通过网络从远程计算机程序上请求服务,而不需要了解底层网络技术的思想 通俗一点:客户端在不知道调用细节的情况下,调用存在于远程计算机上的某个对象,就像调 ...
- 转:写一个块设备驱动
----------------------- Page 1----------------------- 第 1章 +---------------------------------------- ...
- Linux驱动开发--写一个块设备驱动
原文地址:[原创] 写一个块设备驱动 http://bbs.chinaunix.net/forum.php?mod=viewthread&tid=2017377&fromuid=288 ...
- 手把手教你写Linux I2C设备驱动
手把手教你写Linux I2C设备驱动 标签:Linux 设备 驱动 详解 i2c 原创作品,允许转载,转载时请务必以超链接形式标明文章 原始出处 .作者信息和本声明.否则将追究法律责任.http:/ ...
- 手把手教你写一个spring IOC容器
本文分享自华为云社区<手把手教你写一个spring IOC容器>,原文作者:技术火炬手. spring框架的基础核心和起点毫无疑问就是IOC,IOC作为spring容器提供的核心技术,成功 ...
最新文章
- SQLAlchemy实现插入数据(使用前端页面)
- git 建立一个私有模块
- pytorch多进程加载数据
- 轻松读懂数据结构系列:早操排队图解选择排序
- wsdl 与 soap协议详解
- 查看android手机原生app包名的方法
- ProgressBar控件在Listview下的多线程应用(转自johngeng)
- Leetcode: LRU Cache
- 筛表合集(素数筛 欧拉函数筛 莫比乌斯函数筛)
- iOS AVPlayer视频播放器
- 商业智能BI有哪些数据价值
- 【HDU2050】折线分割平面
- RS232 RS422 RS485详细介绍
- 计算机函数求奖学金,Excel函数在高校奖学金评定中的实践应用参考.pdf
- 大学python教材实验七字典与集合答案_Python中的字典与集合
- matlab画收敛域,第二章习题与答案1.求以下序列的z变换并画出零极点图和收敛域.doc...
- 王者荣耀英雄分析--孙悟空
- 几行代码实现视图循环旋转,可用于Loading
- Java将html转换成pdf、html转换成图片
- 辨别 利用AAC转成无损格式音乐 的假无损