linux驱动是分层设计的,所以各个具体设备都能找到归属的类型,从而套用对应的框架。这样,就只需要实现最底层的部分。
但是也有一部分设备并不能确定它的类型,这时,可以使用miscdevice的框架。
本质上,miscdevice也是字符设备的一种,只是对CDEV进行了二次封装。
在miscdevice的misc_init函数中,
regiser_chrdev(MISC_MAJOR, “misc”, &misc_fops);
注册了CDEV。
miscdeviece的主号固定为10,区别在于各自的次号不同。
struct miscdevice{
int minor;
const char* name;
const char* file_operations * fops;
...
};
misc_register(struct miscdevice* misc);
misc_deregister(struct miscdevice* misc);
是内核提供的API。
类似于CDEV,如果要设计一个miscdevice,我们可以衍生出一个设备结构体。
struct xxx_miscdev{
struct miscdevice miscdev;
....
};
miscdev的注册,也十分类似于CDEV。
static struct xxx_miscdev xxx_dev{
.minor = MISC_DYNAMIC_MINOR,
.name = "xxx",
.fops = &xxx_ops,
};
static const struct file_operations xxx_ops = {
.mmap = xxx_mmap,
.unlocked_ioctl = xxx_ioctl,
...
};
...
static long xxx_ioctl(struct file* filp, unsigned int cmd, unsigned long arg)
{
struct xxx_miscdev * xxx = container_of(filp->private_data, struct xxx_miscdev, miscdev);
...
}
...
static int __init xxx_init(void)
{
int ret;
ret = misc_register(&xxx_dev.miscdev);
return ret;
}
在miscdevice的CORE中的misc_open函数中,已经帮助创建了回溯引用。将filp->private_data填充为注册的miscdev。
所以,我们在操作函数中,通过container和private_data可以索引到我们需要的衍生对象。