转:1.1 cdev_init cdev_alloc 使用说明

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

时间: 2024-08-06 19:55:26

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

【整理】--cdev_init()/cdev_alloc(),cdev_add(),cdev_del()

(1) 内核中每个字符设备都对应一个 cdev结构的变量,下面是它的定义: linux-2.6.22/include/linux/cdev.h struct cdev { struct kobject kobj;          // 每个 cdev都是一个 kobject struct module *owner;       //指向实现驱动的模块 const struct file_operations *ops;   // 操纵这个字符设备文件的方法 struct list_head

【原创】--【字符设备】--全流程

一.概述 1.字符设备:是指只能一个字节一个字节读写的设备,不能随机读取设备内存中的某一数据,读取数据需要按照先后数据.字符设备是面向流的设备,常见的字符设备有鼠标.键盘.串口.控制台和LED设备等.2.块设备:是指可以从设备的任意位置读取一定长度数据的设备.块设备包括硬盘.磁盘.U盘和SD卡等. 每一个字符设备或块设备都在/dev目录下对应一个设备文件.linux用户程序通过设备文件(或称设备节点)来使用驱动程序操作字符设备和块设备. 二.设备使用流程:1.生成设备号,dev_t 一个字符设备

cdev_init函数

linux-2.6.22/include/linux/cdev.hstruct cdev {   struct kobject kobj;          // 每个 cdev 都是一个 kobject   struct module *owner;       // 指向实现驱动的模块   const struct file_operations *ops;   // 操纵这个字符设备文件的方法   struct list_head list;       // 与 cdev 对应的字符设备

转:Linux 内核中的 cdev_alloc和cdev_add

内核中每个字符设备都对应一个 cdev 结构的变量,下面是它的定义:linux-2.6.22/include/linux/cdev.hstruct cdev {struct kobject kobj;          // 每个 cdev 都是一个 kobjectstruct module *owner;       // 指向实现驱动的模块const struct file_operations *ops;   // 操纵这个字符设备文件的方法struct list_head list; 

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

内核中每个字符设备都对应一个 cdev 结构的变量,下面是它的定义: linux-2.6.22/include/linux/cdev.h struct cdev {    struct kobject kobj;          // 每个 cdev 都是一个 kobject    struct module *owner;       // 指向实现驱动的模块    const struct file_operations *ops;   // 操纵这个字符设备文件的方法    struct

Linux系统下安装rz/sz命令及使用说明

对于经常使用Linux系统的人员来说,少不了将本地的文件上传到服务器或者从服务器上下载文件到本地,rz / sz命令很方便的帮我们实现了这个功能,但是很多Linux系统初始并没有这两个命令.今天,我们就简单的讲解一下如何安装和使用rz.sz命令. 1.软件安装 (1)编译安装 root 账号登陆后,依次执行以下命令: 1 cd /tmp 2 wget http://www.ohse.de/uwe/releases/lrzsz-0.12.20.tar.gz 3 tar zxvf lrzsz-0.1

android 布局之滑动探究 scrollTo 和 scrollBy 方法使用说明

涉及到滑动,就涉及到VIEW,大家都知道,android的UI界面都是由一个一个的View以及View的派生类组成,View作为基类,而常用的布局里面的各种布局就是它派生出来的ViewGroup的子类,ViewGroup作为各个组件的容器搭建了整体的UI.以下是android UI的结构示示意图: 查看源码 /** * Implement this to do your drawing. * * @param canvas the canvas on which the background w

Git使用说明

Git是流行的分布式版本控制系统,可以方便的管理多人协作的项目并避免了集中式版本控制系统对服务器的依赖. 简介 版本库(repository)是一个由Git管理的目录,Git将跟踪工作目录下文件的修改并在需要时还原. 在工作目录下.git子目录则维护着版本库 . Git维护三个组件: 工作目录: 保存实际文件 暂存区 : 保存尚未提交的修改 版本库: 保存已提交的修改 HEAD: 指向最后一次提交后的结果(checkout 命令的主要功能就是操作HEAD). Git将每次提交的修改串成一条时间线

软件使用说明

软件使用说明:我们的软件是一款基于在校女大学生的记账软件,根据调查身边的女生,统计出了大部分的女生每个月开销的分类.在我们的软件中按这些分类记录每个人的每天的开销,根据调查结果显示有部分学生还有收入,所以我们的记账软件还增加了收入项.我们软件可以根据你每天输入的开销来计算你每个月的开销,并且你可以在其中设置每个月开销的额度,根据你的开销对你的花销进行提醒.在每天你也可以收到关于你这一天开销的提醒,比如:提醒你录入,提醒你今天的花费是否超标.