通过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接口也可以实现文件接口来修改内核模块中的变量,详细见之前的文章。

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

苦梨甜

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值