MISC 的意思是混合、杂项的,因此MISC 驱动也叫做杂项驱动,也就是当我们板子上的某些外设无法进行分类的时候,可以使用MISC 驱动。
MISC 驱动其实就是最简单的字符设备驱动,通常嵌套在platform 总线驱动中,实现复杂的驱动。
所有的MISC 设备驱动的主设备号都为10,不同的设备使用不同的从设备号。随着Linux 字符设备驱动的不断增加,设备号变得原来越紧张,尤其是主设备号,MISC 设备驱动就用于解决此问题。
MISC 设备会自动创建cdev,不需要像我们以前那样手动创建,因此采用MISC 设备驱动可以简化字符设备驱动的编写。
我们需要向Linux 注册一个miscdevice 设备,miscdevice 是一个结构体,内容如下:
struct miscdevice {
int minor;
const char *name;
const struct file_operations *fops;
struct list_head list;
struct device *parent;
struct device *this_device;
const struct attribute_group **groups;
const char *nodename;
umode_t mode;
};
定义一个MISC 设备,我们需要设置minor,name , fops 这三个成员变量。
Linux 系统已经预定义了一些MISC 设备的子设备号,在include/linux/miscdevice.h 文件中。我们在使用的时候可以从这些预定义的子设备号中挑选一个,当然也可以自己定义,只要这个子设备号没有被其他设备使用接口。
#define PSMOUSE_MINOR 1
#define MS_BUSMOUSE_MINOR 2 /* unused */
#define ATIXL_BUSMOUSE_MINOR 3 /* unused */
/*#define AMIGAMOUSE_MINOR 4 FIXME OBSOLETE */
#define ATARIMOUSE_MINOR 5 /* unused */
#define SUN_MOUSE_MINOR 6 /* unused */
#define APOLLO_MOUSE_MINOR 7 /* unused */
#define PC110PAD_MINOR 9 /* unused */
/*#define ADB_MOUSE_MINOR 10 FIXME OBSOLETE */
#define WATCHDOG_MINOR 130 /* Watchdog timer */
#define TEMP_MINOR 131 /* Temperature Sensor */
#define RTC_MINOR 135
#define EFI_RTC_MINOR 136 /* EFI Time services */
#define VHCI_MINOR 137
#define SUN_OPENPROM_MINOR 139
#define DMAPI_MINOR 140 /* unused */
#define NVRAM_MINOR 144
#define SGI_MMTIMER 153
#define STORE_QUEUE_MINOR 155 /* unused */
#define I2O_MINOR 166
#define MICROCODE_MINOR 184
#define VFIO_MINOR 196
#define TUN_MINOR 200
#define CUSE_MINOR 203
#define MWAVE_MINOR 219 /* ACP/Mwave Modem */
#define MPT_MINOR 220
#define MPT2SAS_MINOR 221
#define MPT3SAS_MINOR 222
#define UINPUT_MINOR 223
#define MISC_MCELOG_MINOR 227
#define HPET_MINOR 228
#define FUSE_MINOR 229
#define KVM_MINOR 232
#define BTRFS_MINOR 234
#define AUTOFS_MINOR 235
#define MAPPER_CTRL_MINOR 236
#define LOOP_CTRL_MINOR 237
#define VHOST_NET_MINOR 238
#define UHID_MINOR 239
#define MISC_DYNAMIC_MINOR 255
当设置好miscdevice 以后就需要使用misc_register 函数向系统注册一个MISC 设备,此函数原型如下:
int misc_register(struct miscdevice * misc)
函数参数和返回值含义如下:
以前我们需要自己调用一堆的函数去创建设备,比如在以前的字符设备驱动中我们会使用如下几个函数完成设备创建过程:
alloc_chrdev_region() ; //申请设备号
cdev_init(); //初始化cdev
cdev_add(); //添加cdev
class_create() ; //创建类
device_create(); //创建设备
当我们卸载设备驱动模块的时候,需要调用misc_deregister 函数来注销掉MISC 设备,函数原型如下:
int misc_deregister(struct miscdevice* misc)
函数参数和返回值含义如下:
以前注销设备驱动的时候,我们需要调用一堆的函数去删除此前的cdev、设备等内容,如下所示:
cdev_del(); //删除cdev
unregister_chrdev_region(); //注销设备号
device_destroy(); //删除设备
class_destroy(); //删除类