4

是否可以在其中一个 sys 文件发生更改时通知模块?我的任务是做一个控制模块内缓冲区大小的文件,我想在文件中的值更改时调整缓冲区的大小。我的另一个想法(如果我不能通知模块)是每次使用模块时检查以前的值,然后调整缓冲区的大小。

4

1 回答 1

9

这不是 Sysfs 的目的吗?

当您创建 akobject并在 Sysfs(它是一个目录)中为其提供表示时,您然后为该对象创建属性,这些属性将成为该目录中的文件。您为 提供一个store和一个show回调kobject,它们基本上等同于 resp。writeread

store是你想要的。它看起来像这样:

ssize_t (*store)(struct kobject *kobj, struct attribute *attr, 
    const char *buffer, size_t size);

一旦虚拟文件写入用户空间,您就会收到size字节。buffer

看看这个模块(取自这里):

#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/init.h>
#include <linux/fs.h>
#include <linux/slab.h>

struct my_attr {
    struct attribute attr;
    int value;
};

static struct my_attr my_first = {
    .attr.name="first",
    .attr.mode = 0644,
    .value = 1,
};

static struct my_attr my_second = {
    .attr.name="second",
    .attr.mode = 0644,
    .value = 2,
};

static struct attribute * myattr[] = {
    &my_first.attr,
    &my_second.attr,
    NULL
};

static ssize_t default_show(struct kobject *kobj, struct attribute *attr,
        char *buf)
{
    struct my_attr *a = container_of(attr, struct my_attr, attr);
    return scnprintf(buf, PAGE_SIZE, "%d\n", a->value);
}

static ssize_t default_store(struct kobject *kobj, struct attribute *attr,
        const char *buf, size_t len)
{
    struct my_attr *a = container_of(attr, struct my_attr, attr);
    sscanf(buf, "%d", &a->value);
    return sizeof(int);
}

static struct sysfs_ops myops = {
    .show = default_show,
    .store = default_store,
};

static struct kobj_type mytype = {
    .sysfs_ops = &myops,
    .default_attrs = myattr,
};

struct kobject *mykobj;
static int __init sysfsexample_module_init(void)
{
    int err = -1;
    mykobj = kzalloc(sizeof(*mykobj), GFP_KERNEL);
    if (mykobj) {
        kobject_init(mykobj, &mytype);
        if (kobject_add(mykobj, NULL, "%s", "sysfs_sample")) {
             err = -1;
             printk("Sysfs creation failed\n");
             kobject_put(mykobj);
             mykobj = NULL;
        }
        err = 0;
    }
    return err;
}

static void __exit sysfsexample_module_exit(void)
{
    if (mykobj) {
        kobject_put(mykobj);
        kfree(mykobj);
    }
}

module_init(sysfsexample_module_init);
module_exit(sysfsexample_module_exit);
MODULE_LICENSE("GPL");

另外:您可能希望在读取条目时向用户输出缓冲区大小。这通常是这样做的方式。还要确保信息(读取和写入)是人类可读的格式,以跟上 Unix 哲学。

更新:请参阅最近由顶级内核开发人员之一 Greg Kroah-Hartman 撰写的关于 Sysfs 文件创建的有趣文章。

于 2012-06-16T16:12:29.253 回答