/proc/prompt 应该将参数设置为 "foo-6ren">
gpt4 book ai didi

c - 通过调用内核模块写入proc文件/给参数

转载 作者:塔克拉玛干 更新时间:2023-11-02 23:36:21 30 4
gpt4 key购买 nike

我应该使用内核模块更改内核的配置参数。内核模块应该创建一个 proc 文件,然后我应该能够使用 cat 命令更改参数,例如cat "foobar">/proc/prompt 应该将参数设置为 "foobar",其中 prompt 是在模块中创建的 proc 文件的名称。

此外,我应该能够在调用模块时通过将参数作为参数传递来初始化参数。

这两篇文章基本上是我找到的唯一相关资源:

http://www.tldp.org/LDP/lkmpg/2.6/html/x769.html用于写入 proc 文件和 http://www.tldp.org/LDP/lkmpg/2.6/html/x323.html用于从命令行初始化参数。

现在我有几个问题,首先这是到目前为止的模块:

#include <linux/kernel.h>
#include <linux/version.h>
#include <linux/list.h>
#include <linux/module.h>
#include <linux/proc_fs.h>

#include "sar_main.h"

#define PROCFS_NAME "sarlkm"

char procfs_buffer[PROCFS_MAX_SIZE];

static unsigned long procfs_buffer_size = 0

struct proc_dir_entry *proc_file_entry;


int procfile_read(char *buffer, char **buffer_location, off_t offset, int buffer_length, int *eof, void *data){
int ret;

printk(KERN_INFO "procfile_read (/proc/%s) aufgerufen \n", PROCFS_NAME);

if (offset > 0){
ret = 0;
}
else{
memcpy(buffer, procfs_buffer, procfs_buffer_size);
ret = procfs_buffer_size;
}
return ret;
}

int procfile_write(struct file *file, const char *buffer, unsigned long count, void *data){

procfs_buffer_size = count;
if (procfs_buffer_size > PROCFS_MAX_SIZE){
procfs_buffer_size = PROCFS_MAX_SIZE;
}
if ( copy_from_user(procfs_buffer, buffer, procfs_buffer)){
return -EFAULT;
}
return procfs_buffer_size;
}




static int __init sar_init(void)
{
prompt_proc = create_proc_entry(PROCFS_NAME, 0644, NULL);

if (prompt_proc = NULL){
remove_proc_entry(PROCFS_NAME, &proc_root);
printk(KERN_ALERT "Error: Konnte proc file nicht kreieren")
return -ENOMEM;
}

prompt_proc->read_proc = procfile_read;
prompt_proc->write_proc = procfile_write;

printk(KERN_INFO "proc/%s wurde erfolgreich kreiert", PROCFS_NAME);
return 0;
}

static void __exit sar_cleanup(void)
{
remove_proc_entry(PROCFS_NAME, &proc_root);
printk(KERN_INFO "proc/%s gelöscht", PROCFS_NAME);
}

module_init(sar_init);
module_exit(sar_cleanup);

MODULE_AUTHOR(DRIVER_AUTHOR);
MODULE_DESCRIPTION(DRIVER_DESC);
MODULE_LICENSE("GPL");

我想我应该说我真的不明白为什么在使用 cat 命令时读取和写入功能应该起作用。我的主要问题是配置参数究竟存储在 proc 文件中的什么位置?如果我使用 cat 将“foobar”写入 proc 文件,然后使用 cat proc/prompt 读取参数,那么读取函数如何实际获取参数的新值,即“foobar”存储在 procfile 中的何处?

如果我尝试使用命令行参数初始化参数,我将不得不使用一个全局变量来存储参数的值,但是我如何在读取函数中使用该全局变量,以便cat proc/prompt 实际上给出了从命令行提供给模块的值?

最佳答案

cat 命令在内部调用 read() 系统调用以从文件中读取数据(参见 man strace)。

read() 将参数传递给 VFS VFS 最终使用传递的参数(以及 VFS 代码传递的一些附加参数)调用您的自定义 procfile_read() 例程。如果您想了解更多信息,请查看内核源代码中的 fs 目录,尤其是文件 read_write.c

如果满足某些条件,您的特定读取函数会将参数值(存储在 procfs_buffer 中以回答您的一个问题)复制到由 cat 分配的用户提供的缓冲区中,在您的特定代码中称为 buffer。它与 read() 系统调用传递的相同:

read(proc_fd, userspace_buf, 10);    /* userspace_buf is buffer! */

为清楚起见,省略了错误检查。

要将值传递给 proc 文件,您有两个选择:

  • 使用module_param() 并将其写入您的缓冲区;只能一次,因为该模块只能加载一次(或者每次您想更改参数时卸载/重新加载它,但这听起来很不方便)
  • 从用户空间调用 write()(如在 cat 中)并根据需要随时修改缓冲区(这是当前使用的通过你的代码)

顺便说一句,我真的认为您的读取函数应该检查指向用户数据的指针,即使用 copy_to_user(),而不是 memcpy()

有关更多信息,请阅读 Linux Device Drivers .目前只有旧版本可用,但正在编写更新版本。

关于c - 通过调用内核模块写入proc文件/给参数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30259717/

30 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com