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

目录
相关文章
|
21天前
使用udev 设置磁盘属性
使用udev 设置磁盘属性
31 0
|
9月前
|
Linux C语言
Linux驱动 | debugfs接口创建
Linux驱动 | debugfs接口创建
|
11月前
|
Linux
通过sysfs文件系统接口来改变内核模块中的变量值(一)--通过fasync实现
通过sysfs文件系统接口来改变内核模块中的变量值(一)--通过fasync实现
70 0
|
11月前
|
Linux
内核是如何运行ko文件的--系统调用
内核是如何运行ko文件的--系统调用
206 0
|
存储
驱动开发:内核枚举LoadImage映像回调
在笔者之前的文章`《驱动开发:内核特征码搜索函数封装》`中我们封装实现了特征码定位功能,本章将继续使用该功能,本次我们需要枚举内核`LoadImage`映像回调,在Win64环境下我们可以设置一个`LoadImage`映像加载通告回调,当有新驱动或者DLL被加载时,回调函数就会被调用从而执行我们自己的回调例程,映像回调也存储在数组里,枚举时从数组中读取值之后,需要进行位运算解密得到地址。
276 1
驱动开发:内核枚举LoadImage映像回调
|
Linux 内存技术
uboot环境变量(设置bootargs向linux内核传递正确的参数)
这是我uboot的环境变量设置,在该设置下可以运行initram内核(从内存下载到nandflash再运行),但是运行nfs根文件系统的时候一直出错,各种错误。查看了很多资料后猜想应该是uboot传递给linux内核的参数有问题,也就是bootargs的设置有问题。
3219 0

热门文章

最新文章