天天看點

cdev結構 and MKDEV

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_tdev;          //裝置号,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_tfrom,unsigned count,const char *name)函數為其配置設定裝置号,此函數可用:int alloc_chrdev_region(dev_t *dev,unsigned baseminor,unsigned count,constchar *name)函數代替,他們之間的差別在于:register_chrdev_region()用于已知裝置号時,另一個用于動态申請,其優點在于不會造成裝置号重複的沖突。

在登出之後,應調用:void unregister_chrdev_region(dev_tfrom,unsigned count)函數釋放原先申請的裝置号。

他們之間的順序關系如下:

register_chrdev_region()-->cdev_add()    //此過程在加載子產品中

cdev_del()-->unregister_chrdev_region()     //此過程在解除安裝子產品中

繼續閱讀