韦东山Linux驱动入门实验班(3)hello驱动---申请指定数量的次设备号

简介: 韦东山Linux驱动入门实验班(3)hello驱动---申请指定数量的次设备号

前言

(1)前面我们介绍了如何自动产生设备节点,详细分析了驱动层代码。但是我们有没有发现一个问题,我们每次设备节点的主设备号都是240,次设备号是0。主设备能够理解,这个是系统自动分配的,那么为什么次设备号永远是0呢?我能不能是其他的?

(2)答案是可以的。


什么是Linux设备号

(1)为了方便管理, Linux 中每个设备都有一个设备号,设备号由主设备号和次设备号两部分组成,主设备号表示某一个具体的驱动,次设备号表示使用这个驱动的各个设备。

(2)Linux 提供了一个名为 dev_t 的数据类型表示设备号,而dev_t 其实就是 unsigned int 类型,是一个 32 位的数据类型。

(3)这 32 位的数据构成了主设备号和次设备号两部分,其中高 12 位为主设备号,第 20 位为次设备号。

(4)因此 Linux系统中主设备号范围为 0~4095。


如何动态分配设备号

alloc_chrdev_region()简单介绍

(1)首先需要知道register_chrdev()对于设备号的申请规则是什么。==在register_chrdev()中,他只会申请到一个没有被使用到的主设备号,及其他之下的所有次设备号。==我们会发现,这样太占用次设备号资源。

(2)所以这里要做的是,申请一个主设备号和指定的次设备号,这一部分用于注册驱动程序。我们也就需要了解alloc_chrdev_region()函数。


  /*申请一个主次设备号空间,这个区域给当前驱动使用,主设备号和次设备号存入第一个参数中
   *参数一 :第一个设备的主设备号和次设备号
   *参数二 : 从哪个次设备号开始
   *参数三 :想获得几个次设备号
   *参数四 :驱动程序名
   *返回值:如果申请失败,返回一个负数
  */
  ret = alloc_chrdev_region(&dev, 0, 2, "hello");

cdev_add()简单介绍

(1)我们申请到了设备号,还需要将设备号传递给驱动,所以需要使用到cdev_add()函数。

(2)而这个传递是如何进行的呢?这里就需要引入一个cdev结构体,这个结构体用于描述一个字符设备,我们需要将设备号传递给这个结构体。

struct cdev {
  struct kobject kobj;
  struct module *owner;
  const struct file_operations *ops;
  struct list_head list;
  dev_t dev;
  unsigned int count;
 };
static struct cdev hello_cdev; //用于表示一个字符设备
  /*申请完设备号之后,需要进行
   *参数一 : 要增加的cdev
   *参数二 : 将主次设备号dev传入
   *参数三 : 决定占用几个主次设备号,这里占2个主次设备号
   *返回值 : 如果返回值不是0,表示错误
  */
    ret = cdev_add(&hello_cdev, dev, 2);

cdev_init()简单介绍

(1)我们的字符设备拥有了设备号之后,还需要file_operations结构体,用于进行描述这个驱动如何使用。

(2)所以这个时候,就需要使用到cdev_init()函数,将file_operations结构体传递给cdv结构体。

static const struct file_operations hello_drv = {
    .owner      = THIS_MODULE,
  .read   = hello_read,
  .write    = hello_write,
  .open   = hello_open,
    .release    = hello_release,
};
  //申请完设备号之后,需要进行初始化。初始化cdev,让cdev与file_operations结构体挂钩
    cdev_init(&hello_cdev, &hello_drv);


cdev_del()

这个就是用来删除cdev 的。传入cdev结构体即可

unregister_chrdev_region()

这个是将申请到的主次设备号空间释放


本章与上一章的改动之处

(1)驱动安装:

<1>上一章在进行次设备号的分配上,是直接采用的register_chrdev()函数进行分配。这种方法会导致所有次设备号都被占用。

<2>于是我们这一章,将使用 alloc_chrdev_region(),cdev_add()和cdev_init()函数来代替register_chrdev()函数。

(2)删除驱动:

<1>在上一章中,我们删除驱动是调用的unregister_chrdev()函数。

<2>而在我们当前这一章节,需要使用cdev_del()删除cdev,然后调用unregister_chrdev_region()释放主次设备号。


/********  驱动安装   ********/
/*
 *原来
*/
major = register_chrdev(0, "100ask_hello", &hello_drv);
/*
 *现在
*/
    // 如果是register_chrdev,那么会申请到一个主设备号,以及他的所有次设备号。那样太占用次设备号资源
    // 所以这里要做,申请一个主设备号和指定的次设备号,这一部分用于注册驱动程序
  /*申请一个主次设备号空间,这个区域给当前驱动使用,主设备号和次设备号存入第一个参数中
   *参数一 :第一个设备的主设备号和次设备号
   *参数二 : 从哪个次设备号开始
   *参数三 :想获得几个次设备号
   *参数四 :驱动程序名
   *返回值:如果申请失败,返回一个负数
  */
  ret = alloc_chrdev_region(&dev, 0, 2, "hello");
  //判断是否申请主次设备号成功,如果失败进入
  if (ret < 0) 
  {
    //打印无法获得设备号
    printk(KERN_ERR "alloc_chrdev_region() failed for hello\n");
    //返回无效参数
    return -EINVAL;
  }
  //申请完设备号之后,需要进行初始化。初始化cdev,让cdev与file_operations结构体挂钩
    cdev_init(&hello_cdev, &hello_drv);
  /*申请完设备号之后,需要进行
   *参数一 : 要增加的cdev
   *参数二 : 将主次设备号dev传入
   *参数三 : 决定占用几个主次设备号,这里占2个主次设备号
   *返回值 : 如果返回值不是0,表示错误
  */
    ret = cdev_add(&hello_cdev, dev, 2);
  //如果返回值不为0,打印错误
  if (ret)
    {
    //打印增加cdev失败
    printk(KERN_ERR "cdev_add() failed for hello\n");
    //返回无效参数
    return -EINVAL;
    }
/********  驱动卸载   ********/
/*
 *原来
*/
  //卸载驱动程序
  //第一个参数是主设备号,第二个是名字
    unregister_chrdev(major, "100ask_hello");
/*
 *现在
*/
    //删除cdev
    cdev_del(&hello_cdev);
  //将申请到的主次设备号空间释放
    unregister_chrdev_region(dev, 2);
目录
相关文章
|
1月前
|
Unix Linux Shell
linux入门!
本文档介绍了Linux系统入门的基础知识,包括操作系统概述、CentOS系统的安装与远程连接、文件操作、目录结构、用户和用户组管理、权限管理、Shell基础、输入输出、压缩打包、文件传输、软件安装、文件查找、进程管理、定时任务和服务管理等内容。重点讲解了常见的命令和操作技巧,帮助初学者快速掌握Linux系统的基本使用方法。
64 3
|
3月前
|
机器学习/深度学习 安全 网络协议
Linux防火墙iptables命令管理入门
本文介绍了关于Linux防火墙iptables命令管理入门的教程,涵盖了iptables的基本概念、语法格式、常用参数、基础查询操作以及链和规则管理等内容。
237 73
|
2月前
|
机器学习/深度学习 Linux 编译器
Linux入门3——vim的简单使用
Linux入门3——vim的简单使用
59 1
|
2月前
|
Linux Shell Windows
Linux入门1——初识Linux指令
Linux入门1——初识Linux指令
33 0
Linux入门1——初识Linux指令
|
2月前
|
存储 数据可视化 Linux
Linux 基础入门
Linux 基础入门
|
2月前
|
Linux Go 数据安全/隐私保护
Linux入门2——初识Linux权限
Linux入门2——初识Linux权限
29 0
|
4月前
|
Ubuntu Linux
内核实验(四):Qemu调试Linux内核,实现NFS挂载
本文介绍了在Qemu虚拟机中配置NFS挂载的过程,包括服务端的NFS服务器安装、配置和启动,客户端的DHCP脚本添加和开机脚本修改,以及在Qemu中挂载NFS、测试连通性和解决挂载失败的方法。
239 0
内核实验(四):Qemu调试Linux内核,实现NFS挂载
|
4月前
|
NoSQL Linux Android开发
内核实验(三):编写简单Linux内核模块,使用Qemu加载ko做测试
本文介绍了如何在QEMU中挂载虚拟分区、创建和编译简单的Linux内核模块,并在QEMU虚拟机中加载和测试这些内核模块,包括创建虚拟分区、编写内核模块代码、编译、部署以及在QEMU中的加载和测试过程。
231 0
内核实验(三):编写简单Linux内核模块,使用Qemu加载ko做测试
|
4月前
|
Linux 网络安全 开发工具
内核实验(二):自定义一个迷你Linux ARM系统,基于Kernel v5.15.102, Busybox,Qemu
本文介绍了如何基于Linux Kernel 5.15.102版本和BusyBox创建一个自定义的迷你Linux ARM系统,并使用QEMU进行启动和调试,包括内核和BusyBox的编译配置、根文件系统的制作以及运行QEMU时的命令和参数设置。
337 0
内核实验(二):自定义一个迷你Linux ARM系统,基于Kernel v5.15.102, Busybox,Qemu
|
4月前
|
NoSQL Linux 编译器
内核实验(一):使用QEMU+GDB断点调试Linux内核代码
如何配置环境并使用QEMU虚拟机结合GDB进行Linux内核代码的断点调试,包括安装QEMU、交叉编译工具链,编译内核以及通过GDB远程连接进行调试的详细步骤。
181 0
内核实验(一):使用QEMU+GDB断点调试Linux内核代码