Linux字符设备注册函数 register_chrdev详解

Linux字符设备注册函数 register_chrdev详解

 

当我们需要注册字符设备的时候,需要module_init()中调用register_chrdev()注册。

下面主要介绍接口的实现过程与细节。

内核函数前面添加__  代表内核级函数。谨慎调用。源代码如下:

int __register_chrdev(unsigned int major, unsigned int baseminor,

      unsigned int count, const char *name,

      const struct file_operations *fops)

{

struct char_device_struct *cd;

struct cdev *cdev;

int err = -ENOMEM;

 

cd = __register_chrdev_region(major, baseminor, count, name);

if (IS_ERR(cd))

return PTR_ERR(cd);

cdev = cdev_alloc();

if (!cdev)

goto out2;

 

cdev->owner = fops->owner;

cdev->ops = fops;

kobject_set_name(&cdev->kobj, "%s", name);

err = cdev_add(cdev, MKDEV(cd->major, baseminor), count);

if (err)

goto out;

 

cd->cdev = cdev;

 

return major ? 0 : cd->major;

out:

kobject_put(&cdev->kobj);

out2:

kfree(__unregister_chrdev_region(cd->major, baseminor, count));

return err;

}

1:参数分析

 * @major: major device number or 0 for dynamic allocation

 主设备号,当用户设置为0时,内核会动态分配一个设备号。

 * @baseminor: first of the requested range of minor numbers

次设备号,要在一定范围内从0开始

 * @count: the number of minor numbers required

次设备号的范围

 * @name: name of this range of devices

设备名称

 * @fops: file operations associated with this devices

文件系统的接口指针

2:接口代码分析

 

struct char_device_struct *cd; //字符设备结构体指针,用于检测存储使用。

static struct char_device_struct {

struct char_device_struct *next;

unsigned int major;

unsigned int baseminor;

int minorct;

char name[64];

struct cdev *cdev; /* will die */

} *chrdevs[CHRDEV_MAJOR_HASH_SIZE];

 __register_chrdev_region():

检查设备号是否有效,注册设备到全局变量chrdevs[i]中。

内核中有字符设备和块设备表,根据设备类型和主设备号既能找到对应的结构跳转函数。


struct cdev *cdev; 字符型设备,这个是真正的实用的。

struct cdev {

struct kobject kobj;

struct module *owner;

const struct file_operations *ops;

struct list_head list;

dev_t dev;

unsigned int count;

};

cdev = cdev_alloc();

分配一个字符设备结构内存大小,返回这个结构,失败返回空。

cdev->owner = fops->owner;

cdev->ops = fops;

kobject_set_name(&cdev->kobj, "%s", name);

设置设备用户,文件操作指针,设备名称。

 

err = cdev_add(cdev, MKDEV(cd->major, baseminor), count);

添加设备到系统中 module结构体链表中,使之模块立即生效。此后文件操作,可以正常使用

 

如果注册失败的话,释放以上的配置。

 

重点是 cdev_add的理解源代码如下:

int cdev_add(struct cdev *p, dev_t dev, unsigned count)

{

p->dev = dev;

p->count = count;

return kobj_map(cdev_map, dev, count, NULL, exact_match, exact_lock, p);

}

 

int kobj_map(struct kobj_map *domain, dev_t dev, unsigned long range,

     struct module *module, kobj_probe_t *probe,

     int (*lock)(dev_t, void *), void *data)

{

unsigned n = MAJOR(dev + range - 1) - MAJOR(dev) + 1;

unsigned index = MAJOR(dev);

unsigned i;

struct probe *p;

 

if (n > 255)

n = 255;  //检测设备号是否在范围内

 

p = kmalloc(sizeof(struct probe) * n, GFP_KERNEL);

 

if (p == NULL)

return -ENOMEM;

 

for (i = 0; i < n; i++, p++) {   //设置字符module结构的配置

p->owner = module;

p->get = probe;

p->lock = lock;

p->dev = dev;

p->range = range;

p->data = data;

}

mutex_lock(domain->lock);

for (i = 0, p -= n; i < n; i++, p++, index++) {

struct probe **s = &domain->probes[index % 255];//添加到全局模块中

while (*s && (*s)->range < range)

s = &(*s)->next;

p->next = *s;

*s = p;

}

mutex_unlock(domain->lock);

return 0;

}

 

取消模块就是删除原来注册的东西,不做详细介绍。

你可能感兴趣的:(linux内核编程)