对 “从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 *p = 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 申请内存。

转:1.1 cdev_init cdev_alloc 使用说明的更多相关文章

  1. 【整理】--【字符设备】cdev_init()/cdev_alloc(),cdev_add(),cdev_del()

    (1) 内核中每个字符设备都对应一个 cdev结构的变量,下面是它的定义: linux-2.6.22/include/linux/cdev.h struct cdev { struct kobject ...

  2. cdev_alloc与cdev_init区别

    struct cdev *cdev_alloc(void) { struct cdev *p = kzalloc(sizeof(struct cdev), GFP_KERNEL); if (p) { ...

  3. cdev_init函数

    linux-2.6.22/include/linux/cdev.hstruct cdev {   struct kobject kobj;          // 每个 cdev 都是一个 kobje ...

  4. 转:Linux 内核中的 cdev_alloc和cdev_add

    内核中每个字符设备都对应一个 cdev 结构的变量,下面是它的定义:linux-2.6.22/include/linux/cdev.hstruct cdev {struct kobject kobj; ...

  5. linux内核cdev_init系列函数(字符设备的注册)

    内核中每个字符设备都对应一个 cdev 结构的变量,下面是它的定义: linux-2.6.22/include/linux/cdev.h struct cdev {    struct kobject ...

  6. cdev_init和register_chrdev区别

    --- 01:include/linux/fs.h static inline int register_chrdev(unsigned int major, const char *name, co ...

  7. Atitit.项目修改补丁打包工具 使用说明

    Atitit.项目修改补丁打包工具 使用说明 1.1. 打包工具已经在群里面.打包工具.bat1 1.2. 使用方法:放在项目主目录下,执行即可1 1.3. 打包工具的原理以及要打包的项目列表1 1. ...

  8. awk使用说明

    原文地址:http://www.cnblogs.com/verrion/p/awk_usage.html Awk使用说明 运维必须掌握的三剑客工具:grep(文件内容过滤器),sed(数据流处理器), ...

  9. “我爱背单词”beta版发布与使用说明

    我爱背单词BETA版本发布 第二轮迭代终于画上圆满句号,我们的“我爱背单词”beta版本已经发布. Beta版本说明 项目名称 我爱背单词 版本 Beta版 团队名称 北京航空航天大学计算机学院  拒 ...

随机推荐

  1. HDU_1071——积分求面积,抛物线顶点公式

    Problem Description Ignatius bought a land last week, but he didn't know the area of the land becaus ...

  2. WebMagic开源垂直爬虫介绍

    WebMagic项目代码分为核心和扩展两部分.核心部分(webmagic-core)是一个精简的.模块化的爬虫实现,而扩展部分则包括一些便利的.实用性的功能.WebMagic的架构设计参照了Scrap ...

  3. PHPExcell单元格中某些时间格式的内容不能正确获得的处理办法

    今天在写导入功能的时候某个时间格式的单元格内容不能正确获得,得出的是一串非时间戳的数字. 此时可以使用PHPExcell中自带的方法进行处理:PHPExcel_Shared_Date::ExcelTo ...

  4. Demon_接金币(三个掉落物品预设体,一接物体的工具)

    接物体的工具 using UnityEngine; using System.Collections; public class Tool : MonoBehaviour { float hor; V ...

  5. Spark RDD API具体解释(一) Map和Reduce

    本文由cmd markdown编辑.原始链接:https://www.zybuluo.com/jewes/note/35032 RDD是什么? RDD是Spark中的抽象数据结构类型,不论什么数据在S ...

  6. 怎样绕过oracle listener 监听的password设置

     怎样绕过oracle 监听的password设置: 1.找到监听进程pid ,并将它kill 掉 ps -ef|grep tns [oracle@lixora admin]$ ps -ef|gr ...

  7. Java编程 的动态性,第 2部分: 引入反射--转载

    在“ Java编程的动态性,第1部分,”我为您介绍了Java编程类和类装入.该篇文章介绍了一些Java二进制类格式的相关信息.这个月我将阐述使用Java反射API来在运行时接入和使用一些相同信息的基础 ...

  8. java byte数组与int,long,short,byte转换

    public class DataTypeChangeHelper { /** * 将一个单字节的byte转换成32位的int * * @param b * byte * @return conver ...

  9. 为什么不使用frame框架的原因

    框架的优点 重载页面时不需要重载整个页面,只需要重载页面中的一个框架页(减少了数据的传输,增加了网页下载速度) 方便制作导航栏 框架的缺点 会产生很多页面,不容易管理 不容易打印 浏览器的后退按钮无效 ...

  10. 如何使用css、布局横向导航栏

    使用css布局横向导航栏,css应用给网页样式的方式,就相当于,给人怎么去穿上衣服,不同的衣服有不同的穿法,这里我们使用的是内联式.在这里 我们可以适当的把值调的大一点,这样我们就可以很容易的对比. ...