我应该使用内核模块来更改内核的配置参数。内核模块应该创建一个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实际上给出了从命令行给模块的值?
答案 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_write
或xxx_read
视为接口工具,
当您在用户空间中呼叫write
或read
时,
内核将在内核空间中调用xxx_write
或xxx_read
。
所以你需要在write
来电时自己存储,
并在read
来电时将其取回
在xxx_write
xxx_read