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

2023年 8月 7日 30.8k 0

前言

(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)驱动安装:
上一章在进行次设备号的分配上,是直接采用的register_chrdev()函数进行分配。这种方法会导致所有次设备号都被占用。
于是我们这一章,将使用 alloc_chrdev_region(),cdev_add()和cdev_init()函数来代替register_chrdev()函数。
(2)删除驱动:
在上一章中,我们删除驱动是调用的unregister_chrdev()函数。
而在我们当前这一章节,需要使用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);

相关文章

服务器端口转发,带你了解服务器端口转发
服务器开放端口,服务器开放端口的步骤
产品推荐:7月受欢迎AI容器镜像来了,有Qwen系列大模型镜像
如何使用 WinGet 下载 Microsoft Store 应用
百度搜索:蓝易云 – 熟悉ubuntu apt-get命令详解
百度搜索:蓝易云 – 域名解析成功但ping不通解决方案

发布评论