【Linux驱动层】iTOP-RK3568学习之路(三):字符设备驱动框架


一、总体框架图


在这里插入图片描述


二、字符设备相关函数


静态申请设备号 register_chrdev_region

函数原型:
	register_chrdev_region(dev_t from, unsigned count, const char *name)
函数作用:
	静态申请设备号,可以一次性申请多个连续的号,count指定申请的个数,from指定起始的设备号
参数含义:
    from: 自定义的 dev_t 类型设备号
    count: 申请设备的数量
    name: 申请的设备名称
函数返回值:
	申请成功返回 0,申请失败返回负数

动态申请设备号 alloc_chrdev_region

函数原型:
	alloc_chrdev_region(dev_t *dev, unsigned baseminor, unsigned count,const char *name)
函数作用:
	动态申请设备号,可以一次性申请多个连续的号,count指定申请的个数,baseminor指定起始的次设备号,
    相较于静态申请设备号,动态申请会避免注册设备号相同引发冲突的问题。
参数含义:
    dev: 分配得到的第一个设备号
    baseminor: 次设备号可申请的最小值,我一般写0
    count: 申请设备的数量
    name: 申请的设备名称
函数返回值:
	申请成功返回 0,申请失败返回负数
ps: 
	dev2.dev_num = dev1.dev_num + 1        

字符设备初始化 cdev_init


cdev 结构体

struct cdev {
    struct kobject kobj; 		//内嵌的内核对象. 
    struct module *owner; 		//该字符设备所在的内核模块的对象指针. 
    const struct file_operations *ops; //该结构描述了字符设备所能实现的方法,是极为关键的一个结构体.
    struct list_head list; 		//用来将已经向内核注册的所有字符设备形成链表. 
    dev_t dev; 					//字符设备的设备号,由主设备号和次设备号构成. 
    unsigned int count; 		//隶属于同一主设备号的次设备号的个数. 
};

cdev_init在“内核源码/include/fs/char_dev.c文件中定义

void cdev_init(struct cdev *cdev, const struct file_operations *fops)
{
    memset(cdev, 0, sizeof *cdev);		//将整个结构体清零;
    INIT_LIST_HEAD(&cdev->list);		//初始化 list 成员使其指向自身;
    kobject_init(&cdev->kobj, &ktype_cdev_default);		//初始化 kobj 成员;
    cdev->ops = fops;					//初始化 ops 成员,建立 cdev 和 file_operations 之间的连接
}

void cdev_init(struct cdev *, const struct file_operations *);
函数作用:
    初始化传入的 cdev 类型的结构体,并与自定义的 file_operations * 类型的结构体进行链接。
参数含义:
    cdev: 要传入的 cdev 类型结构体,为要初始化的字符设备。
    fops:要传入的 file_operations * 类型结构体,关于 file_operations 结构体的相关的知识会在下一章节进行讲解。
函数返回值:
    无返回值。

字符设备的注册 cdev_add


函数原型:
	int cdev_add(struct cdev *p, dev_t dev, unsigned count)
函数作用:
	该函数向内核注册一个 struct cdev 结构体
参数含义:
    (1)第一个参数为要添加的 struct cdev 类型的结构体
    (2)第二个参数为申请的字符设备号
    (3)第三个参数为和该设备关联的设备编号的数量。
函数返回值:
    添加成功返回 0,添加失败返回负数。

字符设备的注销 cdev_del


函数原型:
	void cdev_del(struct cdev *p);
函数作用:
	该函数会向内核删除一个 struct cdev 类型结构体
参数含义:
	该函数只有一个参数,为要删除的 struct cdev 类型的结构体
函数返回值:无返回值

手动创建设备节点

原型:
	mknod NAME TYPE MAJOR MINOR
参数含义:
    NAME: 要创建的节点名称
    TYPE: b 表示块设备,c 表示字符设备,p 表示管道
    MAJOR:要链接设备的主设备号
    MINOR: 要链接设备的从设备号

例如,使用以下命令创建一个名为 device_test 的字符设备节点,链接设备的主设备号和从设备号分别为 236 和 0:

 mknod /dev/device_test c 236 0

class_create(…)函数

函数原型:
    #define class_create(owner, name) 
    ({ 
        static struct lock_class_key __key; 
        __class_create(owner, name, &__key); 
    })

函数作用:
    用于动态创建设备的逻辑类,并完成部分字段的初始化,然后将其添加进 Linux 内核系统。
参数含义:
	owner:struct module 结构体类型的指针,指向函数即将创建的这个 struct class 的模块。一般赋值为 THIS_MODULE。
name:
    char 类型的指针,代表即将创建的 struct class 变量的名字。
返回值:
    struct class * 类型的结构体。

class_destroy(…)函数

函数原型:
	extern void class_destroy(struct class *cls);
函数作用:
	用于删除设备的逻辑类,即从 Linux 内核系统中删除设备的逻辑类。
参数含义:
	cls:代表即将要删除的 struct class 变量的名字。
返回值:无

device_create(…)函数

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

函数作用:
	用来在 class 类中下创建一个设备属性文件,udev 会自动识别从而进行设备节点的创建。
参数含义:
    cls:指定所要创建的设备所从属的类。
    parent:指定该设备的父设备,如果没有就指定为 NULL。
    devt:指定创建设备的设备号。
    drvdata:被添加到该设备回调的数据,没有则指定为 NULL。
    fmt:添加到系统的设备节点名称。
返回值:
    struct device * 类型结构体

device_destroy(…)函数

函数原型:
	extern void device_destroy(struct class *cls, dev_t devt);
函数作用:
	用来删除 class 类中的设备属性文件,udev 会自动识别从而进行设备节点的删除。
参数含义:
    cls:指定所要创建的设备所从属的类。
    devt:指定创建设备的设备号。
返回值:无

container_of 函数

函数原型:
	container_of(ptr,type,member);
函数作用:
	通过结构体变量中某个成员的首地址获取到整个结构体变量的首地址。
参数含义:
    ptr 是结构体变量中某个成员的地址。
    type 是结构体的类型
    member 是该结构体变量的具体名字
container_of 宏的作用是通过结构体内某个成员变量的地址和该变量名,以及结构体类型。找到该结构体变量的地址。

三、实例代码


file.c

#include <linux/module.h>
#include <linux/init.h>
#include <linux/moduleparam.h>
#include <linux/fs.h>
#include <linux/kdev_t.h>
#include <linux/cdev.h>
#include <linux/device.h>
#include <linux/uaccess.h>

struct device_test{
    int major;                  // 主设备号
    int minor;                  // 次设备号
    dev_t dev_num;              // 用来一个32位无符号整形的设备号
    struct cdev cdev_test;      // 定义一个字符设备的cdev结构体
    struct class *class;        // 类
    struct device *device;      // 设备文件
    char kbuf[32];
};

struct device_test dev1;        // 定义一个设备文件结构体


static int cdev_test_open(struct inode * inode, struct file * file){
    file->private_data = &dev1;     // 设置私有数据
    printk("This is cdev_test_open!\n");
    return 0;
}


static ssize_t cdev_test_read(struct file * file, char __user *buf, size_t size, loff_t *off){

    struct device_test *temp_dev = (struct device_test *)file->private_data;

    // strcpy(temp_dev->kbuf, "This is cdev_test_read!")   //模拟内核空间数据

    // copy_to_user 成功返回0,失败返回没有拷贝成功的数据字节数
    if(copy_to_user(buf, temp_dev->kbuf, strlen(temp_dev->kbuf)) != 0){
        printk("copy_to_user error\r\n"); //打印 copy_to_user 函数执行失败
        return -1;
    }

    printk("This is cdev_test_read!\n");
    return 0;
}


static ssize_t cdev_test_write(struct file * file, const char __user * buf, size_t size, loff_t *off){
    struct device_test *temp_dev = (struct device_test *)file->private_data;
    

    if(copy_from_user(temp_dev->kbuf, buf, size) != 0){   // copy_from_user:用户空间向内核空间传数据
        printk("copy_from_user error\r\n");//打印 copy_from_user 函数执行失败
        return -1;
    }
    printk("This is cdev_test_write!\n");
    printk("kbuf is %s! \n", temp_dev->kbuf);
    return 0;
}


static int cdev_test_release(struct inode * inode, struct file * file){
    printk("This is cdev_test_release!\n");
    return 0;
}


struct file_operations cdev_test_ops={
    .owner = THIS_MODULE,    // 文件操作结构体
    .open = cdev_test_open,
    .read = cdev_test_read,
    .write = cdev_test_write,
    .release = cdev_test_release,
};


static int chardev_init(void){
    int ret;     //返回值判断
   
    // 动态注册一个设备号,成功返回0,失败返回复数
    ret = alloc_chrdev_region(&dev1.dev_num, 0, 1, "char_devname");
    if(ret < 0){
        printk("动态注册设备号失败!\n");
        goto err_alloc;
    }
    printk("动态注册设备号成功!\n");
    dev1.major = MAJOR(dev1.dev_num);     //通过 MAJOR()函数进行主设备号获取
    dev1.minor = MINOR(dev1.dev_num);     //通过 MINOR()函数进行次设备号获取
    printk("major = %d!\n",dev1.major);
    printk("minor = %d!\n",dev1.minor);

    cdev_init(&dev1.cdev_test, &cdev_test_ops);  //初始化字符设备结构体cdev_test,并连接到文件操作结构体cdev_test_ops
    dev1.cdev_test.owner = THIS_MODULE;
    ret = cdev_add(&dev1.cdev_test, dev1.dev_num, 1); //使用cdev_add()函数进行字符设备的添加。
    if(ret < 0){
        printk("字符设备注册失败!\n");
        goto err_cdev_add;
    }
    printk("字符设备注册成功!\n");

    dev1.class = class_create(THIS_MODULE, "class");   // 初始化class_create
    if(IS_ERR(dev1.class)){
        ret = PTR_ERR(dev1.class);
        goto err_class_create;
    }
    dev1.device = device_create(dev1.class, NULL, dev1.dev_num, NULL, "device"); //
    if(IS_ERR(dev1.device)){
        ret = PTR_ERR(dev1.device);
        goto err_device_create;
    }


err_device_create:
    class_destroy(dev1.class);      //设备节点建立失败,删除类

err_class_create:
    cdev_del(&dev1.cdev_test);      //类创建失败,删除设备cdev

err_cdev_add:
    unregister_chrdev_region(dev1.dev_num, 1);  //注册设备失败,释放设备号

err_alloc:
    return ret;     //申请设备号失败,返回

    return 0;
}


static void chardev_exit(void){
    //先删除字符设备
    cdev_del(&dev1.cdev_test);
    //再释放字符驱动设备号
    unregister_chrdev_region(dev1.dev_num, 1);

    //删除创建的设备
    device_destroy(dev1.class, dev1.dev_num);
    //删除创建的类
    class_destroy(dev1.class);

    printk("chardev_exit!\n");
}


module_init(chardev_init);
module_exit(chardev_exit);


MODULE_LICENSE("GPL v2");
MODULE_AUTHOR("ZSY");
MODULE_VERSION("V1.0");

Makefile

# 只对当前终端有效
export ARCH=arm64
export CROSS_COMPILE=/home/topeet/source/linux/rk356x_linux/prebuilts/gcc/linux-x86/aarch64/gcc-linaro-6.3.1-2017.05-x86_64_aarch64-linux-gnu/bin/aarch64-linux-gnu-

obj-m += file.o
KDIR := /home/topeet/source/linux/rk356x_linux/kernel
PWD ?= $(shell pwd)
all:
	make -C $(KDIR) M=$(PWD) modules
clean:
	make -C $(KDIR) M=$(PWD) clean #make clean 操作 

app.c

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>

int main(int argc, char *argv[]){
    int fd;

    char buf1[32] = {0};        // 定义读写缓存区buf1
    char buf2[32] = "nihao!";   // 定义读写缓存区buf2

    fd = open("/dev/device",O_RDWR);   // 打开字符设备驱动文件"device_test"
    if(fd < 0){
        perror("open error \n");
        return fd;
    }

    // read(fd, buf1, sizeof(buf1));
    // printf("buf1 is %s! \n", buf1);

    write(fd, buf2,sizeof(buf2));

    close(fd);
    return 0;
}




本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/564227.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Python读取influxDB数据库(二)(influxDB2.X版本)

1. influxDB连接 首先在浏览器中输入influxDB的IP和端口&#xff0c;然后输入账号密码进入到influxDB数据库来进行数据的相关操作&#xff1a; 里面的bucket相当于sql中的数据库&#xff0c;_measurement相当于sql中的表 2. 获取influxDB数据库的token方法 3. 写查询语句来查询…

新火种AI|号称“史上最强大开源模型”的Llama3,凭什么价值百亿美金?

作者&#xff1a;小岩 编辑&#xff1a;彩云 4月19日&#xff0c;Facebook母公司Meta重磅推出了Llama3。 即便大家现在对于大厂和巨头频繁迭代AI模型的行为已经见怪不怪&#xff0c;Meta的Llama3仍旧显得与众不同&#xff0c;因为这是迄今最强大的开源AI模型。 Meta推出了重…

Redis从入门到精通(二十一)Redis最佳实践(二)mset、pipeline、慢查询优化、内存划分

文章目录 前言7.2 批处理优化7.2.1 命令执行流程7.2.2 mset7.2.3 Pipeline7.2.4 集群下的批处理7.2.4.1 问题与解决方案7.2.4.2 基于Spring的串行化执行 7.3 服务器端优化7.3.1 持久化配置7.3.2 慢查询优化7.3.2.1 什么是慢查询7.3.2.2 如何查看慢查询 7.3.3 命令及安全配置7.3…

智慧安防边缘计算硬件AI智能分析网关V4算法启停的操作步骤

TSINGSEE青犀视频智能分析网关V4内置了近40种AI算法模型&#xff0c;支持对接入的视频图像进行人、车、物、行为等实时检测分析&#xff0c;上报识别结果&#xff0c;并能进行语音告警播放。硬件管理平台支持RTSP、GB28181协议、以及厂家私有协议接入&#xff0c;可兼容市面上常…

MySQL 基础语法(2)

文章目录 创建表查看表修改表表数据插入 本文为表结构相关的基础语言库相关的基础语句 创建表 CREATE TABLE table_name ( field1 datatype comment xxx, field2 datatype, field3 datatype ) character set 字符集 collate 校验规则 engine 存储引擎;CREATE TABLE&#xff1…

算法竞赛相关问题总结记录

前言 日常在校生或者是工作之余的同学或多或少都会参加一些竞赛,参加竞赛一方面可以锻炼自己的理解与实践能力&#xff0c;也能够增加自己的生活费&#xff0c;竞赛中的一些方案也可以后续作为自己论文的base,甚至是横向课题的框架。在算法竞赛中算法的差别个人感觉差距都不大&…

创建一个空的maven项目,整合SpringBoot和Redis

创建一个空的maven项目&#xff0c;整合SpringBoot和Redis 创建空的maven项目 在最新版的idea中创建maven项目的时候会让选择模板 如下图&#xff1a; 我们选择quickstart快速开始模板&#xff0c;quickstart快速开始模板创建的maven项目里面什么都不带&#xff0c;只有一个…

便携式手提万兆网络协议测试仪

便携式手提万兆网络协议测试仪 平台简介 便携式手提万兆网络协议测试仪&#xff0c;以FPGA万兆卡和X86主板为基础&#xff0c;构建便携式的手提设备。 FPGA万兆卡是以Kintex-7XC7K325T PCIeX4的双路万兆光纤网络卡&#xff0c;支持万兆网络数据的收发和网络协议的定制设计。 …

微服务之.SpringCloud AlibabaSentinel实现熔断与限流

一、概述 1.1介绍 Sentinel是阿里巴巴开源的一款服务保护框架&#xff0c;目前已经加入SpringCloudAlibaba中。官方网站&#xff1a; 官网https://sentinelguard.io/zh-cn/ 从流量路由、流量控制、流量整形、熔断降级、系统自适应过载保护、热点流量防护等多个维度来帮助开…

计算机服务器中了locked勒索病毒怎么办,locked勒索病毒解密工具流程步骤

随着网络技术的不断应用与发展&#xff0c;越来越多的企业离不开网络&#xff0c;网络大大提升了企业的办公效率水平&#xff0c;也为企业的带来快速发展&#xff0c;对于企业来说&#xff0c;网络数据安全成为了大家关心的主要话题。近日&#xff0c;云天数据恢复中心接到多家…

注意libaudioProcess.so和libdevice.a是不一样的,一个是动态链接,一个是静态

libaudioProcess.so是动态链接&#xff0c;修改需要改根文件系统&#xff0c;需要bsp重新配置 libdevice.a是静态链接&#xff0c;直接替换就行 动态链接文件修改 然后执行fw_update.sh

JAVA学习笔记29(集合)

1.集合 ​ *集合分为&#xff1a;单列集合、双列集合 ​ *Collection 接口有两个重要子接口 List Set&#xff0c;实现子类为单列集合 ​ *Map接口实现子类为双列集合&#xff0c;存放的King–Value ​ *集合体系图 1.1 Collection接口 1.接口实现类特点 1.collection实现…

射频识别技术助力产品分拣:提升效率与准确性

射频识别技术助力产品分拣&#xff1a;提升效率与准确性 RFID技术在产品分拣中具有重要的应用&#xff0c;它利用射频信号进行非接触式的自动识别&#xff0c;能够高效、准确地完成产品分拣工作。 在产品分拣中&#xff0c;RFID技术的主要应用方式是在产品上粘贴RFID电子标签&…

阿里云mysql8.0 this is incompatible withsql mode=only full group by

阿里云RDS中mysql5.6升级为8.0后&#xff0c;出现如下问题&#xff1a; ### Error querying database. Cause:java.sql.SQLSyntaxErrorException: Expression #1 of SELECT listis not in GROUP BY clause and contains nonaggregatedcolumn temp.product_id which is not fun…

电商平台数据有哪些(淘宝1688京东API)?如何进行电商平台数据分析?(内附测试方式)

电商平台数据是一个庞大且复杂的体系&#xff0c;涵盖了多个维度和类型。在淘宝、1688、京东等电商平台中&#xff0c;数据主要分为以下几个类别&#xff1a; 用户数据&#xff1a;包括用户属性&#xff08;如年龄、性别、地域、职业等&#xff09;、用户行为&#xff08;如浏…

本地环境测试

1. 在 Anaconda Navigator 中&#xff0c;打开 Jupyter Notebook &#xff0c;在网页中&#xff0c;点击进入本地环境搭建中创 建的工作目录&#xff0c;点击右上角的 New- 》 Folder &#xff0c;将新出现的 Untitled Folder 选中&#xff0c;并使用左上角 的 Rename 按钮重…

CSS基础常用属性之字体属性(如果想知道CSS的字体属性知识点,那么只看这一篇就足够了!)

前言&#xff1a;在我们学习CSS的时候&#xff0c;主要学习选择器和常用的属性&#xff0c;而这篇文章讲解的就是最基础的属性之一——文字属性。 ✨✨✨这里是秋刀鱼不做梦的BLOG ✨✨✨想要了解更多内容可以访问我的主页秋刀鱼不做梦-CSDN博客 废话不多说&#xff0c;让我们直…

STP学习的第一篇

1.STP的基本概念&#xff1a;根桥 &#xff08;1&#xff09;STP的主要作用之一是在整个交换网络中计算出一棵无环的“树”&#xff08;STP树&#xff09;。 &#xff08;2&#xff09;根桥是一个STP交换网络中的“树根”。 &#xff08;3&#xff09;STP开始工作后&#xf…

Hack The Box-Runner

总体思路 子域名扫描->CVE-2023-42793利用->获取敏感信息->user->端口转发->CVE-2024-21626利用->root 信息收集&端口利用 nmap -sSVC 10.10.11.13目标开放22、80、8000端口&#xff0c;这里先将runner.htb加入到hosts文件后&#xff0c;访问之 查看源…

一文读懂链游!探索链游的前世今生,区块链与游戏结合的新兴趋势

区块链技术的崛起给游戏行业带来了前所未有的变革&#xff0c;而链游&#xff08;Blockchain Games&#xff09;正是这一变革的产物。本文将带您一览链游的前世今生&#xff0c;探索区块链与游戏结合的新兴趋势。 1. 链游的起源 链游&#xff0c;顾名思义&#xff0c;是指利用…