Chinaunix首页 | 论坛 | 博客
  • 博客访问: 112819
  • 博文数量: 40
  • 博客积分: 0
  • 博客等级: 民兵
  • 技术积分: 10
  • 用 户 组: 普通用户
  • 注册时间: 2016-09-22 17:28
文章分类

全部博文(40)

文章存档

2021年(2)

2018年(3)

2017年(29)

2016年(6)

我的朋友

分类: LINUX

2018-02-06 20:29:08

本文的copyright归yuweixian4230@163.com 所有,使用GPL发布,可以自由拷贝,转载。但转载请保持文档的完整性,注明原作者及原链接,严禁用于任何商业用途。
作者:yuweixian4230@163.com
博客:
yuweixian4230.blog.chinaunix.net  

对 “从globalmem学习linux字符设备驱动” 的 cdev_init 和 cdev_alloc中一些不清楚的地方进行说明:

cdev_init 和 cdev_alloc函数定义如下:

 1.1 cdev_init
函数用于初始化cdev的成员,并建立生cdev和file_operations之间连接
  1. /fs/char_dev.c 2.6.35
  2.  550 void cdev_init(struct cdev *cdev, const struct file_operations *fops)
  3.  551 {
  4.  552  memset(cdev, 0, sizeof *cdev);
  5.  553  INIT_LIST_HEAD(&cdev->list); //初始化链表
  6.  554  kobject_init(&cdev->kobj, &ktype_cdev_default);//初始化 kobject 对象
  7.  555  cdev->ops = fops; //将传入的fops 传递给 cdev结构体
  8.  556 }
程序 memest主要是对 cdev已经分配好的内存进行清零,说明在cdev_init函数外,我们已经完成了对 cdev结构体的 内存分配工作。

1.2 cdev_alloc()
用于动态申请一个cdev内存
  1. /fs/char_dev.c 2.6.35
  2.  532 struct cdev *cdev_alloc(void)
  3.  533 {
  4.  534  struct cdev *= kzalloc(sizeof(struct cdev), GFP_KERNEL);
  5.  535  if (p) {
  6.  536  INIT_LIST_HEAD(&p->list); //操作与 cdev_init中的部分是一致的
  7.  537  kobject_init(&p->kobj, &ktype_cdev_dynamic);
  8.  538  }
  9.  539  return p;
  10.  540 }
cdev_init() 和 cdev_alloc()有什么关系呢?
是不是可以不用cdev_alloc()这个函数呢?


在cdev_alloc()函数中,kzalloc已经完成了 申请内存,并且对申请的内存进行

memset清零操作,这个操作和 cdev_init()中的 memset清零操作是一样的。注

意在后面的程序操作是一样的。只是在cdev_init中只是添加了 file_operations的指针。然后再 cdev_alloc,返回struct cdev *的指针。

如果我们这么使用:
struct cdev * x_cdev;
x_cdev = cdev_alloc();//完成内存申请,
cdev_init(x_cdev,&fops);//完成 fops 链接操作。只是我们重复了一些操作。
如下:
在cdev_alloc(),完成了内存申请,然后初始化 struct kobject 和 struct list_head,
在cdev_init(x_cdev,&fops)中,我们又 重新设置了 struct kobject 和 struct list_head ,最后完成 fops链接操作。
其实,我们只是做了一些重复的无用工作,其实完全不需要的。


如果我们定义的是一个 struct cdev * 的指针类型,我们可以使用cdev_alloc对其申请内存,然后使用 cdev_init。
如果我们定义的是一个 struct cdev x_cdev结构体而并非是一个指针,这里,就不需要cdev_alloc了,我们需要使用kzalloc(已经完成内存清零操作)或kmalloc进行内存申请,然后 cdev_init,

=================================================
下面是在内核中具体的例子说明:

例一:
  1. src/linux/drivrs/char/raw.c

  2. static struct cdev raw_cdev;//分配结构体类型
  3. static const struct file_operations raw_fops={
  4.     .read = do_siync_read,
  5.     .write = ...
  6. };

  7. static int __init raw_init(void)
  8. {
  9.         ...
  10.         cdev_init(&raw_cdev,&raw_fops);
  11.         cdev_add();
  12.         ...
  13. }
因为这里分配的是 struct cdev 结构体类型,所以没有使用 cdev_alloc。


例二:
  1. src/drivers/char/snsc.c

  2. struct sysctl_data_s{
  3. struct cdev scd_cdev;
  4. nasid_t scd_nasid;
  5. };

  6. static const struct file_operations scdrev_fops={
  7. .read=scdrv_read,
  8. ...
  9. }

  10. static int __init scrdv_init(void)
  11. {
  12. ...
  13. struct sysctl_data_s *scd;//申请指针
  14. ...
  15. scd=kzalloc(sizeof(struct sysctl_data_s),GFP_KERKEL);
  16. ...
  17. cdev_init(&scd->scd_cdev,&scdrv_fops);
  18. cdev_add();

  19. }
这里申请的是 指针类型,而且指针指向的结构体 不仅仅包含 cdev 类型,所以这里不能用 cdev_alloc 申请内存, 我们自己使用kzalloc 申请内存。

===================================================


阅读(2237) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~