基于Linux的驱动开发:内核模块传参、内核到处符号表、字符设备驱动

news2025/1/11 2:17:09

内核模块传参

        内核模块:

                int a , b;

        安装内核模块时:insmod demo.ko a = 100 b =10;

1.内核模块传参的意义

        在安装内核模块时给内核模块中的变量进行数值传递,这样可以让我们的内核模块向上兼容更为复杂的应用程序,向下适配多种硬件

2.内核模块传参相关API

        1.函数原型:module_param(name, type, perm)

        功能: 声明可以进行内核模块传参的变量

        参数:name : 变量名

                   type:要传参的数值类型---》byte(传递char), hexint(16进制的int)、short 、int、uint、long、ulong、charp(传递字符指针)、bool(0/1 y/n Y/N) 

                   perm:文件权限,通过module_param声明了要传参的变量,那么在sys/module/当前模块/parameters/下会生成一个以当前变量为名的文件,文件的权限为perm指定的权限,文件内容为变量的值

        注:通过modinfo查看您当前内核模块可以进行命令行传参的变量有哪些

        2.函数原型:MODULE_PARM_DESC(_parm, desc)

        功能:添加要传参的变量的描述,这个描述也可以通过modinfo查看

        参数:_parm :传参的变量名

                  desc:添加的描述

        注:给char类型的变量传参时,需要传递对应的ASCII十进制形式

                给字符指针传递字符串时,字符串不可以有空格,若有空格,空格前的被当作参数传递给变量,后面的会被认为时一个不认识的变量

示例代码 

内核导出符号表

 

1.导出符号表的意义

到处符号表可以让不同模块之间实现资源的相互访问 ,比如模块2想要访问模块1的资源,只需将模块1资源的符号表导出给模块2,此时,模块2就可以访问模块1的资源了

2.导出符号表的相关API

EXPORT_SYMBOL(变量名|函数名)

或者

EXPORT_SYMBOL_CPL(变量名|函数名)

3.导出符号表测试实例

3.1编写代码

定义demo1.c,完成函数的定义

#include <linux/init.h>
#include <linux/module.h>
int add(int i,int j)
{
    return i+j;
}
//生成add的符号表文件
EXPORT_SYMBOL(add);
static int __init mycdev_init(void)
{
    
    return 0;
}
static void __exit mycdev_exit(void)
{


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

 定义dem2.c,完成调用demo1.c中的函数

#include <linux/init.h>
#include <linux/module.h>

extern int add(int i,int j);
static int __init mycdev_init(void)
{
    printk("调用模块1函数执行结果为:%d",add(3,5));
    return 0;
}
static void __exit mycdev_exit(void)
{


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

3.2编译

先编译demo1.c将生成的符号表文件Module.symvers复制到demo2的路径下再编译demo2.c

此时通过modinifo查看demo2.ko,显示demo2依赖于demo1

注意 :在高版本的内核中不支持符号表文件直接复制,不然编译报未定义错误

解决办法:在demo2的Makefile中加上demo1符号表的文件路径3.3安装流程

先安装demo1,再安装demo2

3.4卸载

先卸载demo2,再卸载demo1

字符设备驱动

1.字符设备驱动的定义

字符设备是以字节流的形式进行顺序访问的设备,针对字节设备设计的驱动框架叫做字符设备驱动。当前市面上绝大多数的设备都属于字符设备,比如键盘、鼠标、摄像头...

2.字符设备驱动的框架

1.当在内核中注册一个字符设备驱动后会得到驱动对应的设备号,设备号是设备的唯一标识。设备号分为主设备号和次设备号,主设备号(高12位):用来标识一类设备 ,此设备号(低20位):用来标识这一类中的某一个设备。设备号=主设备号<<20 | 次设备号

2.根据设备号可以通过某种方式在文件系统中创建设备文件,相当于通过设备号将设备文件和驱动绑定

3.当设备文件被创建后,应用程序中调用 open() | read() | write() | close()

4.调用以上函数时,驱动中对应的操作应用方法会被回调

5.在驱动的操作方法中完成硬件的控制

3.字符设备驱动的注册和注销

3.1相关API 

头文件:#include <linux/fs.h>

注册字符设备驱动

int register_chrdev(unsigned int major, const char *name, const struct file_operations *fops)

功能:实现字符设备驱动的注册,一次申请256个设备的资源(0-255个此设备号)

参数:major:主设备号

                >0 静态指定主设备号

                =0 动态申请主设备号

           name:注册得到的驱动名字

           fops:操作方法结构体指针,指向操作方法结构体变量

返回值:

        失败返回错误码

        成功:若major>0,则返回申请得到主设备号

                   若major=0,则返回0

操作方法结构体,用于保存和管理驱动的各自操作方法

struct file_operations {

        int (*open) (struct inode *, struct file *);

        ssize_t (*read) (struct file *, char __user *, size_t, loff_t *);

        ssize_t (*write) (struct file *, const char __user *, size_t, loff_t *);

        int (*release) (struct inode *, struct file *);

};

注销字符设备驱动

void unregister_chrdev(unsigned int major, const char *name)

功能:实现字符设备驱动的注销

参数:major:驱动对应的主设备号

           name:注册时填写的驱动号

返回值:无

3.2字符设备驱动注册实例

1.编写代码

#include <linux/init.h>
#include <linux/module.h>
#include <linux/fs.h>
unsigned int major;
// 封装操作方法,这些操作方法在应用层进行系统调用时被回调
int mycdev_open(struct inode *inode, struct file *file)
{
    printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);
    return 0;
}
ssize_t mycdev_read(struct file *file, char *ubuf, size_t size, loff_t *lof)
{
    printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);
    return 0;
}
ssize_t mycdev_write(struct file *file, const char *ubuf, size_t size, loff_t *lof)
{
    printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);
    return 0;
}
int mycdev_close(struct inode *inode, struct file *file)
{
    printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);
    return 0;
}
// 定义操作方法结构体对象,保存封装的操作方法
struct file_operations fops = {
    .open=mycdev_open,
    .read=mycdev_read,
    .write=mycdev_write,
    .release=mycdev_close,
};
//入口函数
static int __init mycdev_init(void)
{
    // 注册字符设备驱动
    major = register_chrdev(0, "mychrdev", &fops);
    if (major < 0)
    {
        printk("字符设备驱动注册失败\n");
        return major;
    }
    printk("注册字符设备驱动成功major=%d\n", major);
    return 0;
}
//出口函数
static void __exit mycdev_exit(void)
{
    //注销字符设备驱动
    unregister_chrdev(major,"mychrdev");
}
module_init(mycdev_init);
module_exit(mycdev_exit);
MODULE_LICENSE("GPL");

2.编译驱动

3.安装驱动内核模块

可以查看/proc/devices文件,确定驱动是否被注册4.创建设备文件

创建设备文件的命令:mknod /dev/mychrdev c 240 0

解析:mknod:创捷设备文件的命令码

           /dev/mychrdev:创建的设备文件的路径和名字

          c 设备文件类型(字符设备文件) d(块设备文件)

          240:主设备号

          0:次设备号 0-255都可以

5.编写应用程序代码测试是否可以关联驱动

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
int main(int argc, char const *argv[])
{
    char buf[128] = {0};
    int fd = open("/dev/mychrdev", O_RDWR);
    if (fd < 0)
    {
        printf("打开设备文件失败\n");
        return -1;
    }
    printf("打开设备文件成功\n");
    //调用read
    read(fd, buf, sizeof(buf));
    //调用write
    write(fd, buf, sizeof(buf));
    //调用close
    close(fd);
    return 0;
}

6.现象

执行应用程序,驱动中的操作方法被回调

4.用户和内核的数据传递

用户空间和内核空间之间无法直接相互访问内存,二者进行数据交互需要使用数据传递函数

4.1 API

头文件 #include <linux/uaccess.h>

函数原型unsigned long copy_to_user(void __user *to, const void *from, unsigned long n)

功能:传递内核空间的数据到用户空间

参数:to:用户空间保存数据的buf首地址

           from:内核空间保存数据的buf首地址

           n:传递的数据长度,以字节为单位

返回值:成功返回0,失败返回未拷贝的字节数

函数原型unsigned log copy_from_user(void *to, const void __user *from, unsigned long n)

功能:传递用户空间的数据到内核空间

参数:to:内核空间保存数据的buf首地址

           from:用户空间保存数据的buf首地址

           n:传递的数据长度,以字节为单位

返回值:成功返回0,失败返回未拷贝的字节数

4.2用户和内核数据传递实例

应用程序代码

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include<string.h>
int main(int argc, char const *argv[])
{
    char buf[128] = {0};
    int fd = open("/dev/mychrdev", O_RDWR);
    if (fd < 0)
    {
        printf("打开设备文件失败\n");
        return -1;
    }
    printf("打开设备文件成功\n");
    fgets(buf,sizeof(buf),stdin);//在终端读一个字符串
    buf[strlen(buf)-1]='\0';
    
    write(fd, buf, sizeof(buf));//将数据传递给内核
    memset(buf,0,sizeof(buf));//清空数组
    read(fd, buf, sizeof(buf));//将内核空间数据传递到用户
    printf("buf:%s\n",buf);
    close(fd);
    return 0;
}

驱动程序代码 

#include <linux/init.h>
#include <linux/module.h>
#include <linux/fs.h>
#include<linux/uaccess.h>
unsigned int major;
char kbuf[128]={};
// 封装操作方法
int mycdev_open(struct inode *inode, struct file *file)
{
    printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);
    return 0;
}
ssize_t mycdev_read(struct file *file, char *ubuf, size_t size, loff_t *lof)
{
    printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);
    int ret;
    //拷贝数据到用户空间
    ret=copy_to_user(ubuf,kbuf,size);
    if(ret)
    {
        printk("copy_to_user filed\n");
        return -EIO;
    }
    return 0;
}
ssize_t mycdev_write(struct file *file, const char *ubuf, size_t size, loff_t *lof)
{
    printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);
    int ret;
    //从用户空间拷贝数据到内核空间
    ret=copy_from_user(kbuf,ubuf,size);
    if(ret)
    {
        printk("copy_from_user filed\n");
        return -EIO;
    }
    return 0;
}
int mycdev_close(struct inode *inode, struct file *file)
{
    printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);
    return 0;
}
// 定义操作方法结构体对象
struct file_operations fops = {
    .open=mycdev_open,
    .read=mycdev_read,
    .write=mycdev_write,
    .release=mycdev_close,
};
static int __init mycdev_init(void)
{
    // 注册字符设备驱动
    major = register_chrdev(0, "mychrdev", &fops);
    if (major < 0)
    {
        printk("字符设备驱动注册失败\n");
        return major;
    }
    printk("注册字符设备驱动成功major=%d\n", major);
    return 0;
}
static void __exit mycdev_exit(void)
{
    //注销字符设备驱动
    unregister_chrdev(major,"mychrdev");
}
module_init(mycdev_init);
module_exit(mycdev_exit);
MODULE_LICENSE("GPL");

现象:在终端输入的字符串可以正常的打印出来,说明用户空间和内核空间数据传递成功

5.物理内存映射相关API

驱动控制硬件需要操作硬件的特殊功能寄存器,但是特殊功能寄存器内存物属于理内存,驱动加载到虚拟内存,想要在驱动中操作硬件寄存器,需要将硬件寄存器内存映射为虚拟内存

#include <linux/io.h>

函数原型:void *ioremap(phys_addr_t paddr, unsigned long size)

功能:映射指定大小的物理内存为虚拟内存

参数:paddr:要映射的物理内存首地址

           size:要映射的物理内存大小

返回值:成功,返回映射成功的虚拟内存首地址,失败,返回NULL

函数原型:void iounmap(const void __iomem *addr)

功能:取消物理内存的映射

参数:addr:要取消的虚拟内存首地址

返回值:无

注:有关字符设备驱动的示例代码,目前可查看上一篇文章"编写驱动代码控制LED灯亮灭"http://t.csdnimg.cn/ZBsgG

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

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

相关文章

网络协议--ARP:地址解析协议

4.1 引言 本章我们要讨论的问题是只对TCP/IP协议簇有意义的IP地址。数据链路如以太网或令牌环网都有自己的寻址机制&#xff08;常常为48 bit地址&#xff09;&#xff0c;这是使用数据链路的任何网络层都必须遵从的。一个网络如以太网可以同时被不同的网络层使用。例如&#…

解剖—单链表相关OJ练习题

目录 一、移除链表元素 二、找出链表的中间节点 三、合并两个有序链表 四、反转链表 五、求链表中倒数第k个结点 六、链表分割 七、链表的回文结构 八、判断链表是否相交 九、判断链表中是否有环(一) 十、 判断链表中是否有环(二) 注&#xff1a;第六题和第七题牛…

三网话费余额查询的API系统 基于thinkphp6.0框架

本套系统是用thinkphp6.0框架开发的&#xff0c;PHP需大于8.2&#xff0c;系统支持用户中心在线查询和通过API接口对接发起查询&#xff0c;用户余额充值是对接usdt接口&#xff0c;源码全开源&#xff0c;支持懂技术的人二次开发~搭建教程1、源码上传后&#xff0c;吧运行目录…

异常数据检测 | Python基于Hampel的离群点检测

文章目录 文章概述模型描述源码分享文章概述 在时间序列数据分析领域,识别和处理异常点是至关重要的任务。异常点或离群点是明显偏离预期模式的数据点,可能表明存在错误、欺诈或有价值的见解。 应对这一挑战的一种有效技术是汉普尔过滤器(Hampel Filter)。 模型描述 汉…

Manacher学习笔记

Manacher 算法&#xff0c;俗称马拉车算法&#xff0c;是一种解决最长回文子串问题的算法。 在 Ybt 的哈希章节中出现了这个&#xff1a; 数据范围 1 0 6 10^6 106&#xff0c;数据相对于马拉车模板较弱。 对于一个普通的字符串&#xff0c;我们要想求出它的回文子串需要考虑它…

聊天机器人语料在开发中的重要性

语料在聊天机器人的开发中起着至关重要的作用&#xff0c;使其能够有效理解和回应用户的查询。语料是聊天机器人的训练数据&#xff0c;通过分析和学习这个语料&#xff0c;聊天机器人可以提高对用户意图的准确理解&#xff0c;并生成恰当的回应。 | 一、聊天机器人语料好在哪&…

Unity 单例-接口模式

单例-接口模式 使用接口方式实现的单例可以继承其它类&#xff0c;更加方便 using System.Collections; using System.Collections.Generic; using UniRx; using UniRx.Triggers; using UnityEngine; namespace ZYF {public interface ISingleton<TMono> where TMono : M…

Redis-Sentinel高可用架构学习

Redis-Sentinel高可用架构 Redis主从复制过程&#xff1a; 主从同步原理 Redis Sentinel&#xff08;哨兵&#xff09;高可用集群方案&#xff1a;Redis-Sentinel是Redis官方推荐的高可用性(HA)解决方案。 当用Redis做Master-slave的高可用方案时&#xff0c;假如master宕机了…

微信小程序之后台首页交互

目录 一.与后台数据进行交互&request封装 后台准备 测试结果 ​编辑 前端 测试结果 二.wxs的介绍以及入门 测试结果 一.与后台数据进行交互&request封装 后台准备 pom.xml文件编写 <?xml version"1.0" encoding"UTF-8"?> <proj…

【JavaEE】常见的锁策略 -- 多线程篇(4)

文章目录 乐观锁 vs 悲观锁读写锁重量级锁 vs 轻量级锁自旋锁&#xff08;Spin Lock&#xff09;公平锁 vs 非公平锁可重入锁 vs 不可重入锁 乐观锁 vs 悲观锁 悲观锁: 总是假设最坏的情况&#xff0c;每次去拿数据的时候都认为别人会修改&#xff0c;所以每次在拿数据的时候都…

全连接网络参数Xavier初始化

1.梯度消失 考虑下图的神经网络&#xff0c;在使用梯度下降法迭代更新W_ki和W_ij时&#xff0c;它们的梯度方向间有什么关系&#xff1f; 它们的梯度关系如下&#xff1a; 从上述两个式子我们大致可以看出&#xff0c;损失函数L关于第h层参数的梯度由两部分组成&#xff1a;…

sql server2014如何添加多个实例 | 以及如何删除多个实例中的单个实例

标题sql server2014如何添加多个实例 前提&#xff08;已安装sql server2014 且已有默认实例MSSQLSERVER&#xff09; 添加新的实例 其实就是根据安装步骤再安装一次&#xff08;区别在过程中说明&#xff09; 双击安装 选择“全新独立安装或添加现有功能” 然后下一步下一…

微信小程序开发之后台数据交互及wxs应用

目录 一、后端准备 1. 应用配置 2. 数据源配置 二、数据库 1. 创建 2. 数据表 3. 数据测试 三、前端 1. 请求方法整合 2. 数据请求 3. WXS的使用 4. 样式美化 5. 页面 一、后端准备 通过SpringMVC及mybatis的技术学习&#xff0c;还有前后端分离的技术应用&…

Linux程序地址

目录 一、定义 二、问题引出 三、虚拟地址和物理地址 &#xff08;一&#xff09;问题解释 &#xff08;二&#xff09;什么是进程地址空间 &#xff08;三&#xff09;为什么要有进程地址空间 一、定义 #include <stdio.h> #include <stdlib.h>//geten…

运维监控Zabbix部署

目录 运维监控Zabbix部署 1. 简介 2. 安装 ​编辑 2.1 安装前准备 - Mysql 2.2 安装Zabbix Server 和 Zabbix Agent 2.2.1 安装Zabbix yum库 2.2.2 安装Zabbix Server、前端、Agent 2.2.3 初始化Mysql数据库 2.2.4 为Zabbix Server配置数据库 2.2.5 配置Zab…

【目标检测】Co-DETR:ATSS+Faster RCNN+DETR协作的先进检测器(ICCV 2023)

论文&#xff1a;DETRs with Collaborative Hybrid Assignments Training 代码**&#xff1a;https://github.com/Sense-X/Co-DETR 文章目录 摘要一、简介二、本文方法2.1.概述2.2.协同混合分配训练2.3. 定制的正 Query 生成2.4. Co-DETR为何有效1、丰富编码器的监督2、通过减少…

QEMU DirtyLimit特性介绍

文章目录 背景基本原理PMLDirty-RingDirty-Limit 具体实现数据结构vcpu_dirty_rate_statdirtylimit_state 算法实现接口逻辑qmp_set_vcpu_dirty_limitqmp_cancel_vcpu_dirty_limit 限制算法算法框架理想效果具体实现 测试验证QEMULibvirt 一个广子 背景 热迁移实现逻辑中&…

---图的遍历和最小生成树

广度优先遍历 --- 针对的是顶点遍历 深度优先遍历 如果给的图不是连通图&#xff1f;以某个点为起点就没有遍历完成。那么怎么保证遍历完剩下的点呢&#xff1f;&#xff1f; 在标记数组当中找没有遍历过的点&#xff0c;在进行遍历 最小生成树 生成树&#xff1a;一个连通…

使用TypeScript和jsdom库实现自动化数据抓取

目录 环境准备 使用TypeScript和jsdom抓取数据 总结 随着网络技术的发展&#xff0c;数据抓取已成为获取信息的重要手段。然而&#xff0c;手动进行数据抓取既耗时又容易出错。因此&#xff0c;本文将介绍如何使用TypeScript和jsdom库实现自动化数据抓取。我们将通过创建一个…

iMazing苹果用户手机备份工具 兼容最新的iOS16操作系统

现在距离苹果秋季新品发布会已过去月余&#xff0c;新iPhone 14系列和新版的iOS 16操作系统也如约与我们见面了&#xff0c;相信大家在9月初抢购的iPhone 14也基本到手了&#xff0c;但随之到来的数据资料备份迁移却是一件令人头大的事情&#xff0c;使用官方提供的iTunes软件卡…