0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

如何在Linux系统下自动创建设备节点

嵌入式开发爱好者 来源:嵌入式开发爱好者 作者:嵌入式开发爱好者 2022-11-06 20:18 次阅读

第一:自动创建设备节点方法

1、使用 class_create 函数创建一个类。

2、使用 device_create 函数在我们创建的类下面创建一个设备。

第二:自动创建设备节点简介

Linux驱动实验中,通过使用insmod命令加载模块后,需要通过mknod命令手动创建设备节点,这样使用起来比较麻烦,并且不可能每个设备都这样操作, Linux 系统的存在就是为了方便使用, 所以我们来看一下如何实现自动创建设备节点, 当加载模块时, 在/dev 目录下自动创建相应的设备文件。怎么自动创建一个设备节点呢?在嵌入式 Linux 中使用 mdev 来实现设备节点文件的自动创建和删除。

udev 是一种工具, 它能够根据系统中的硬件设备的状态动态更新设备文件, 包括设备文件的创建, 删除等。设备文件通常放在/dev 目录下。使用 udev 后, 在/dev 目录下就只包含系统中真正存在的设备。而mdev 是 udev 的简化版本,是 busybox 中所带的程序,最适合用在嵌入式系统,而 udev 一般用在 PC 上的linux 中,相对 mdev 来说要复杂些, 所以在嵌入式 Linux 中使用 mdev 来实现设备节点文件的自动创建和删除。

第三:创建和删除类函数

内核中定义了struct class结构体,一个struct class结构体类型变量对应一个类,内核同时提供了class_create用来创建一个类,这个类存放于 sysfs 下面, 一旦创建好了这个类, 再调用 device_create来在/dev 目录下创建相应的设备节点。这样, 加载模块的时候, 用户空间中的 udev 会自动响应 device_create,去/sysfs 下寻找对应的类从而创建设备节点。

在 Linux 驱动程序中一般通过 class_create 和 class_destroy 来完成设备节点的创建和删除。首先要创建一个 class 类结构体, class 结构体定义在 include/linux/device.h 里面。class_create 是个宏, 宏定义如下:#define class_create(owner, name)

({ 
static struct lock_class_key __key; 
__class_create(owner, name, &__key); 
})
 
struct class *__class_create(struct module *owner, const char *name,struct lock_class_key *key)

class_create一共有两个参数,参数owner 一般为 THIS_MODULE, 参数 name 是类名字。返回值是个指向结构体 class 的指针, 也就是创建的类。

卸载驱动程序的时候需要删除掉类,类删除函数为class_destroy, 函数原型如下:
void class_destroy(struct class *cls);//参数 cls 就是要删除的类。
第四:创建设备函数

当使用上节的函数创建完成一个类后,使用device_create 函数在这个类下创建一个设备。

device_create函数原型如下:
struct device *device_create(struct class *class,
struct device *parent,
dev_t devt,
void *drvdata,
const char *fmt, ...)

device_create是个可变参数函数,参数class就是设备要创建哪个类下面;参数parent是父设备,一般为NULL,也就是没有父设备;参数devt是设备号;参数drvdata是设备可能会使用的一些数据,一般为NULL;参数fmt是设备名字,如果设置fmt=xxx的话,就会生成/dev/xxx这个设备文件。返回值就是创建好的设备。同样的, 卸载驱动的时候需要删除掉创建的设备, 设备删除函数为 device_destroy, 函数原型如下:

void device_destroy(struct class *class, dev_t devt)
参数class是要删除的设备所处的类,参数devt是要删除的设备号。

第五:创建类函数

chrdev.c文件完整代码如下所示:

#include  //初始化头文件
#include  //最基本的文件, 支持动态添加和卸载模块。
#include  //包含了文件操作相关 struct 的定义, 例如大名鼎鼎的struct file_operations
#include 
#include  //对字符设备结构 cdev 以及一系列的操作函数的定义。 
                        //包含了 cdev 结构及相关函数的定义。
 
#define DEVICE_NUMBER 1 //定义次设备号的个数
#define DEVICE_SNAME "schrdev" //定义静态注册设备的名称
#define DEVICE_ANAME "achrdev" //定义动态注册设备的名称
#define DEVICE_MINOR_NUMBER 0 //定义次设备号的起始地址
 
#include  //包含了 device、 class 等结构的定义
 
#define DEVICE_CLASS_NAME "chrdev_class" //宏定义类名
 
static int major_num, minor_num; //定义主设备号和次设备号
struct class *class; //定义类
struct cdev cdev;//定义一个 cdev 结构体
 
module_param(major_num, int, S_IRUSR); //驱动模块传入普通参数 major_num
module_param(minor_num, int, S_IRUSR); //驱动模块传入普通参数 minor_num
 
dev_t dev_num;
 
int chrdev_open(struct inode *inode, struct file *file)
{
    printk("chrdev_open
");
    return 0;
} 
 
struct file_operations chrdev_ops = {
.owner = THIS_MODULE,
.open = chrdev_open
};
 
static int hello_init(void)
{
    int ret; //函数返回值
    if (major_num)
    {
         /*静态注册设备号*/
         printk("major_num = %d
", major_num); //打印传入进来的主设备号
         printk("minor_num = %d
", minor_num); //打印传入进来的次设备号
         //MKDEV 将主设备号和次设备号合并为一个设备号
         dev_num = MKDEV(major_num, minor_num);
         ret = register_chrdev_region(dev_num, DEVICE_NUMBER, DEVICE_SNAME); //注册设备号
         if (ret < 0)
         {
              printk("register_chrdev_region error
");
         } 
         printk("register_chrdev_region ok
"); //静态注册设备号成功
    } 
    else
    {
         /*动态注册设备号*/
         ret = alloc_chrdev_region(&dev_num, DEVICE_MINOR_NUMBER, 1, DEVICE_ANAME);
         if (ret < 0)
         {
             printk("alloc_chrdev_region error
");
         } 
         printk("alloc_chrdev_region ok
"); //动态注册设备号成功
         major_num = MAJOR(dev_num); //将主设备号取出来
         minor_num = MINOR(dev_num); //将次设备号取出来
         printk("major_num = %d
", major_num); //打印传入进来的主设备号
         printk("minor_num = %d
", minor_num); //打印传入进来的次设备号
    } 
    cdev.owner = THIS_MODULE;
    //cdev_init 函数初始化 cdev 结构体成员变量
    cdev_init(&cdev, &chrdev_ops);
    //完成字符设备注册到内核
    cdev_add(&cdev, dev_num, DEVICE_NUMBER);
    //创建类
    class = class_create(THIS_MODULE, DEVICE_CLASS_NAME);
    return 0;
}
 
static void hello_exit(void)
{
    unregister_chrdev_region(MKDEV(major_num, minor_num), DEVICE_NUMBER);
    //注销设备号
    cdev_del(&cdev);
    //删除类
    class_destroy(class);
   printk("gooodbye! 
");
}
 
module_init(hello_init);
module_exit(hello_exit);
MODULE_LICENSE("GPL");

将代码编译成模块,利用驱动程序里面的Makefile文件。

编译并加载,如果创建类成功以后,会在开发板的/sys/class/下面生成一个名为“chrdev_class”的类。现在没有加载驱动的情况,如下图所示:ls /sys/class

第六:创建设备函数

在前面代码的基础上添加创建设备的代码,如下所示:

#include  //初始化头文件
#include  //最基本的文件, 支持动态添加和卸载模块。
#include  //包含了文件操作相关 struct 的定义, 例如struct file_operations
#include 
 
#include  //对字符设备结构 cdev 以及一系列的操作函数的定义。
                        //包含了 cdev 结构及相关函数的定义。
 
#define DEVICE_NUMBER 1 //定义次设备号的个数
#define DEVICE_SNAME "schrdev" //定义静态注册设备的名称
#define DEVICE_ANAME "achrdev" //定义动态注册设备的名称
#define DEVICE_MINOR_NUMBER 0 //定义次设备号的起始地址
 
#include  //包含了 device、 class 等结构的定义
 
#define DEVICE_CLASS_NAME "chrdev_class"
#define DEVICE_NODE_NAME "chrdev_test" //宏定义设备节点的名字
 
static int major_num, minor_num; //定义主设备号和次设备号
struct class *class; /* 类 */
struct device *device; /* 设备 */
struct cdev cdev; //定义一个 cdev 结构体
 
module_param(major_num, int, S_IRUSR); //驱动模块传入普通参数 major_num
module_param(minor_num, int, S_IRUSR); //驱动模块传入普通参数 minor_num
dev_t dev_num; /* 设备号 */
 
/***
* @description: 打开设备
* @param {structinode} *inode:传递给驱动的 inode
* @param {structfile} *file:设备文件, file 结构体有个叫做 private_data 的成员变量,
* 一般在 open 的时候将 private_data 指向设备结构体。
* @return: 0 成功;其他 失败
*/
int chrdev_open(struct inode *inode, struct file *file)
{
    printk("chrdev_open
");
    return 0;
}
 
// 设备操作函数结构体
struct file_operations chrdev_ops = {
.owner = THIS_MODULE,
.open = chrdev_open
};
 
/**
* @description: 驱动入口函数
* @param {*}无
* @return {*} 0 成功;其他 失败
*/
static int hello_init(void)
{
    int ret; //函数返回值
    if (major_num)
    {
        /*静态注册设备号*/
        printk("major_num = %d
", major_num); //打印传入进来的主设备号
        printk("minor_num = %d
", minor_num); //打印传入进来的次设备号
        dev_num = MKDEV(major_num, minor_num); 
        //MKDEV 将主设备号和次设备号合并为一个设备号
        ret = register_chrdev_region(dev_num, DEVICE_NUMBER, DEVICE_SNAME); //注册设备号
        if (ret < 0)
        {
            printk("register_chrdev_region error
");
        } 
        printk("register_chrdev_region ok
"); //静态注册设备号成功
    } 
    else
    {
        /*动态注册设备号*/
        ret = alloc_chrdev_region(&dev_num, DEVICE_MINOR_NUMBER, 1, DEVICE_ANAME);
        if (ret < 0)
        {
            printk("alloc_chrdev_region error
");
        }
        printk("alloc_chrdev_region ok
"); //动态注册设备号成功
        major_num = MAJOR(dev_num); //将主设备号取出来
        minor_num = MINOR(dev_num); //将次设备号取出来
        printk("major_num = %d
", major_num); //打印传入进来的主设备号
        printk("minor_num = %d
", minor_num); //打印传入进来的次设备号
    } 
    // 初始化 cdev
    cdev.owner = THIS_MODULE;
    cdev_init(&cdev, &chrdev_ops);
    // 向系统注册设备
    cdev_add(&cdev, dev_num, DEVICE_NUMBER);
    // 创建 class 类
    class = class_create(THIS_MODULE, DEVICE_CLASS_NAME);
    // 在 class 类下创建设备
    device = device_create(class, NULL, dev_num, NULL, DEVICE_NODE_NAME);
    return 0;
} 
 
/**
* @description: 驱动出口函数
* @param {*}无
* @return {*}无
*/
static void hello_exit(void)
{
    //注销设备号
    unregister_chrdev_region(MKDEV(major_num, minor_num), DEVICE_NUMBER);
    //删除设备
    cdev_del(&cdev);
    //注销设备
    device_destroy(class, dev_num);
    //删除类
    class_destroy(class);
    printk("gooodbye! 
");
} 
 
// 将上面两个函数指定为驱动的入口和出口函数
module_init(hello_init);
module_exit(hello_exit);
// LICENSE 和作者信息
MODULE_LICENSE("GPL");
MODULE_AUTHOR("zuomu");

编写应用测试程序如下所示:

#include 
#include 
#include 
#include 
#include 
int main(int argc,char *argv[])
{
    int fd;
    char buf[64] = {0};
    fd = open("/dev/chrdev_test",O_RDWR); //打开设备节点
    if(fd < 0)
    {
         perror("open error 
");
         return fd;
    } 
    close(fd);
    return 0;
}

输入命令编译app.c ,利用驱动里面的Makefile文件实现。

第七:具体效果如下:

将前面加载的驱动卸载掉,再加载新编译好的的驱动, 如下图所示:
rmmod chrdev
insmod chrdev.ko

c2a2be98-5d1f-11ed-a3b6-dac502259ad0.png

输入以下命令查看/sys/class 下面是否生成类, 如下图所示:ls /sys/class/chrdev_class/

输入以下命令查看下是否生成了设备节点ls /dev/chrdev_test

c2beedca-5d1f-11ed-a3b6-dac502259ad0.png

总结:利用标准字符驱动模型,自动生成设备节点,在开发过程具有重要意义。

审核编辑:郭婷


声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 嵌入式
    +关注

    关注

    4984

    文章

    18299

    浏览量

    288638
  • Linux
    +关注

    关注

    87

    文章

    10994

    浏览量

    206749

原文标题:Linux系统下自动创建设备节点方法

文章出处:【微信号:嵌入式开发爱好者,微信公众号:嵌入式开发爱好者】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    Linux添加磁盘创建分区、挂载

    Linux添加磁盘创建分区、挂载
    发表于 04-20 17:49 101次阅读
    <b class='flag-5'>Linux</b>添加磁盘<b class='flag-5'>创建</b>分区、挂载

    英特尔旗下自动驾驶公司发布DXP操作系统

    英特尔旗下自动驾驶技术公司Mobileye Global在CES 2024上推出了一款名为DXP的操作系统,旨在帮助汽车制造商开发特定的自动驾驶系统。这款新
    的头像 发表于 01-10 14:47 286次阅读

    何在LoRaWAN网关的内置NS创建应用与节点设备

    本文章将说明如何在LoRaWAN网关的内置NS创建应用与节点设备,配置安信可Ra-08模块连接至网关NS服务,实现数据发送与接收。注: 该教程仅适用于支持Class B模式通讯的网关。
    的头像 发表于 12-16 16:18 703次阅读
    如<b class='flag-5'>何在</b>LoRaWAN网关的内置NS<b class='flag-5'>创建</b>应用与<b class='flag-5'>节点</b><b class='flag-5'>设备</b>

    何在Linux中使用htop命令

    本文介绍如何在 Linux 中使用 htop 命令。
    的头像 发表于 12-04 14:45 555次阅读
    如<b class='flag-5'>何在</b><b class='flag-5'>Linux</b>中使用htop命令

    【米尔-TIAM62开发板-接替335x-试用评测】+(三)手把手创建Uboot设备树与内核设备树实战

    节点,以进一步描述硬件的详细信息。例如,网络接口节点可能包含一个子节点,描述MAC地址、IP地址等信息。 当U-Boot或Linux内核引导时,它们会读取和解析这个
    发表于 11-28 09:54

    ROS主控如何创建设备别名

    创建设备别名 需要创建设备别名原因: 在运行一个ros程序的时候需要提供一个端口名,这个端口名一般是ttyUSBx,设备每次插拔对应的这个端口名它都会不一样,需要创建一个
    的头像 发表于 11-17 18:07 379次阅读
    ROS主控如何<b class='flag-5'>创建设备</b>别名

    linux创建group的命令

    Linux中,创建用户组的命令是 groupadd 。 以下是一个例子: groupadd groupName 这个命令将创建一个新的用户组,名称为groupName。 在创建用户组
    的头像 发表于 11-08 11:35 629次阅读

    何在Linux创建节点

    手把手教你在 Linux创建节点,使其可以进行 cat 和 echo 。 我们测试驱动加载是否正常工作,一般都会写应用程序去测试,这样驱动程序中需要实现 open、read 函数和 write
    的头像 发表于 10-07 15:25 582次阅读
    如<b class='flag-5'>何在</b><b class='flag-5'>Linux</b>中<b class='flag-5'>创建</b><b class='flag-5'>节点</b>

    何在linux系统上移植驱动 - 第6节

    Linux系统
    充八万
    发布于 :2023年09月01日 19:13:36

    linux操作系统中的进程创建和销毁函数解析

    第一次遇见创建进程是在Linux启动流程中,reset_init函数调用kernel_thread函数创建了2个内核进程:kernel_init和kthreadd。
    发表于 06-26 09:12 415次阅读
    <b class='flag-5'>linux</b>操作<b class='flag-5'>系统</b>中的进程<b class='flag-5'>创建</b>和销毁函数解析

    智能办公自动系统创建

    电子发烧友网站提供《智能办公自动系统创建.zip》资料免费下载
    发表于 06-25 15:35 0次下载
    智能办公<b class='flag-5'>自动</b>化<b class='flag-5'>系统</b>的<b class='flag-5'>创建</b>

    Linux驱动中创建procfs接口的方法

    上篇介绍了Linux驱动中sysfs接口的创建,今天介绍procfs接口的创建
    发表于 05-31 16:48 571次阅读
    <b class='flag-5'>Linux</b>驱动中<b class='flag-5'>创建</b>procfs接口的方法

    何在yocto中创建新相机(IMX219)设备树文件和.ko文件?

    我正在研究 I.MX 8 QM SBC。我需要如何在 yocto 中创建新相机(IMX219)设备树文件和 .ko 文件的步骤。
    发表于 05-24 07:29

    Linux中如何如何为现有用户创建主目录?

    Linux系统中,每个用户都有一个主目录,通常称为home目录,用于存储用户的个人文件和配置信息。然而,有时候我们会创建一个新的用户,但是忘记或者没有选择为其创建一个主目录。在这种情
    的头像 发表于 05-12 16:24 2026次阅读
    <b class='flag-5'>Linux</b>中如何如何为现有用户<b class='flag-5'>创建</b>主目录?

    何在没有udev的情况下启动之前创建静态/dev/节点

    / 节点,所有 /dev/ 节点必须在 rootfs 启动之前创建“ [color=\\\"#FF0000\\\"]我想知道如何在没有 udev 的情况下启动之前
    发表于 05-05 10:27