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

本文主要是介绍基于Linux的驱动开发:内核模块传参、内核到处符号表、字符设备驱动,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

内核模块传参

        内核模块:

                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");//调用readread(fd, buf, sizeof(buf));//调用writewrite(fd, buf, sizeof(buf));//调用closeclose(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

这篇关于基于Linux的驱动开发:内核模块传参、内核到处符号表、字符设备驱动的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/252733

相关文章

Linux脚本(shell)的使用方式

《Linux脚本(shell)的使用方式》:本文主要介绍Linux脚本(shell)的使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录概述语法详解数学运算表达式Shell变量变量分类环境变量Shell内部变量自定义变量:定义、赋值自定义变量:引用、修改、删

Python实例题之pygame开发打飞机游戏实例代码

《Python实例题之pygame开发打飞机游戏实例代码》对于python的学习者,能够写出一个飞机大战的程序代码,是不是感觉到非常的开心,:本文主要介绍Python实例题之pygame开发打飞机... 目录题目pygame-aircraft-game使用 Pygame 开发的打飞机游戏脚本代码解释初始化部

使用Python开发一个现代化屏幕取色器

《使用Python开发一个现代化屏幕取色器》在UI设计、网页开发等场景中,颜色拾取是高频需求,:本文主要介绍如何使用Python开发一个现代化屏幕取色器,有需要的小伙伴可以参考一下... 目录一、项目概述二、核心功能解析2.1 实时颜色追踪2.2 智能颜色显示三、效果展示四、实现步骤详解4.1 环境配置4.

Linux链表操作方式

《Linux链表操作方式》:本文主要介绍Linux链表操作方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、链表基础概念与内核链表优势二、内核链表结构与宏解析三、内核链表的优点四、用户态链表示例五、双向循环链表在内核中的实现优势六、典型应用场景七、调试技巧与

Python使用smtplib库开发一个邮件自动发送工具

《Python使用smtplib库开发一个邮件自动发送工具》在现代软件开发中,自动化邮件发送是一个非常实用的功能,无论是系统通知、营销邮件、还是日常工作报告,Python的smtplib库都能帮助我们... 目录代码实现与知识点解析1. 导入必要的库2. 配置邮件服务器参数3. 创建邮件发送类4. 实现邮件

详解Linux中常见环境变量的特点与设置

《详解Linux中常见环境变量的特点与设置》环境变量是操作系统和用户设置的一些动态键值对,为运行的程序提供配置信息,理解环境变量对于系统管理、软件开发都很重要,下面小编就为大家详细介绍一下吧... 目录前言一、环境变量的概念二、常见的环境变量三、环境变量特点及其相关指令3.1 环境变量的全局性3.2、环境变

C#如何去掉文件夹或文件名非法字符

《C#如何去掉文件夹或文件名非法字符》:本文主要介绍C#如何去掉文件夹或文件名非法字符的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C#去掉文件夹或文件名非法字符net类库提供了非法字符的数组这里还有个小窍门总结C#去掉文件夹或文件名非法字符实现有输入字

Linux系统中的firewall-offline-cmd详解(收藏版)

《Linux系统中的firewall-offline-cmd详解(收藏版)》firewall-offline-cmd是firewalld的一个命令行工具,专门设计用于在没有运行firewalld服务的... 目录主要用途基本语法选项1. 状态管理2. 区域管理3. 服务管理4. 端口管理5. ICMP 阻断

Linux实现线程同步的多种方式汇总

《Linux实现线程同步的多种方式汇总》本文详细介绍了Linux下线程同步的多种方法,包括互斥锁、自旋锁、信号量以及它们的使用示例,通过这些同步机制,可以解决线程安全问题,防止资源竞争导致的错误,示例... 目录什么是线程同步?一、互斥锁(单人洗手间规则)适用场景:特点:二、条件变量(咖啡厅取餐系统)工作流

Linux中修改Apache HTTP Server(httpd)默认端口的完整指南

《Linux中修改ApacheHTTPServer(httpd)默认端口的完整指南》ApacheHTTPServer(简称httpd)是Linux系统中最常用的Web服务器之一,本文将详细介绍如何... 目录一、修改 httpd 默认端口的步骤1. 查找 httpd 配置文件路径2. 编辑配置文件3. 保存