通过sysfs文件系统接口来改变内核模块中的变量值(二)

简介: 通过sysfs文件系统接口来改变内核模块中的变量值(二)

给出一个完整的源码来展示如何创建、初始化并向系统中添加一个对象,以及如何通过sysfs文件系统接口在用户空间和内核空间进行沟通,另一个有趣的事情是它通过/sbin/hotplug机制来通知用户空间某一个kobject状态的变化。在这个例子中,我们将用自己编译的一个应用程序取代系统的/sbin/hotplug,该应用程序会打出一些环境变量,记在/var/log/messages文件中。

kobject.c

kobject.c中中实现sysfs文件接口下文件的创建,及其建立变量与文件之间的联系

#include <linux/module.h>
#include <linux/init.h>
#include <linux/kobject.h>
#include <linux/sysfs.h>
#include <linux/slab.h>
static struct kobject *parent;
static struct kobject *child;
static struct kset *c_kset;
static unsigned long flag;
ssize_t att_show(struct kobject *kobj, struct attribute *attr, char *buf){
    size_t count = 0;
    count += sprintf(&buf[count],"%lu\n",flag);
    return count;
}
ssize_t att_store(struct kobject *kobj, struct attribute *attr, const char *buf, size_t count){
    flag = buf[0] - '0';
    switch(flag){
        case 0:
            kobject_uevent(kobj,KOBJ_ADD);
            break;
        case 1:
            kobject_uevent(kobj,KOBJ_REMOVE);
            break;
        case 2:
            kobject_uevent(kobj,KOBJ_CHANGE);
            break;
        case 3:
            kobject_uevent(kobj,KOBJ_MOVE);
            break;
        case 4:
            kobject_uevent(kobj,KOBJ_ONLINE);
            break;
        case 5:
            kobject_uevent(kobj,KOBJ_OFFLINE);
            break;
    }
    return count;
}
static struct attribute cld_att = {
    .name = "cldatt",
    .mode = S_IRUGO | S_IWUSR,
};
static const struct sysfs_ops att_ops = {
    .show = att_show,
    .store = att_store,
};
static struct kobj_type cld_ktype = {
    .sysfs_ops = &att_ops,
};
static int kobj_demo_init(void){
    int err;
    parent = kobject_create_and_add("pa_obj", NULL);
    child = kzalloc(sizeof(* child),GFP_KERNEL);
    if(!child){
        return PTR_ERR(child);
    }
    //一个能够通知用户空间状态变化的kobject必須隶屬于某一个kset,也就是所谓的
    //subsystem,所以此处给内核对象chlld创建一个kset对象c_kset
    c_kset = kset_create_and_add("c_kset", NULL, parent);
    if(!c_kset){
        return -1;
    }
    child-> kset = c_kset;
    err = kobject_init_and_add(child,&cld_ktype,parent,"cld_obj");
    if(err)
        return err;
    err = sysfs_create_file(child,&cld_att);
    return err;
}
static void kobj_demo_exit(void) {
    sysfs_remove_file(child,&cld_att);
    kset_unregister(c_kset);
    kobject_del(child);
    kobject_del(parent);
}
module_init(kobj_demo_init);
module_exit(kobj_demo_exit);
MODULE_LICENSE("GPL");

app文件原本用于测试取代系统的/sbin/hotplug,该应用程序会打出一些环境变量,记在/var/log/messages文件中。但只在使用时,这个功能并没有实现。

app

#include <stdio.h>
#include <syslog.h>
extern char **environ;
int main(int argc,char *argv[]){
    char **var;
    syslog(LOG_INFO | LOG_LOCAL0,"---------------------------\n");
    syslog(LOG_INFO | LOG_LOCAL0,"argv[1]=%s\n",argv[1]);
    for(var = environ; *var != NULL;++var){
        syslog(LOG_INFO | LOG_LOCAL0,"argv[1]=%s\n",argv[1]);
    }
    syslog(LOG_INFO | LOG_LOCAL0,"---------------------------\n");
    return 0;
}

Makefile

  • make 编译项目
  • make file 在存放.ko文件目录中创建对应项目的目录
  • make install 将*.ko及其应用测试文件移动到根文件中
    所有路径需要按照自己的路劲来修改
#make 编译项目
#make file 在存放.ko文件目录中创建对应项目的目录
#make install 将*.ko及其应用测试文件移动到根文件中
 # 1. 使用不同的开发板内核时, 一定要修改KERN_DIR
# 2. KERN_DIR中的内核要事先配置、编译, 为了能编译内核, 要先设置下列环境变量:
# 2.1 ARCH,          比如: export ARCH=arm64
# 2.2 CROSS_COMPILE, 比如: export CROSS_COMPILE=aarch64-linux-gnu-
# 2.3 PATH,          比如: export PATH=$PATH:/home/book/100ask_roc-rk3399-pc/ToolChain-6.3.1/gcc-linaro-6.3.1-2017.05-x86_64_aarch64-linux-gnu/bin 
# 注意: 不同的开发板不同的编译器上述3个环境变量不一定相同,
#       请参考各开发板的高级用户使用手册
# ROOTFS_DIR 根文件系统中存放 *.ko文件所在目录
# PROJECT_NAME 在存放.ko文件目录中创建对应项目的目录
# DRIVER_NAME 项目中需要编译出.ko来的驱动
# APP_NAME 项目中的应用测试文件
#make 编译项目
#make file 在存放.ko文件目录中创建对应项目的目录
#make install 将*.ko及其应用测试文件移动到根文件中
KERN_DIR = /home/alientek/linux/IMX6ULL/linux/temp/linux-imx-rel_imx_4.1.15_2.1.0_ga_alientek
ROOTFS_DIR = /home/alientek/linux/nfs/rootfs/experiment
#项目名字
PROJECT_NAME = kobject
#各驱动名字,ko
DRIVER_NAME1 = kobject
DRIVER_NAME2 = 
#app名字
APP_NAME = app
all:
    make -C $(KERN_DIR) M=`pwd` modules 
    $(CROSS_COMPILE)arm-linux-gnueabihf-gcc -o $(APP_NAME) $(APP_NAME).c 
clean:
    make -C $(KERN_DIR) M=`pwd` modules clean
    rm -rf modules.order
    rm -f $(APP_NAME)
file:
    mkdir $(ROOTFS_DIR)/$(PROJECT_NAME)
install:
    cp *.ko $(ROOTFS_DIR)/$(PROJECT_NAME)
    cp $(APP_NAME) $(ROOTFS_DIR)/$(PROJECT_NAME)
# 参考内核源码drivers/char/ipmi/Makefile
# 要想把a.c, b.c编译成ab.ko, 可以这样指定:
# ab-y := a.o b.o
# obj-m += ab.o
obj-m += $(DRIVER_NAME1).o

结果

将编译好的内核模块kodemo.ko通过insmod加入系统后,除了在/sys目录下生成parent与child内核对象所对应的入口点pa_obj

还会在/sys/pa_obj/cld_obj目录下生成child内核对象的一个属性文件cldatt:

通过sysfs文件接口来改变内核模块中的变量flag

通过fasync接口也可以实现文件接口来修改内核模块中的变量,详细见之前的文章。

目录
相关文章
|
3天前
使用udev 设置磁盘属性
使用udev 设置磁盘属性
27 0
|
8月前
|
Linux C语言
Linux驱动 | debugfs接口创建
Linux驱动 | debugfs接口创建
|
10月前
|
Linux
通过sysfs文件系统接口来改变内核模块中的变量值(一)--通过fasync实现
通过sysfs文件系统接口来改变内核模块中的变量值(一)--通过fasync实现
67 0
|
Linux Shell
内核模块中对文件的读写
平时网络部分的东西碰的多些,这块一开始还真不知道怎么写,因为肯定和在用户空间下是不同的。google过后,得到以下答案。一般可以用两种方法:第一种是用系统调用。第二种方法是filp->open()等函数。下面分别来说下这两种方法。 1 利用系统调用: sys_open,sys_write,sys_read等。 其实分析过sys_open可以知道,最后调用的也是filp->open。 sys_open ==> do_sys_open ==> filp->open 在linuxsir上的一个帖子,上面一个版主说:sys_open和进程紧密相关,往往不在内核中使用。 而其实sys_open最后也
372 0
|
Linux Shell
为Tiny4412设备驱动在proc目录下添加一个可读版本信息的文件
http://blog.csdn.net/morixinguan/article/details/77808088 上节,我们明白了proc文件系统的作用,接下来我们在友善之臂已经写好的led驱动的基础上,在proc目录下创建一个文件夹,然后加入led驱动的版本信息读取。
1362 0
|
Linux
linux内核initrd文件自定义方法
<span style="font-family: 'microsoft yahei';"> </span><h1 style="margin: 0px; padding: 0px; display: inline-block; vertical-align: middle; font-size: 18px; font-family: 'microsoft yahei';"><span cla
4294 0