MKDEV宏 功能:将主设备号和次设备号转换成dev_t类型
cdev结构
在Linux2.6内核中一个字符设备用cdev结构来描述,其定义如下:
struct cdev { struct kobject kobj; struct module *owner; //所属模块 const struct file_operations*ops; //文件操作结构,在写驱动时,其结构体内的大部分函数要被实现 struct list_head list; dev_t dev; //设备号,int 类型,高12位为主设备号,低20位为次设备号 unsigned int count;};
可以使用如下宏调用来获得主、次设备号:
MAJOR(dev_t dev)MINOR(dev_t dev)MKDEV(int major,int minor) //通过主次设备号来生成dev_t
以上宏调用在内核源码中如此定义:
#define MINORBITS 20#define MINORMASK ((1U << MINORBITS)- 1) //(1<<20 -1) 此操作后,MINORMASK宏的低20位为1,高12位为0#define MAJOR(dev) ((unsigned int) ((dev)>> MINORBITS))#define MINOR(dev) ((unsigned int) ((dev) &MINORMASK))#define MKDEV(ma,mi) (((ma) << MINORBITS) | (mi))//摘自:http://lxr.linux.no/linux/include/linux/kdev_t.h#L1
参数介绍:
- ma 为主设备号
- mi 为次设备号
返回值:成功执行返回dev_t类型的设备编号
我的理解是ma是主设备号,mi是次设备号
主设备号由dev_t的高位表示
次设备号是dev_t的低位
我感觉MKDEV应该是把主设备号和次设备号合成dev_t
下面一组函数用来对cdev结构体进行操作:
void cdev_init(struct cdev *, const struct file_operations *); //初始化,建立cdev和file_operation 之间的连接struct cdev *cdev_alloc(void); //动态申请一个cdev内存void cdev_put(struct cdev *p); //释放int cdev_add(struct cdev *, dev_t, unsigned); //注册设备,通常发生在驱动模块的加载函数中void cdev_del(struct cdev *); //注销设备,通常发生在驱动模块的卸载函数中
在注册时应该先调用:int register_chrdev_region(dev_t from,unsigned count,const char *name)函数为其分配设备号,此函数可用:int alloc_chrdev_region(dev_t *dev,unsigned baseminor,unsigned count,const char *name)函数代替,他们之间的区别在于:register_chrdev_region()用于已知设备号时,另一个用于动态申请,其优点在于不会造成设备号重复的冲突。
在注销之后,应调用:void unregister_chrdev_region(dev_t from,unsigned count)函数释放原先申请的设备号。
他们之间的顺序关系如下:
register_chrdev_region() --> cdev_add() //此过程在加载模块中
cdev_del() --> unregister_chrdev_region() //此过程在卸载模块中
后记: 这一套2.X内核的驱动相关技术,现在好像是过时了,以兹纪念。
© 版权声明
本文中引用的各种信息及资料(包括但不限于文字、数据、图表及超链接等)均来源于该信息及资料的相关主体(包括但不限于公司、媒体、协会等机构)的官方网站或公开发表的信息。部分内容参考包括:(百度百科,百度知道,头条百科,中国民法典,刑法,牛津词典,新华词典,汉语词典,国家院校,科普平台)等数据,内容仅供参考使用,不准确地方联系删除处理!
THE END
暂无评论内容