本文介紹linux字符設備註冊相關的四個函數:cdev_alloc、cdev_init、cdev_add和cdev_del。這四個函數在文件:fs/char_dev.c中定義,在頭文件include/linux/cdev.h中聲明。其中cdev_alloc和cdev_init是一對“互斥”函數,以不同的方式完成“相同”的功能:爲函數cdev_add做前期準備。
cdev_alloc
509 struct cdev *cdev_alloc(void)
510 {
511 struct cdev *p = kzalloc(sizeof(struct cdev), GFP_KERNEL);
512 if (p) {
513 p->kobj.ktype = &ktype_cdev_dynamic;
514 INIT_LIST_HEAD(&p->list);
515 kobject_init(&p->kobj);
516 }
517 return p;
518 }
從函數名稱和第511行的代碼可以看出:這個函數動態申請結構體struct cdev,並對其進行初始化,最後將其指針返回。下面結合cdev_init進行進一步說明。
cdev_init
528 void cdev_init(struct cdev *cdev, const struct file_operations *fops)
529 {
530 memset(cdev, 0, sizeof *cdev);
531 INIT_LIST_HEAD(&cdev->list);
532 cdev->kobj.ktype = &ktype_cdev_default;
533 kobject_init(&cdev->kobj);
534 cdev->ops = fops;
535 }
cdev_alloc和cdev_init的主要區別是:前者動態申請結構體struct cdev並對其進行初始化,後者將通過參數傳進來的結構體struct cdev進行初始化。
另一個主要區別是:cdev_alloc函數中沒有對struct cdev的ops域進行初始化,需要在cdev_alloc函數調用之後有專門的代碼對struct cdev的ops域進行初始化,而cdev_init函數中使用通過參數傳進來的struct file_operations結構體指針對struct cdev的ops域進行初始化,所以在函數cdev_init調用之後不需要再對struct cdev的ops域進行初始化。
cdev_add
457 int cdev_add(struct cdev *p, dev_t dev, unsigned count)
458 {
459 p->dev = dev;
460 p->count = count;
461 return kobj_map(cdev_map, dev, count, NULL, exact_match, exact_lock, p);
462 }
函數cdev_alloc和cdev_init只是(申請)並初始化了(部分)結構體struct cdev,此時,struct cdev和內核還沒有任何關係。
函數cdev_add就是將函數cdev_alloc和cdev_init初始化後的struct cdev結構體註冊到內核中(第461行),自此內核就可以訪問設備了。
cdev_del
本函數和函數cdev_add功能相反,從內核中刪除設備。