接着上一篇文章的内容,继续编写我们的led驱动程序。
通过上一篇文章的实验,我们知道内核在加载模块驱动时,会进入的函数是module_init()里面指定的函数,我们称之为入口函数,因此,要向内核注册设备,肯定要在入口函数处完成。
注册字符设备的函数是:
int register_chrdev_region(dev_t from, unsigned count, const char *name);
参数分别是:dev_t 设备号,设备数量,设备名字;
这里有个参数是设备号,Linux系统里面是通过设备号对设备进行访问的,可以看到这里的设备号是一个dev_t类型,查看这个类型的定义,是一个u32类型。找有关于设备号描述的资料,有关于设备号操作定义在include/linux/kdev_t.h文件中。重点关注这几行代码:
#define MINORBITS 20
#define MINORMASK ((1U << MINORBITS) - 1)
#define MAJOR(dev) ((unsigned int) ((dev) >> MINORBITS))
#define MINOR(dev) ((unsigned int) ((dev) & MINORMASK))
#define MKDEV(ma,mi) (((ma) << MINORBITS) | (mi))
设备号分为主设备号和次设备号。主设备号使用的是高12位,次设备号使用的是低32位。在某些场景中,一个驱动程序可能对应多个设备,那么这些设备就可以使用同一个主设备号,不同的设备用次设备号进行区分。比如IIC设备,IIC总线中挂载了多个设备,那么主设备号对应的是IIC驱动,次设备号对应的是具体的设备。这里我们以led为例,只用到了一个设备,所以次设备号设置为0.
那么,该如何确定我们要注册的设备号是什么呢?两种方法:一、手动查询当前系统中未使用的设备号。二、由系统自动分配。
手动查询的方法:到开发板上 查看 /proc下的devices文件: cat /proc/devices. 然后找一个未被使用的设备号,范围(0-4096).
使用第一种方法的话,直接使用register_chrdev_region函数注册一个字符设备,使用第二种方法,需要用alloc_chrdev_region函数让系统给我们分配设备号,函数原型是:
int alloc_chrdev_region(dev_t *dev, unsigned baseminor, unsigned count,const char *name)
参数分别是:设备号输出,次设备号从多少开始分配,设备数量,设备名字。
到这里,完成了设备的注册和设备号的绑定,接下来需要把设备的操作函数(open、read、write、close等)与设备关联起来。我们用cdev的方式。
先查看cdev的数据结构,cdev数据类型定义在include/linux/cdev.h中,
struct cdev {
struct kobject kobj;
struct module *owner;
const struct file_operations *ops;
struct list_head list;
dev_t dev;
unsigned int count;
};
通过查看参数可以发现,这里有一个成员是设备号,后续我们操作驱动都要通过cdev来进行,所以需要把之前申请或者定义的设备号,传到这个cdev里面。 还有一个需要重点关注的成员是file_operations结构体,查看它的定义,可以看到这里定义了很多函数指针:
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 *);
......
};
这些指针,需要指向我们自己编写的操作函数里面。我们自己编写了一些程序,比如写了一个open_xxx函数,应用程序在使用该设备的时候,一般都过标准的系统调用open函数打开这个设备文件,那么它是如何进入到我们自己编写的函数中执行的呢,就是通过cdev里的file_operations里的成员指向我们编写的函数实现。
这里我们先定义两个结构体实体(变量),分别是cdev类型和file_operations类型;
static struct cdev led_cdev;
static struct file_operations led_ops={
.owner = THIS_MODULE,
.open = led_open,
.release = led_release,
.write = led_write,
};
在定义变量的时候,可以直接给成员赋值,这里我直接让open等指针指向了我们自己编写的操作函数led_open。这里需要注意,自己编写的操作函数需要与file_operations里面的成员定义的参数一致。如fops中write的原型是
ssize_t (*write) (struct file *, const char __user *, size_t, loff_t *);
则我们编写的led_write函数的定义应为:
static ssize_t led_write(struct file *filp,const char __user *buf,size_t cnt,loff_t *offt)
到这里我们可以先把用到的几个函数先实现一下,内容暂时为空。如下:
static int led_open(struct inode *inode, struct file *filp)
{
return 0;
}
static int led_release(struct inode *inode, struct file *filp)
{
return 0;
}
static ssize_t led_write(struct file *filp,const char __user *buf,size_t cnt,loff_t *offt)
{
return 0;
}
刚刚我们定义了两个结构体变量,接下来要通过这两个变量实现一个完整驱动的编写。
我们是通过cdev的方式实现设备文件与驱动的关联,所以接下来是cdev的操作步骤。
一:初始化cdev。
二:向内核添加一个cdev。
内核提供cdev初始化的函数是:
void cdev_init(struct cdev *, const struct file_operations *);
通过参数可以大概知道,这里的初始化操作其实就是把file_opeartions的内容赋给cdev的ops成员。前面这两个结构体变量已经定义好了,所以我们可以这样调用。
cdev_init(&led_cdev, &led_ops);
内核提供添加cdev的函数是:
int cdev_add(struct cdev *p, dev_t dev, unsigned count)
这里的参数有一个dev_t类型,前面已经提到,这个是关于设备号的数据类型,前面我们已经申请/定义了设备号,但是并没有传入到cdev里面,这一步会把前面的设备号写入cdev里面。
到这里,一个字符设备驱动初始化的流程(框架)就完成了。注册Linux字符设备会占用系统的一些资源,对于可加载卸载的设备,在卸载时(出口函数)必须释放注册时占用的资源。
前面在加载(注册)的时候,我们使用了一个设备号,还有向内核添加了一个cdev。所以在卸载的时候需要释放这些资源,编写出口函数如下:
static void __exit led_exit(void)
{
printk("led_driver_deinit\r\n");
cdev_del(&led_dev.led_cdev);
unregister_chrdev_region(led_dev.devid, LED_COUNT);
}
编写完这些,一个可用的字符设备驱动框架已经搭建完成了,但在实际运行中,加载模块后,开发板/dev目录下并没有对应的设备文件,那么我们的应用程序也就无法调用操作我们的设备驱动,这时候需要手动创建一个设备节点,名字和驱动程序名字相同,比如这里的驱动名字为led_driver。 那么需要在开发板上执行命令:mknod /dev/led_driver c xxx xxx 。 其中“/dev/led_driver”是设备节点的路径名字,c表示这是一个字符设备(char) xxx 表示设备的主设备号和次设备号。在加载驱动的时候,我们可以使用printk把主次设备号都打印出来。
事实上,创建设备节点这个步骤可以让驱动程序自动完成,但目前还是有比较多的驱动程序没有实现自动创建设备节点,因此有必要了解一下如何创建与驱动匹配的设备节点。
自动创建设备节点的方法如下:
一、创建一个类。
二、在这个类下创建一个设备。
函数分别是:
#define class_create(owner, name) \
({ \
static struct lock_class_key __key; \
__class_create(owner, name, &__key); \
})
struct device *device_create(struct class *class, struct device *parent,
dev_t devt, void *drvdata, const char *fmt, ...)
创建类的参数:所有者(一般设置为THIS_MODULE),名字。
创建设备的参数:类,父设备(没有的话就是NULL),设备号,驱动数据(没有可以为NULL),名字。
需要注意这里执行的是创建操作,可以想象得到它一定会从系统中申请一些资源,因此在卸载模块驱动的时候也要释放这些资源。
附上本例的全部代码:
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#define MODULE_NAME "led_driver"
#define LED_COUNT 1
struct led_dev_t{
int major;
int minor;
dev_t devid;
struct cdev led_cdev;
struct class *class;
struct device *device;
};
struct led_dev_t led_dev;
static int led_open(struct inode *inode, struct file *filp)
{
return 0;
}
static int led_release(struct inode *inode, struct file *filp)
{
return 0;
}
static ssize_t led_write(struct file *filp,const char __user *buf,size_t cnt,loff_t *offt)
{
return 0;
}
static struct file_operations led={
.owner = THIS_MODULE,
.open = led_open,
.release = led_release,
.write = led_write,
};
static int __init led_init(void)
{
u32 val;
printk("led_driver_init\r\n");
if(led_dev.major){
led_dev.devid = MKDEV(led_dev.major,led_dev.minor);
register_chrdev_region(led_dev.devid,LED_COUNT,MODULE_NAME);
}else{
alloc_chrdev_region(&led_dev.devid,0,LED_COUNT,MODULE_NAME);
led_dev.major = MAJOR(led_dev.devid);
led_dev.minor = MINOR(led_dev.devid);
}
printk("led dev: major:%d,minor:%d\r\n",led_dev.major,led_dev.minor);
led_dev.led_cdev.owner = THIS_MODULE;
cdev_init(&led_dev.led_cdev, &led);
cdev_add(&led_dev.led_cdev, led_dev.devid, LED_COUNT);
led_dev.class = class_create(THIS_MODULE, MODULE_NAME);
led_dev.device = device_create(led_dev.class,NULL,led_dev.devid,NULL,MODULE_NAME);
return 0;
}
static void __exit led_exit(void)
{
printk("led_driver_deinit\r\n");
cdev_del(&led_dev.led_cdev);
unregister_chrdev_region(led_dev.devid, LED_COUNT);
device_destroy(led_dev.class,led_dev.devid);
class_destroy(led_dev.class);
}
module_init(led_init);
module_exit(led_exit);
MODULE_LICENSE("GPL");
MODULE_AUTHOR("weymin");
此时我们还没有对GPIO进行任何操作,也就是说我们的led驱动程序其实还没有完成,目前只是完成一个基础框架,下一篇文章将加入对硬件处理器的操作,完善led驱动程序。
(完)2020.05.03