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<> MINORBITS))
#define MINOR(dev)      ((unsigned int) ((dev) &MINORMASK))
#define MKDEV(ma,mi)    (((ma) << MINORBITS) | (mi))
//摘自://lxr.linux.no/linux/include/linux/kdev_t.h#L1

参数介绍:

返回值:成功执行返回dev_t类型的设备编号

我的理解是ma是主设备号,mi是次设备号

linux设备驱动开发详解 源码_linux驱动程序开发_linux驱动开发项目

主设备号由dev_t的低位表示

次设备号是dev_t的高位

我觉得MKDEV应当是把主设备号和次设备号合成dev_t

linux驱动开发项目_linux设备驱动开发详解 源码_linux驱动程序开发

下边一组函数拿来对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 *);  //注销设备,通常发生在驱动模块的卸载函数中

在注册时应当先调用:intregister_chrdev_region(dev_tfrom,unsignedcount,constchar*name)函数为其分配设备号,此函数可用:intalloc_chrdev_region(dev_t*dev,unsignedbaseminor,unsignedcount,constchar*name)函数替代,她们之间的区别在于:register_chrdev_region()用于已知设备号时linux怎么查看系统版本,另一个用于动态申请linux设备驱动开发详解 源码,其优点在于不会导致设备号重复的冲突。

在注销以后linux入门,应调用:voidunregister_chrdev_region(dev_tfrom,unsignedcount)函数释放原本申请的设备号。

她们之间的次序关系如下:

register_chrdev_region()-->cdev_add()//此过程在加载模块中

cdev_del()-->unregister_chrdev_region()//此过程在卸载模块中

杂记:这一套2.X内核的驱动相关技术,如今似乎是过时了linux设备驱动开发详解 源码,以兹记念。

本文原创地址://q13zd.cn/mhdyznhsdzfs.html编辑:刘遄,审核员:暂无