copy from : https://blog.csdn.net/weixin_42462202/article/details/100039448
文章目录
Linux misc设备(一)misc驱动框架
一、misc简介
二、misc驱动框架
三、misc源码剖析
四、misc设备实例驱动编写模板
一、misc简介
Linux的驱动设计是趋向于分层的,大多数设备都有自己归属的类型,例如按键、触摸屏属于输入设备,Linux有一个input子系统框架。但是对于adc、蜂鸣器等设备,无法明确其属于什么类型,对于这种设备一般推荐使用misc驱动框架编写驱动程序
misc设备也是一个字符设备,在misc的初始化函数中注册了一个字符设备,主设备号为MISC_MAJOR (10)
这个字符设备并不是具体的misc设备实例的实现,只是misc核心层用来将应用层的操作转发到具体的misc实例中
在编写misc设备驱动实例时,通过misc_register()向misc核心层注册一个字符设备,在此函数中完成了生成设备节点、动态获取次设备号的动作
二、misc驱动框架
misc设备实例表明一个具体的驱动程序(例如adc、蜂鸣器),每一个misc设备实例都有对应得次设备号和相应驱动程序实现的文件操作集
misc设备实例驱动通过misc_register向misc核心层注册misc设备,所有的misc的主设备号都是(MISC_MAJOR),misc核心为注册的misc设备维护一个双向链表,如下图所示
当应用层通过open打开设备节点时,内核会为打开的设备文件创建一个struct file对象
然后调用到misc核心层的cdev的open函数,在cdev的open函数中,从misc维护的misc设备链表中,根据次设备号找到对应的misc设备实例,将struct file对象中的文件操作集合更改为misc设备实例的文件操作集合,然后调用misc设备实例的open函数,如下图所示
之后应用层通过read、ioctl等来操作设备文件,都会调用对应的struct file对象中的文件操作集合,进而操作具体的misc实例,如下图所示
以上就是misc设备的驱动框架
三、misc源码剖析
在内核文件driverscharmisc.c中时misc驱动核心的实现,下面详细剖析这个源码
首先看驱动的入口
static struct class *misc_class;
static const struct file_operations misc_fops = {
.owner = THIS_MODULE,
.open = misc_open,
};
static int __init misc_init(void)
{
/* 创建misc类 */
misc_class = class_create(THIS_MODULE, "misc");
/* 注册字符设备 */
register_chrdev(MISC_MAJOR,"misc",&misc_fops); //MISC_MAJOR主设备号(10)
}
首先创建misc类,这将给misc设备实例生成设备节点时使用
然后注册字符设备,可以看到misc_fops中只实现了open函数,此函数是用于转发系统调用到具体的misc设备实例中,是misc驱动框架的核心
再来看注册函数misc_register
int misc_register(struct miscdevice * misc)
{
/* 找到空闲的次设备号 */
int i = find_first_zero_bit(misc_minors, DYNAMIC_MINORS);
/* 设备号 */
dev = MKDEV(MISC_MAJOR, misc->minor);
/* 生成设备节点 */
misc->this_device = device_create(misc_class, misc->parent, dev,
misc, "%s", misc->name);
list_add(&misc->list, &misc_list); //将设备添加到链表中
}
可以看到注册函数会分配一个次设备号,然后生成设备节点,再将misc设备添加到链表中
接下来分析misc_open
static int misc_open(struct inode * inode, struct file * file)
{
/* 遍历misc设备链表,根据次设备号找到对应misc设备实例的fops */
list_for_each_entry(c, &misc_list, list) {
if (c->minor == minor) {
new_fops = fops_get(c->fops);
break;
}
}
/* 将file对象的fops设置为misc实例的fops */
file->f_op = new_fops;
}
从上面代码中可以看到,首先从misc设备链表中找到misc设备实例,然后将file对应得fops设置为misc设备实例对应的fops,其中file是内核为每个打开文件创建的一个对象
之后应用层再进行系统调用,都会调用到misc设备实例
四、misc设备实例驱动编写模板
如何基于misc写一个设备驱动非常简单,下面给出一个模板
static int buzzer_open(struct inode *inode, struct file *file)
{
return 0;
}
static int buzzer_close(struct inode *inode, struct file *file)
{
return 0;
}
static int buzzer_ioctl(struct inode *inode, struct file *file, unsigned int cmd, unsigned long arg)
{
return 0;
}
static struct file_operations buzzer_fops = {
.owner = THIS_MODULE,
.open = buzzer_open,
.release = buzzer_close,
.ioctl = buzzer_ioctl,
};
static struct miscdevice buzzer_dev = {
.minor = MISC_DYNAMIC_MINOR, //动态分配次设备号
.name = "buzzer",
.fops = &buzzer_fops, //文件操作集
};
static int __init buzzer_init(void)
{
/* 注册杂项设备 */
misc_register(&buzzer_dev);
return 0;
}
static void __exit buzzer_exit(void)
{
/* 注销杂项设备 */
misc_deregister(&buzzer_dev);
}
module_init(buzzer_init);
module_exit(buzzer_exit);
MODULE_LICENSE("GPL");
————————————————
版权声明:本文为CSDN博主「JT同学」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/weixin_42462202/article/details/100039448