Linux内核驱动学习(三)字符型设备驱动之初体验
Linux字符型设备驱动之初体验
文章目录
前言
驱动总共分为字符型设备驱动,块设备驱动,网络设备驱动。对于字符型设备驱动的资料,网上比较多,《Linux Kernel Driver》这本书可以了解一下,对于学习Linux驱动有很大的帮助,当然还有很多优秀的书籍,暂不一一列举,本文简单总结了在学习字符型设备驱动的过程中遇到的问题,以及对该类驱动的理解。
框架
字符型设备
什么是字符型设备?字符型以字符(Byte/Char)为单位进行数据传输的设备,如键盘,串口等等设备,所以Linux环境编程中文件I/O进行操作的系统接口如open
,read
,write
,close
等等,在字符型设备驱动中同样需要支持这些接口。这里会用到file_operations
结构体,在后面会讲到。
程序实现
下面是一个简单字符型设备驱动程序,可以在系统注册一个字符型设备驱动,目前未实现open
,read
,write
,close
等接口。
#include <linux/init.h>
#include <linux/module.h>
#include <linux/cdev.h>
#include <linux/fs.h>
#include <linux/slab.h>
#define DRIVER_DATA_SIZE 4096
static int major_dev_index = 0;
struct cnc_character_st{
struct cdev device;
u8 data[DRIVER_DATA_SIZE];
};
static struct cnc_character_st *character_dev;
//TODO
static ssize_t cnc_character_read (struct file * fd, char __user * data, size_t len, loff_t * offset){
ssize_t ret = 0;
printk("%s call\n",__func__);
return ret;
}
//TODO
static ssize_t cnc_character_write (struct file * fd, const char __user * data, size_t len, loff_t * offset){
ssize_t ret = 0;
return ret;
}
//TODO
static long cnc_character_unlocked_ioctl (struct file * fd, unsigned int data, unsigned long cmd){
long ret = 0;
return ret;
}
//TODO
static int cnc_character_open (struct inode * node, struct file * fd){
int ret = 0;
return ret;
}
//TODO
static int cnc_character_release (struct inode * node, struct file * fd){
int ret = 0;
return ret;
}
static const struct file_operations cnc_character_ops = {
.owner = THIS_MODULE,
.read = cnc_character_read,
.write = cnc_character_write,
.open = cnc_character_open,
.unlocked_ioctl = cnc_character_unlocked_ioctl,
.release = cnc_character_release,
};
static int register_device(struct cnc_character_st *mdev,int major_dev_index,int minor_dev_index){
int ret = 0;
int dev_no = MKDEV(major_dev_index, minor_dev_index);
// 初始化dev
cdev_init(&mdev->device, &cnc_character_ops);
mdev->device.owner = THIS_MODULE;
ret = cdev_add(&mdev->device,dev_no,1);
if(ret){
printk(KERN_ERR "cdev add device failed\n");
}
return ret;
}
static int unregister_device(struct cnc_character_st *mdev){
int ret= 0;
kfree(character_dev);
return ret;
}
static int __init cnc_character_init(void){
int ret = 0;
dev_t devno = MKDEV(major_dev_index, 0);
if(major_dev_index){
ret = register_chrdev_region(devno, 1, "cnc_character");
}else{
ret = alloc_chrdev_region(&devno, 0, 1, "cnc_character");
major_dev_index = MAJOR(devno);
}
if(ret < 0){
return ret;
}
character_dev = kmalloc(sizeof(struct cnc_character_st),GFP_KERNEL);
if(!character_dev){
printk("%s failed malloc character_dev call\n",__func__);
ret = -ENOMEM;
goto failed;
}else{
printk("%s success malloc character_dev call\n",__func__);
}
register_device(character_dev,major_dev_index,0);
return 0;
failed:
unregister_chrdev_region(devno, 1);
return ret;
}
module_init(cnc_character_init);
static void __exit cnc_character_exit(void){
printk("%s call\n",__func__);
unregister_device(character_dev);
}
module_exit(cnc_character_exit);
MODULE_AUTHOR("zhaojunhui@cncgroup.top");
MODULE_VERSION("1.0");
MODULE_LICENSE("GPL");
cdev
在linux/cdev.h
中可以阅读相关字符型设备驱动的信息,其中包括cdev
结构体可以做一下分析,先定位到源码做一下分析
#ifndef _LINUX_CDEV_H
#define _LINUX_CDEV_H
#include <linux/kobject.h>
#include <linux/kdev_t.h>
#include <linux/list.h>
struct file_operations;
struct inode;
struct module;
struct cdev {
struct kobject kobj;
struct module *owner;
const struct file_operations *ops;
struct list_head list;
dev_t dev;
unsigned int count;
};
void cdev_init(struct cdev *, const struct file_operations *);
struct cdev *cdev_alloc(void);
void cdev_put(struct cdev *p);
int cdev_add(struct cdev *, dev_t, unsigned);
void cdev_del(struct cdev *);
void cd_forget(struct inode *);
#endif
其中包括结构体cdev
和cdev的一系列函数接口cdev_init
,cdev_alloc
,cdev_put
,cdev_add
,cdev_del
,cd_forget
。
kobj
kobject
是所有设备驱动模型的基类,而cdev
可以理解为是它的派生类,这里使用了面向对象的思想,通过访问cdev
中的kobj
成员,就能使用kobject
中所有功能。关于kobject
的详细内容可以参考内核文档Documentation/kobject.txt
。
owner
首先明确一点的是owner
是struct module
的指针变量,owner=THIS_MODULE;
,这里将指针指向当前的模块,关于THIS_MODULE
以及struct module
的知识可以参考这篇博客。
file_operations
这个结构体位于/linux/include/fs.h
,代码如下。
struct file_operations {
struct module *owner;
loff_t (*llseek) (struct file *, loff_t, int);
ssize_t (*read) (struct file *, char __user *, size_t, loff_t *);
ssize_t (*write) (struct file *, const char __user *, size_t, loff_t *);
ssize_t (*read_iter) (struct kiocb *, struct iov_iter *);
ssize_t (*write_iter) (struct kiocb *, struct iov_iter *);
int (*iterate) (struct file *, struct dir_context *);
unsigned int (*poll) (struct file *, struct poll_table_struct *);
long (*unlocked_ioctl) (struct file *, unsigned int, unsigned long);
long (*compat_ioctl) (struct file *, unsigned int, unsigned long);
int (*mmap) (struct file *, struct vm_area_struct *);
int (*open) (struct inode *, struct file *);
int (*flush) (struct file *, fl_owner_t id);
int (*release) (struct inode *, struct file *);
int (*fsync) (struct file *, loff_t, loff_t, int datasync);
int (*aio_fsync) (struct kiocb *, int datasync);
int (*fasync) (int, struct file *, int);
int (*lock) (struct file *, int, struct file_lock *);
ssize_t (*sendpage) (struct file *, struct page *, int, size_t, loff_t *, int);
unsigned long (*get_unmapped_area)(struct file *, unsigned long, unsigned long, unsigned long, unsigned long);
int (*check_flags)(int);
int (*flock) (struct file *, int, struct file_lock *);
ssize_t (*splice_write)(struct pipe_inode_info *, struct file *, loff_t *, size_t, unsigned int);
ssize_t (*splice_read)(struct file *, loff_t *, struct pipe_inode_info *, size_t, unsigned int);
int (*setlease)(struct file *, long, struct file_lock **, void **);
long (*fallocate)(struct file *file, int mode, loff_t offset,
loff_t len);
void (*show_fdinfo)(struct seq_file *m, struct file *f);
#ifndef CONFIG_MMU
unsigned (*mmap_capabilities)(struct file *);
#endif
};
在file_operations
定义了很多I/O操作接口,这里同样使用了面向对象编程的思想,每个接口可以在重新定义file_operations
结构体变量的时候,重新赋于自定义功能的函数,如下,可以理解read
,write
,open
,unlocked_ioctl
,release
是对抽象函数的实现。
static const struct file_operations cnc_character_ops = {
.owner = THIS_MODULE,
.read = cnc_character_read,
.write = cnc_character_write,
.open = cnc_character_open,
.unlocked_ioctl = cnc_character_unlocked_ioctl,
.release = cnc_character_release,
};
dev_t
设备注册过程
设备的初始化在函数cnc_character_init
中完成具体的功能实现,主要分为两个部分,设备号的申请和设备的注册。其中设备注册单独封装到register_device
函数中。
申请设备号
dev_t devno = MKDEV(major_dev_index, 0);
if(major_dev_index){
ret = register_chrdev_region(devno, 1, "cnc_character");
}else{
ret = alloc_chrdev_region(&devno, 0, 1, "cnc_character");
major_dev_index = MAJOR(devno);
}
注册设备
character_dev = kmalloc(sizeof(struct cnc_character_st),GFP_KERNEL);
if(!character_dev){
printk("%s failed malloc character_dev call\n",__func__);
ret = -ENOMEM;
goto failed;
}else{
printk("%s success malloc character_dev call\n",__func__);
}
register_device(character_dev,major_dev_index,0);
register_device
在register_device
中,主要用到了cdev
提供的函数接口。
cdev_init
初始化一个字符型设备并传入自定义的file_operations
类型变量cnc_character_ops
。
cdev_add
将初始化的字符型设备添加到内核,并分配已经申请好的设备号。
static int register_device(struct cnc_character_st *mdev,int major_dev_index,int minor_dev_index){
int ret = 0;
int dev_no = MKDEV(major_dev_index, minor_dev_index);
// 初始化dev
cdev_init(&mdev->device, &cnc_character_ops);
mdev->device.owner = THIS_MODULE;
ret = cdev_add(&mdev->device,dev_no,1);
if(ret){
printk(KERN_ERR "cdev add device failed\n");
}
return ret;
}
如何构建
模块编译
使用这个Makefile
KVERS = $(shell uname -r)
# Kernel modules
obj-m += demo_character.o
# Specify flags for the module compilation.
EXTRA_CFLAGS=-g -O0
build: kernel_modules
kernel_modules:
make -C /lib/modules/$(KVERS)/build M=$(CURDIR) modules
clean:
make -C /lib/modules/$(KVERS)/build M=$(CURDIR) clean
内核编译
Makefile
obj-$(CONFIG_DEMO_CHARACTER_DRIVER) +=demo_character.o
Kconfig
menuconfig DEMO_DRIVERS
tristate "demo drivers"
config DEMO_CHARACTER_DRIVER
tristate "the most simplest character driver"
help
character driver
endif
总结
总体上来说,字符型设备驱动框架还是相对简单的,通过这次学习加深了对cdev
的认识和linux内核源码中面向对象的设计思想,但是这里还没有对devfs
和sysfs
做相应的介绍,后面继续学习这两者的区别以及总线驱动模型,总之,加油吧。
参考
https://blog.csdn.net/lucky_liuxiang/article/details/83413946
https://www.cnblogs.com/helloahui/p/3677192.html
https://blog.csdn.net/jk110333/article/details/8563647
Linux内核驱动学习(三)字符型设备驱动之初体验的更多相关文章
- 嵌入式Linux学习笔记(三) 字符型设备驱动--LED的驱动开发
在成功构建了一个能够运行在开发板平台的系统后,下一步就要正式开始应用的开发(这里前提是有一定的C语言基础,对ARM体系的软/硬件,这部分有疑问可能要参考其它教程),根据需求仔细分解任务,可以发现包含的 ...
- linux 块设备驱动 (三)块设备驱动开发
一: 块设备驱动注册与注销 块设备驱动中的第1个工作通常是注册它们自己到内核,完成这个任务的函数是 register_blkdev(),其原型为:int register_blkdev(unsigne ...
- Linux设备驱动开发基础--阻塞型设备驱动
1. 当一个设备无法立刻满足用户的读写请求时(例如调用read时,设备没有数据提供),驱动程序应当(缺省的)阻塞进程,使它进入等待(睡眠)状态,知道请求可以得到满足. 2. Linux内核等待队列:在 ...
- 字符型设备驱动程序-first-printf以及点亮LED灯(一)
学习使用 Linux 的 字符型设备驱动 来 进行 . 学习地址:http://edu.51cto.com/lesson/id-25710.html 第一步: 首先写 三个函数 ,2017年5月17 ...
- linux设备驱动归纳总结(三):1.字符型设备之设备申请【转】
本文转载自:http://blog.chinaunix.net/uid-25014876-id-59416.html linux设备驱动归纳总结(三):1.字符型设备之设备申请 操作系统:Ubunru ...
- 【Linux开发】linux设备驱动归纳总结(三):1.字符型设备之设备申请
linux设备驱动归纳总结(三):1.字符型设备之设备申请 操作系统:Ubunru 10.04 实验平台:S3C2440 + linux2.6.29内核 注:在今后驱动程序的学习中经常需要查看内核源代 ...
- linux设备驱动归纳总结(三):2.字符型设备的操作open、close、read、write【转】
本文转载自:http://blog.chinaunix.net/uid-25014876-id-59417.html linux设备驱动归纳总结(三):2.字符型设备的操作open.close.rea ...
- 【Linux开发】linux设备驱动归纳总结(三):2.字符型设备的操作open、close、read、write
linux设备驱动归纳总结(三):2.字符型设备的操作open.close.read.write 一.文件操作结构体file_operations 继续上次没讲完的问题,文件操作结构体到底是什么东西, ...
- 字符型设备驱动程序-first-printf以及点亮LED灯(三)
根据 字符型设备驱动程序-first-printf以及点亮LED灯(二) 学习 修改函数 中的printf 为 printk. #include <linux/module.h> /* ...
随机推荐
- React AntDesign 引入css
React项目是用umi脚手架搭建的AntDesign,用到一个第三方表格组件Jexcel,npm install 之后组件的样式加载不上,犯了愁,翻阅各种资料,踏平两个小坑. 大家都知道,安装完成的 ...
- JSP中引用CSS样式文件却无法显示的问题解决方案
你也遇到过这种问题吗,CSS写好了,JSP写好了,在JSP中调用CSS文件,路径检查后也正确,但是无法显示渲染后的页面 原因:罪魁祸首就是过滤器响应数据的时候,响应头设置为了“text/html”,但 ...
- python 异步Web框架sanic
我们继续学习Python异步编程,这里将介绍异步Web框架sanic,为什么不是tornado?从框架的易用性来说,Flask要远远比tornado简单,可惜flask不支持异步,而sanic就是类似 ...
- deepin15.11小毛病解决
目录 边缘花屏问题 QQ`Tim头像问题 ssh卡死问题 看直播卡 边缘花屏问题 sudo apt install systemsettings 打开kde系统设置 打开显示与设置,修改如图下,基本上 ...
- Mac home 目录下创建文件夹
example:sudo vim /etc/auto_master before: # Automounter master map +auto_master # Use directory serv ...
- 2019-2020-1 20199328《Linux内核原理与分析》第六周作业
使用gdb跟踪分析一个系统调用内核函数 首先我们删除本身的menu目录,并从github上克隆一个menu,并进行编译 编译过程 现在找到test.c文件,加入上个实验中做的getPid()方法 利用 ...
- [Inno Setup] 退出安装程序的两种方式
1. 完全静默的退出 procedure ExitProcess(exitCode:integer); external 'ExitProcess@kernel32.dll stdcall'; ... ...
- FluentAspects -- 基于 Fluent API 的 Aop
FluentAspects -- 基于 Fluent API 的 Aop Intro 上次我们做了一个简单的 AOP 实现示例,但是实现起来主要是基于 Attribute 来做的,对于代码的侵入性太强 ...
- web前端开发中的各种居中
居中是我们使用css来布局时常遇到的情况.使用css来进行居中时,有时一个属性就能搞定,有时则需要一定的技巧才能兼容到所有浏览器,本文就居中的一些常用方法做个简单的介绍. 注:本文所讲方法除了特别说明 ...
- Netty(七):EventLoop学习前导——Reactor模式
了解Netty的人多少都会知道Netty的高性能的一个原因就是它是基于事件驱动的,而这一事件的原型就是Reactor模式. 所以在学习EventLoop前,很有必要先搞懂Reactor模式. 本文目录 ...