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

时间:2015-05-15 12:36:47

标签: c linux linux-kernel

我应该使用内核模块来更改内核的配置参数。内核模块应该创建一个proc文件然后我应该能够使用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来读取参数,那么read函数如何实际获取参数的新值,即procfile中存储的“foobar”在哪里?

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

2 个答案:

答案 0 :(得分:3)

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。目前只提供旧版本,但正在编写更新版本。

答案 1 :(得分:1)

您可以将驱动程序中的xxx_writexxx_read视为接口工具,

当您在用户空间中呼叫writeread时, 内核将在内核空间中调用xxx_writexxx_read

所以你需要在write来电时自己存储,
并在read来电时将其取回   在xxx_write xxx_read

enter image description here