头文件:linux/cdev.h
//描述字符设备共性信息的结构体
struct cdev {
struct kobject kobj; //内核用于管理驱动的结构体
struct module *owner; //THIS_MODULE
const struct file_operations *ops; //操作方法集,提供给用户层的方法
struct list_head list; //链表
dev_t dev; //设备号
unsigned int count; //该驱动操作的设备数量
};
linux/fs.h
操作方法集(向应用层提供接口) 成员大多都是函数指针变量
struct file_operations {
struct module *owner;
ssize_t (*read) (struct file *, char __user *, size_t, loff_t *);
ssize_t (*write) (struct file *, const char __user *, size_t, loff_t *);
unsigned int (*poll) (struct file *, struct poll_table_struct *);
long (*unlocked_ioctl) (struct file *, unsigned int, unsigned long);
int (*mmap) (struct file *, struct vm_area_struct *);
int (*open) (struct inode *, struct file *);
int (*release) (struct inode *, struct file *);
int (*fasync) (int, struct file *, int);
}
设备号,本质是32位无符号整数,由两部分组成:主设备号和次设备号
主次设备号的关系:(类似于电影院的几排几列)
MAJOR(dev_t dev) //从设备号中提取主设备号
MINOR(dev_t dev) //从设备号中提取次设备号
MKDEV(int ma,int mi) //由主次设备好生成设备号
三个函数的宏定义实现:
提取主设备号:向右移动20位,无符号整型,前面补零,主设备号由设备号的高12位组成.次设备号由设备号的低20位组成。
1左移20位以后从0到19位全为0, -1以后,第20位变为0,其他全为1,用其值和32位无符号数相与,0和任何值相与都为0,所以前面全是0,从第19位开始,1与任何值相与,你是啥就是啥~所以次设备号是0-19(低20位).