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

标签 c linux linux-kernel

我应该使用内核模块更改内核的配置参数。内核模块应该创建一个 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/

相关文章:

linux - 如何从文件中提取单个字节 block ?

linux - 为什么当我尝试远程执行命令时出现错误?

c - 内核空间和用户空间之间的大共享内存

linux - 是否可以用 gcc 以外的东西编译 Linux 内核

c - 如何将整数算术变成 long long?

c++ - memset() 导致数据中止

c - 从 csv 读取,strtod 无法读取带有数字的字符串

c++ - 没有溢出的无限递归 - 这可能吗?

python - 路径内文件夹内文件内的深度小写

linux - USER_HZ 如何解决 jiffy 缩放问题?