6.22 驱动开发作业

news2025/1/23 2:10:14

字符设备驱动内部实现原理

1.字面理解解析:

字符设备驱动的内部实现有两种情况:

情况1.应用层调用open函数的内部实现:

open函数的第一个参数是要打开的文件的路径,根据这个路径 虚拟文件系统层VFS 可以找到这个文件在文件系统中唯一的标识,也就是inode号,通过inode号作为索引可以找到储存在内核中的struct inode结构体,struct inode结构体内部储存着 struct cdev结构体 和 储存该文件设备号的变量dev,因为设备文件想联系设备驱动,就要在inode结构体中保存该驱动的设备号 通过解析struct cdev结构体可知:结构体内部也有储存设备号的变量dev和操作方法结构体指针,通过操作方法结构体指针 VFS 就可以帮助我们回调对应的 mycdev_open 函数

open函数回调实现路线:

1.应用层open函数+打开路径参数 ----> 2.VFS层 --->3.对应设备文件inode号--->4.索引得 对应的struct inode结构体---> 5.struct cdev结构体---> 6.操作方法结构体、设备号---> 7.回调对应操作函数 myopen

情况2.应用层调用write/read等函数的内部实现:

write/read 函数没有指定路径的参数,换成了使用从open函数返回值得到的文件描述符来进行回调对应的操作方法 首先,当一个进程运行在操作系统中,那么就一定会在内核中的task_struct结构体空间中封存放进程的相关信息, 在task_struct结构体中, 有着存放着打开文件相关的结构体成员struct files_struct ,files_struct结构体成员struct file __rcu * fd_array[fd] 的下标 就是文件描述符的本质,这个结构体指针指向的结构体类型struct file 里就有操作方法结构体,通过文件描述符就可以确认是数组的哪个下标成员,VFS 虚拟文件系统层 来帮助我们回调对应的操作方法

read、write函数回调实现路线:

1.应用层write/read函数+fd文件描述符参数 ---> 2.VFS层---> 3.task_struct结构体--->  4.struct files_struct *files; //打开的文件相关结构体---> 5.struct file __rcu * fd_array[NR_OPEN_DEFAULT];//结构体指针数组,fd本质就是这个数组的下标---> 6.确定是数组中哪个struct file类型的成员---> 7.调用操作方法结构体成员---> 8.回调对应read、write函数

 

文件信息结构体:
struct inode
{
    umode_t i_mode;
    unsigned short i_opflags;
    kuid_t i_uid;
    kgid_t i_gid;
    dev_t i_rdev;
    union
    {
        struct block_device *i_dev;
        struct cdev;
        char *i_link;
        unsigned i_dir_seq;
    };
};



字符设备驱动对象结构体:
struct cdev {
    struct kobject kobj;
    struct module *owner;//THIS_MODULE
    const struct file_operations *ops;//操作方法结构体
    struct list_head list;//构成链表
    dev_t dev;//设备号
  unsigned int count;//设备数量
};


分步注册流程和代码实例:

#include <linux/init.h>
#include <linux/module.h>
#include <linux/fs.h>
#include <linux/io.h>
#include <linux/device.h>
#include <linux/slab.h>
#include <linux/cdev.h>

unsigned int major;
dev_t devno;
struct cdev *cdev;
unsigned int major=500;
unsigned int minor=0;
struct class *cls;
struct device *dev;

char kbuf[128]={0};

 int mycdev_open(struct inode *inode, struct file *file)
 {
    return 0;
 }

 ssize_t mycdev_read(struct file *file, char *ubuf, size_t size, loff_t *iof)
 {
    return 0;
 }

ssize_t mycdev_write(struct file *file, const char  *ubuf, size_t size, loff_t *iof)
{
    return 0;
}

int mycdev_close(struct inode *inode, struct file *file)
{
    return 0;
}

//定义一个操作方法结构体变量并且初始化
//结构体解析:需要使用一个结构体内的成员时才需要初始化这个变量的成员,
//此处需要准备4个函数初始化函数指针成员
//如果是结构体指针则需要实例化一个对应的结构体变量,或者指向函数申请的堆区空间
struct file_operations fops={
    .open=mycdev_open,
    .read=mycdev_read,
    .write=mycdev_write,
    .release=mycdev_close,
};


static int __init mycdev_init(void)
{
    int ret,i;

    //1.分配字符设备驱动对象空间
    cdev=cdev_alloc();
    if(cdev==NULL)
    {
        printk("分配字符设备驱动对象失败\n");
        ret=-EFAULT;
        goto LOOP1;
    }
    printk("分配对象空间成功\n");

    //2.字符驱动对象初始化
    cdev_init(cdev,&fops);

    //3.申请设备号
    //静态指定设备号
    if(major>0)
    {
        ret=register_chrdev_region(MKDEV(major,minor),3,"myled");
        if(ret)
        {
            printk("静态指定设备号失败\n");
            goto LOOP2;
        }
    }
    else if(major==0) //动态申请设备号
    {
        ret=alloc_chrdev_region(&devno,minor,3,"myled");
        if(ret)
        {
            printk("动态申请设备号失败\n");
             goto LOOP2;
        }
        major=MAJOR(devno);
        minor=MINOR(devno);
    }
    printk("申请设备号成功\n");
  
    //4.添加字符设备驱动对象注册进内核
    ret=cdev_add(cdev,MKDEV(major,minor),3);
    if(ret)
    {
        printk("字符设备驱动对象注册失败\n");
        goto LOOP3;
    }
    printk("添加字符设备驱动对象注册进内核成功\n");

    cls=class_create(THIS_MODULE,"myled");
    if(IS_ERR(cls))
    {
        printk("向上提交目录失败\n");
        ret=-PTR_ERR(cls);
        goto LOOP4;
    }
    printk("向上提交目录成功\n");

    //向上提交设备节点信息
    for(i=0;i<3;i++)
    {
        dev=device_create(cls,NULL,MKDEV(major,i),NULL,"myled%d",i);
        if(IS_ERR(dev))
        {
            ret=-PTR_ERR(dev);
            goto LOOP5;
        }
    }

    return 0;

LOOP5:
    //释放已经申请的设备节点信息
    for(--i;i>=0;i--)
    {
        device_destroy(cls,MKDEV(major,i));
    }

    //释放目录空间
    class_destroy(cls);

LOOP4:
    //注销字符设备驱动对象
    cdev_del(cdev);

LOOP3:
    //释放设备号
    unregister_chrdev_region(MKDEV(major,minor),3);

LOOP2:
    kfree(cdev);    

LOOP1:
    return ret; 
}

static void __exit mycdev_exit(void)
{
    //销毁设备节点
    int i;
    for(i=0;i<3;i++)
    {
        device_destroy(cls,MKDEV(major,i));
    }

    //释放目录空间
    class_destroy(cls);

    //注销字符设备驱动对象
    cdev_del(cdev);

    //释放设备号
    unregister_chrdev_region(MKDEV(major,minor),3);

    //释放对象空间
    kfree(cdev);
}

module_init(mycdev_init);
module_exit(mycdev_exit);
MODULE_LICENSE("GPL");

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Cookie,Session,Token,JWT授权方式对比

文章目录 HTTPCookieSessionSession认证流程Session 共享方案 TokenToken认证流程 JWTJWT认证流程 HTTP HTTP 本质上是无状态的&#xff0c;每个请求都是互相独立、毫无关联的&#xff0c;协议不要求客户端或服务器记录请求相关的信息。服务端无法确认当前访问者的身份信息&…

升级一下《单词猜谜》

网上的单词猜谜都是英文的&#xff0c;不会英语的头痛 于是&#xff0c;我把《单词猜谜》改成中文&#xff0c;并且加上了点新功能&#xff1a; import random as rdmc rdm.randint(1, 3)name1 input("你的名字叫什么&#xff1f;&#xff1a;") if c 1:turns1 …

详解BigDecimal

目录 1.概述 2.基本API 2.1.创建 BigDecimal 对象&#xff1a; 2.3.基本运算方法&#xff1a; 2.4.精度控制方法&#xff1a; 2.5.比较 2.6.转换 3.注意事项 4.底层实现原理 1.概述 精度丢失&#xff0c;由于现代计算机中采用了浮点数来表示小数&#xff0c;这种表示…

小马哥训练营-Java EE单体架构

什么是Servlet Servlet 是一种基于 Java 技术的 Web 组件&#xff0c;用于生成动态内容&#xff0c;由容器管理。类似于其他 Java 技术组件&#xff0c;Servlet 是平台无关的 Java 类组成&#xff0c;并且由 Java Web 服务器加载执行。通常情况&#xff0c;由 Servlet 容器提供…

(十一)CSharp-LINQ-查询表达式(2)

一、查询表达式 1、查询表达式的结构 查询表达式由 from 子句和查询主体组成。 子句必须按照一定的顺序出现。from 子句和 select…group 子句这两部分是必需的。其他子句是可选的。在 LINQ 查询表达式中&#xff0c;select 子句在表达式最后。可以有任意多的 from…let…wh…

Nginx服务器的六个修改小实验

一、Nginx虚拟主机配置 1.基于域名 &#xff08;1&#xff09;为虚拟主机提供域名解析 配置DNS 修改/etc/hosts文件 &#xff08;2&#xff09;为虚拟主机准备网页文档 #创建网页目录 mkdir -p /var/www/html/abc mkdir -p /var/www/html/def ​ #编写简易首页html文件 ec…

Finalshell安全吗?Xshell怎么样?

文章目录 一、我的常用ssh连接工具二、Xshell2.1 下载&#xff1a;认准官网2.2 Xshell 配置2.3 Xftp和WinSCP 一、我的常用ssh连接工具 之前讲过&#xff1a; 【服务器】远程连接选SSH&#xff08;PUTTY、Finalshell、WinSCP&#xff09; 还是 远程桌面&#xff08;RDP、VNC、…

代码随想录训练营第四十二天|01背包、416.分割等和子集

01背包 代码随想录理论讲解 背包问题分类 01背包 问题描述 有n件物品和一个最多能背重量为w的背包。第i件物品的重量是weight[i]&#xff0c;得到的价值是value[i]。每件物品只能用一次&#xff0c;求解将哪些物品装入背包里物品价值总和最大。 动归五部曲 确定dp数组及下…

基于java web高校社交系统 /springboot高校社交系统

摘 要 随着信息技术和网络技术的飞速发展&#xff0c;人类已进入全新信息化时代&#xff0c;传统管理技术已无法高效&#xff0c;便捷地管理信息。为了迎合时代需求&#xff0c;优化管理效率&#xff0c;各种各样的系统应运而生&#xff0c;各行各业相继进入信息管理时代&#…

CSS处理页面元素浮动的几个办法

CSS处理页面元素浮动的几个办法 不使用浮动的情况下&#xff0c;子盒子1和2分别在盒子1中占据一行的空间。 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta http-equiv"X-UA-Compatible" cont…

Java 8新特性:方法引用的介绍与使用

&#x1f389;&#x1f389;&#x1f389;点进来你就是我的人了博主主页&#xff1a;&#x1f648;&#x1f648;&#x1f648;戳一戳,欢迎大佬指点! 欢迎志同道合的朋友一起加油喔&#x1f93a;&#x1f93a;&#x1f93a; 目录 1. 什么是方法引用&#xff1f; 2. 方法引用的…

【30天熟悉Go语言】9 Go函数全方位解析

作者&#xff1a;秃秃爱健身&#xff0c;多平台博客专家&#xff0c;某大厂后端开发&#xff0c;个人IP起于源码分析文章 &#x1f60b;。 源码系列专栏&#xff1a;Spring MVC源码系列、Spring Boot源码系列、SpringCloud源码系列&#xff08;含&#xff1a;Ribbon、Feign&…

【好书精读】网络是怎样连接的 之 数据收发完成之后 从服务器断开并删除套接字

&#xff08; 该图由AI制作 &#xff09; 目录 数据收发完成后协议栈要执行的操作 数据发送完毕后断开连接 删除套接字 数据收发操作小结 第一步是创建套接字 然后 客户端会向服务器发起连接操作 数据收发阶段 执行断开操作 数据收发完成后协议栈要执行的操作 数据发…

html_css模拟端午赛龙舟运动

文章目录 ⭐前言&#x1f496; 样式布局&#x1f496; 添加龙舟&#x1f496; 添加css_animation运动 ⭐结束 ⭐前言 大家好&#xff0c;我是yma16&#xff0c;本期给大家分享css实现赛龙舟运动。 &#x1f496; 样式布局 风格&#xff1a;卡通 首先采用一张包括水元素的照片…

液体泄露识别检测算法 监控识别管道液体泄漏

液体泄露识别检测算法通过 yolov8python网络模型技术&#xff0c;液体泄露识别检测算法对管道的液体泄露情况进行全天候不间断实时监测&#xff0c;检测到画面中管道设备液体泄露现象时&#xff0c;将自动发出警报提示相关人员及时采取措施。YOLOv8 算法的核心特性和改动可以归…

硬件入门之什么是二级管

硬件入门之什么是二级管 文章目录 硬件入门之什么是二级管一、二极管是什么&#xff1f;二极管在电路中的作用很广泛&#xff1a; 单向导电&#xff0c;续流&#xff0c;稳压等等 二、分类二极管实际应用场景二极管分类1.通用二极管2..肖特二极管 3.稳压二极管&#xff08;利用…

零基础学会python编程——输入 / 输出函数与变量

作者简介&#xff1a;一名云计算网络运维人员、每天分享网络与运维的技术与干货。 座右铭&#xff1a;低头赶路&#xff0c;敬事如仪 个人主页&#xff1a;网络豆的主页​​​​​ 目录 前言 学习目标 一.输入与输出函数 1.print 函数 【例2-1】 【例2-2】 【例2-3】 …

深度详解Nginx正向代理与反向代理

正向代理和反向代理的概念 1、正向代理2、反面代理以租房为例解释正向代理和反向代理&#xff1f;正向代理和反向代理配置 1、正向代理 它的工作原理就像一个跳板,简单的说,我是一个用户,我访问不了某网站,但是我能访问一个代理服务器&#xff0c;这个代理服务器呢,他能访问那个…

一文弄懂git原理及相关操作

文章目录 前言Git 是什么&#xff1f;Git 与 SVN 的区别Git 快速入门Git 相关网站Git 工作流程Git 工作空间Git 文件状态 Git 安装在 Linux 上安装在 Mac 上安装在 Windows 上安装从源代码安装 Git 配置Git 常用命令git initgit clonegit addgit commitgit branchgit tag查看信…

字符设备驱动内部实现原理解析

字符设备驱动内部实现原理解析 一. 字符设备驱动对象内部实现原理解析二. 字符设备驱动的注册流程三. 代码示例 一. 字符设备驱动对象内部实现原理解析 用户层&#xff1a; ​ 当用户打开&#xff08;open&#xff09;一个文件时,会生成一个文件描述符表 内核层&#xff1a; 内…