编写Linux内核模块实现文件拷贝

2024-06-17 04:48

本文主要是介绍编写Linux内核模块实现文件拷贝,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

经评论区反馈:如果Linux内核版本为4.XX,那么需要在 vfs_read 和 vfs_write 的地方更改为 kernel_read 和 kernel_write

操作系统课程实验1 添加内核模块

每个人题目都是自己选择的,题目1已经有一位校友给出了标准答案,我捣鼓的是文件拷贝这题,弄了很久也记录一下,给后面的学弟学妹参考

题目4:设计一个带参数的模块,其参数为源文件和目标文件的文件名(可能带路径),模块功能是实现文件拷贝功能。

踩的大坑:

一、 最初用 open()、read()、write() 函数,发现头文件路径很奇怪,怎么包含也弄不对

这是因为在VFS的支持下,用户态进程读写任何类型的文件系统都可以使用read和write这两个系统调用,但是在linux内核中没有这样的系统调用。也就是说,这几个函数是给用户在用户态下使用的,内核中没有这样的函数。所以继续寻找,找到了这一组函数:

功能函数原型
打开文件struct file *filp_open(const char *, int, umode_t)
读文件ssize_t vfs_read(struct file *, char __user *, size_t, loff_t *)
写文件ssize_t vfs_write(struct file *, const char __user *, size_t, loff_t *)
关闭文件int filp_close(struct file *, fl_owner_t id)
四个函数的详解大家就自行百度一下,(或者我后面有时间的话再补进来qaq
二、关于 buffer 这个缓冲区

如果直接把 buffer 作为参数执行 vfs_read 和 vfs_write 这两个函数的话,会有错误,返回值应该是-14。这是因为 vfs_read 和 vfs_write 函数中,其第二个参数指向的是用户空间的内存地址,如果我们直接使用内核空间的指针,则会返回 -EFALUT。所以我们需要使用 set_fs() 和 get_fs() 宏来改变内核对内存地址检查的处理方式。这两个宏包含在 uaccess.h 头文件中。

简单补充知识点(详细的还要大家自行去学习):

一、module_param 中的 perm 参数(Linux权限问题)

书上示例程序有一句

static char *who;
module_param(who,char,0644);

(先纠正一下错误,这样写的话是过不了编译的,应该要改成:(纠正错了欢迎指出))

static char *who;
module_param(who,charp,0644);

这里的charp应该是 char point 的意思,那么 0644 是哪里来的?
大家看这张表应该就能明白了:

宏定义含义
S_IRWXU00700权限,代表该文件所有者拥有读,写和执行操作的权限
S_IRUSR00400权限,代表该文件所有者拥有可读的权限
S_IWUSR00200权限,代表该文件所有者拥有可写的权限
S_IXUSR00100权限,代表该文件所有者拥有执行的权限
S_IRWXG00070权限,代表该文件用户组拥有读,写和执行操作的权限
S_IRGRP00040权限,代表该文件用户组拥有可读的权限
S_IWGRP00020权限,代表该文件用户组拥有可写的权限
S_IXGRP00010权限,代表该文件用户组拥有执行的权限
S_IRWXO00007权限,代表其他用户拥有读,写和执行操作的权限
S_IROTH00004权限,代表其他用户拥有可读的权限
S_IWOTH00002权限,代表其他用户拥有可写的权限
S_IXOTH00001权限,代表其他用户拥有执行的权限
注意这些是数字0开头,表明这些其实是八进制数。这些权限可以组合,因此可以得到:
标志含义
:–
-rw------- (600)只有用户有读写权限。
-rw-r–r-- (644)只有用户有读写权限;而组用户和其他用户只有读权限。
-rwx------ (700)只有用户有读、写、执行权限。
-rwxr-xr-x (755)用户有读、写、执行权限;而组用户和其他用户只有读、执行权限。
-rwx–x–x (711)用户有读、写、执行权限;而组用户和其他用户只有执行权限。
-rw-rw-rw- (666)所有用户都有文件读、写权限。这种做法不可取。
-rwxrwxrwx (777)所有用户都有读、写、执行权限。更不可取的做法。

代码

module1.c

#include <linux/init.h>
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/fs.h>
#include <asm/uaccess.h>#define BUFF_SIZE 512static char *source;
static char *target;module_param(source, charp, 0644);
module_param(target, charp, 0644);static int __init module1_init(void)
{int cnt;char buffer[BUFF_SIZE];struct file *source_filp;struct file *target_filp;loff_t source_pos;loff_t target_pos;mm_segment_t fs;printk(KERN_ALERT "hello module1\n");//open source fileif (IS_ERR(source_filp = filp_open(source, O_RDONLY, 0))){printk(KERN_ALERT "Open %s Error\n", source);return -1;}//create and open target fileif (IS_ERR(target_filp = filp_open(target, O_CREAT | O_WRONLY | O_TRUNC, S_IRUSR | S_IWUSR))){printk(KERN_ALERT "Open %s Error\n", target);return -1;}//copysource_pos = source_filp->f_pos;target_pos = target_filp->f_pos;fs = get_fs();set_fs(KERNEL_DS);while ((cnt = vfs_read(source_filp, buffer, BUFF_SIZE, &source_pos)) > 0){vfs_write(target_filp, buffer, cnt, &target_pos);}set_fs(fs);filp_close(source_filp, NULL);filp_close(target_filp, NULL);return 0;
}static void module1_exit(void)
{printk(KERN_ALERT "goodbye module1\n");
}module_init(module1_init);
module_exit(module1_exit);
MODULE_LICENSE("GPL");

Makefile

obj-m:=module1.o
KDIR:=/lib/modules/$(shell uname -r)/build
PWD:=$(shell pwd)default:make -C $(KDIR) M=$(PWD) modules clean:make -C $(KDIR) M=$(PWD) clean

插入模块命令

sudo insmod module1.ko source="./module1.c" target="./module2.c"

这里是在当前目录下新建一个module2.c文件,然后把module1.c的内容拷贝到module2.c

这篇关于编写Linux内核模块实现文件拷贝的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

linux生产者,消费者问题

pthread_cond_wait() :用于阻塞当前线程,等待别的线程使用pthread_cond_signal()或pthread_cond_broadcast来唤醒它。 pthread_cond_wait() 必须与pthread_mutex 配套使用。pthread_cond_wait()函数一进入wait状态就会自动release mutex。当其他线程通过pthread

C++对象布局及多态实现探索之内存布局(整理的很多链接)

本文通过观察对象的内存布局,跟踪函数调用的汇编代码。分析了C++对象内存的布局情况,虚函数的执行方式,以及虚继承,等等 文章链接:http://dev.yesky.com/254/2191254.shtml      论C/C++函数间动态内存的传递 (2005-07-30)   当你涉及到C/C++的核心编程的时候,你会无止境地与内存管理打交道。 文章链接:http://dev.yesky

Linux 安装、配置Tomcat 的HTTPS

Linux 安装 、配置Tomcat的HTTPS 安装Tomcat 这里选择的是 tomcat 10.X ,需要Java 11及更高版本 Binary Distributions ->Core->选择 tar.gz包 下载、上传到内网服务器 /opt 目录tar -xzf 解压将解压的根目录改名为 tomat-10 并移动到 /opt 下, 形成个人习惯的路径 /opt/tomcat-10

RedHat运维-Linux文本操作基础-AWK进阶

你不用整理,跟着敲一遍,有个印象,然后把它保存到本地,以后要用再去看,如果有了新东西,你自个再添加。这是我参考牛客上的shell编程专项题,只不过换成了问答的方式而已。不用背,就算是我自己亲自敲,我现在好多也记不住。 1. 输出nowcoder.txt文件第5行的内容 2. 输出nowcoder.txt文件第6行的内容 3. 输出nowcoder.txt文件第7行的内容 4. 输出nowcode

【Linux进阶】UNIX体系结构分解——操作系统,内核,shell

1.什么是操作系统? 从严格意义上说,可将操作系统定义为一种软件,它控制计算机硬件资源,提供程序运行环境。我们通常将这种软件称为内核(kerel),因为它相对较小,而且位于环境的核心。  从广义上说,操作系统包括了内核和一些其他软件,这些软件使得计算机能够发挥作用,并使计算机具有自己的特生。这里所说的其他软件包括系统实用程序(system utility)、应用程序、shell以及公用函数库等

通过SSH隧道实现通过远程服务器上外网

搭建隧道 autossh -M 0 -f -D 1080 -C -N user1@remotehost##验证隧道是否生效,查看1080端口是否启动netstat -tuln | grep 1080## 测试ssh 隧道是否生效curl -x socks5h://127.0.0.1:1080 -I http://www.github.com 将autossh 设置为服务,隧道开机启动

Windows/macOS/Linux 安装 Redis 和 Redis Desktop Manager 可视化工具

本文所有安装都在macOS High Sierra 10.13.4进行,Windows安装相对容易些,Linux安装与macOS类似,文中会做区分讲解 1. Redis安装 1.下载Redis https://redis.io/download 把下载的源码更名为redis-4.0.9-source,我喜欢跟maven、Tomcat放在一起,就放到/Users/zhan/Documents

时序预测 | MATLAB实现LSTM时间序列未来多步预测-递归预测

时序预测 | MATLAB实现LSTM时间序列未来多步预测-递归预测 目录 时序预测 | MATLAB实现LSTM时间序列未来多步预测-递归预测基本介绍程序设计参考资料 基本介绍 MATLAB实现LSTM时间序列未来多步预测-递归预测。LSTM是一种含有LSTM区块(blocks)或其他的一种类神经网络,文献或其他资料中LSTM区块可能被描述成智能网络单元,因为

vue项目集成CanvasEditor实现Word在线编辑器

CanvasEditor实现Word在线编辑器 官网文档:https://hufe.club/canvas-editor-docs/guide/schema.html 源码地址:https://github.com/Hufe921/canvas-editor 前提声明: 由于CanvasEditor目前不支持vue、react 等框架开箱即用版,所以需要我们去Git下载源码,拿到其中两个主

android一键分享功能部分实现

为什么叫做部分实现呢,其实是我只实现一部分的分享。如新浪微博,那还有没去实现的是微信分享。还有一部分奇怪的问题:我QQ分享跟QQ空间的分享功能,我都没配置key那些都是原本集成就有的key也可以实现分享,谁清楚的麻烦详解下。 实现分享功能我们可以去www.mob.com这个网站集成。免费的,而且还有短信验证功能。等这分享研究完后就研究下短信验证功能。 开始实现步骤(新浪分享,以下是本人自己实现