Linux设备管理模型-02:sysfs

文章目录

  • sysfs
  • 1 使用sysfs控制GPIO
  • 2 sysfs编程
    • 2.1 完善sysfs属性文件的读写操作

上一篇文: 设备管理模型中的基础数据结构

sysfs

sysfs是用于导出内核对象的文件系统,它是一个基于ram的文件系统,最初基于ramfs。
sysfs通常挂载在/sys目录下。它提供了一种层次结构来表示设备、驱动程序和总线之间的关系,以及设备属性的信息。用户和管理者可以使用sysfs来查询和配置设备的状态和属性。

如果定义了CONFIG_SYSFS,那么sysfs总是编译进内核。

~ # ls /sys
block     class     devices   fs        module
bus       dev       firmware  kernel    power

sysfs中的目录树是由kobject和kset组织而成的。
对于每一个注册到系统的kobject,都会在sysfs中创建一个对应的目录,他是kobject父对象的子目录,以此像用户空间表达内部对象层级。
相同的kset下有多个kobject,kset又由链表连接,他们共同组成了这个目录树。

一个kset下的kobject可以有相同的ktype,也可以不同。

sys目录下的各目录作用如下:

目录用途
block
bus包含内核中各种总线类型的扁平化布局,每个目录下都分别包含devicesdrivers 两个目录。
class
dev包含 2 个目录: char 和 block,它们里面是 <major>:<minor> 格式指向设备的符号链。
devices包含表示设备树的一个文件系统,它直接映射至内部内核设备树,即 struct device 层级
firmware包含硬件固件相关信息
fs包含针对一些文件系统的目录,每个需要导出属性的文件系统都必须在/fs下
kernel包含内核信息和控制接口
module
power

1 使用sysfs控制GPIO

make menuconfig 确保 Device Drivers > LED Support 没有 LED Class Support(如果使用LED做实验) , 并且Device Drivers > GPIO Support 已使能。

GPIO1_0 : (1 - 1) * 32 + 0 = 0
如下命令导出GPIO1_0
echo 0 > /sys/class/gpio/export

GPIOX_N引脚:(X - 1) * 32 + N

[root@qemu_imx6ul:/sys/class/gpio]# ls
export       gpiochip0    gpiochip32   gpiochip96
gpio0        gpiochip128  gpiochip64   unexport
[root@qemu_imx6ul:/sys/class/gpio]# ls gpio0
active_low  direction   power       uevent
device      edge        subsystem   value

echo out > /sys/class/gpio/gpio0/direction
echo 1 > /sys/class/gpio/gpio0/value

2 sysfs编程

2.1 完善sysfs属性文件的读写操作

上一篇文kset例程中只有kset和kobject的目录组织关系,它还需要能够读写控制才比较完整。本节基于该例程完善了kobject在sysfs的各种行为。

在1.2节的例程中仅定义了kobj_type的release方法,属性的通用读写操作也在kobj_type中:

#include <linux/kobject.h>
#include <linux/string.h>
#include <linux/sysfs.h>
#include <linux/slab.h>
#include <linux/module.h>
#include <linux/init.h>MODULE_LICENSE("GPL");struct foo_obj {struct kobject kobj;int foo;int baz;int bar;
};
/* 从类型为kobj的结构体成员的指针x,获取该foo_obj类型结构体的指针 */
#define to_foo_obj(x) container_of(x, struct foo_obj, kobj)/* 自定义属性,继承自attribute */
struct foo_attribute {struct attribute attr; /* 包含name 和 mode 成员变量*/ssize_t (*show)(struct foo_obj *foo, struct foo_attribute *attr, char *buf);ssize_t (*store)(struct foo_obj *foo, struct foo_attribute *attr, const char *buf, size_t count);
};
#define to_foo_attr(x) container_of(x, struct foo_attribute, attr)/** 每当与已注册kobject关联的sysfs文件上的show函数被用户调用时,这个函数就会被sysfs调用。* 需要把传入的kobj转置为自己的kobj子类,然后调用这个特定对象的show函数 */
static ssize_t foo_attr_show(struct kobject *kobj,struct attribute *attr,char *buf)
{struct foo_attribute *attribute;struct foo_obj *foo;attribute = to_foo_attr(attr);foo = to_foo_obj(kobj);if (!attribute->show)return -EIO;return attribute->show(foo, attribute, buf);
}/* 通过sysfs写入属性文件时被调用 */
static ssize_t foo_attr_store(struct kobject *kobj,struct attribute *attr,const char *buf, size_t len)
{struct foo_attribute *attribute;struct foo_obj *foo;attribute = to_foo_attr(attr);foo = to_foo_obj(kobj);if (!attribute->store)return -EIO;return attribute->store(foo, attribute, buf, len);
}/* Our custom sysfs_ops that we will associate with our ktype later on */
static const struct sysfs_ops foo_sysfs_ops = {.show = foo_attr_show,.store = foo_attr_store,
};/* 属性的读取函数,每个属性可以有不同的读取 */
static ssize_t var_show(struct foo_obj *foo_obj, struct foo_attribute *attr,char *buf)
{int var;if (strcmp(attr->attr.name, "len_attr") == 0)var = foo_obj->foo;else if (strcmp(attr->attr.name, "derict_attr") == 0)var = foo_obj->baz;elsevar = foo_obj->bar;return sprintf(buf, "%d\n", var);
}
/* 属性的写入函数,每个属性可以有不同的写入 */
static ssize_t var_store(struct foo_obj *foo_obj, struct foo_attribute *attr,const char *buf, size_t count)
{int var, ret;ret = kstrtoint(buf, 10, &var);if (ret < 0)return ret;if (strcmp(attr->attr.name, "len_attr") == 0)foo_obj->foo = var;else if (strcmp(attr->attr.name, "derict_attr") == 0)foo_obj->baz = var;elsefoo_obj->bar = var;return count;
}
/* Sysfs attributes cannot be world-writable.   * __ATTR(name, mode, show, store) : 生成一个包含属性的结构体,以及相关的访问函数。* 这样,可以将属性与对应的读写函数关联起来,并在 sysfs 中创建相应的属性文件,用于读取或写入属性值。*/
static struct foo_attribute foo_attribute =__ATTR(len_attr, 0664, var_show, var_store);
static struct foo_attribute baz_attribute =__ATTR(derict_attr, 0664, var_show, var_store);
static struct foo_attribute bar_attribute =__ATTR(depth_attr, 0664, var_show, var_store);/** Create a group of attributes so that we can create and destroy them all* at once.*/
static struct attribute *foo_default_attrs[] = {&foo_attribute.attr,&baz_attribute.attr,&bar_attribute.attr,NULL,	/* need to NULL terminate the list of attributes */
};static void foo_release(struct kobject *kobj)
{struct foo_obj *foo;foo = to_foo_obj(kobj);kfree(foo);
}/* 可以定义所属kset的kobject的一些行为到default_attrs和sysfs_ops属性中 */
static struct kobj_type foo_ktype = {.sysfs_ops = &foo_sysfs_ops,.release = foo_release,.default_attrs = foo_default_attrs,
};static struct kset *example_kset;
static struct foo_obj *foo_obj;
static struct foo_obj *bar_obj;
static struct foo_obj *baz_obj;static struct foo_obj *create_foo_obj(const char *name)
{struct foo_obj *foo;int retval;/* allocate the memory for the whole object */foo = kzalloc(sizeof(*foo), GFP_KERNEL);if (!foo)return NULL;/* 初始化kobject之前先确定所属kset */foo->kobj.kset = example_kset;/* 初始化kobject添加到kernel中,并关联ktype,会在sysfs中创建名为name的kobject文件夹* 第三个参数是父kobj,由于已确定kset,写为NULL */retval = kobject_init_and_add(&foo->kobj, &foo_ktype, NULL, "%s", name);if (retval) {kobject_put(&foo->kobj);return NULL;}/* 通知用户空间有一个新的内核对象(kobject)已经被添加到 sysfs 中。* 这对于用户空间的监控和管理工具来说是很有用的 */kobject_uevent(&foo->kobj, KOBJ_ADD);return foo;
}static void destroy_foo_obj(struct foo_obj *foo)
{kobject_put(&foo->kobj);
}static int __init example_init(void)
{/* 创建一个名为 "kset_example" 的kset, 路径在/sys/kernel/ */example_kset = kset_create_and_add("kset_example", NULL, kernel_kobj);if (!example_kset)return -ENOMEM;/* 在已定义的kset下新增kobject */foo_obj = create_foo_obj("foo");if (!foo_obj)goto foo_error;bar_obj = create_foo_obj("bar");if (!bar_obj)goto bar_error;baz_obj = create_foo_obj("baz");if (!baz_obj)goto baz_error;return 0;baz_error:destroy_foo_obj(bar_obj);
bar_error:destroy_foo_obj(foo_obj);
foo_error:kset_unregister(example_kset);return -EINVAL;
}static void __exit example_exit(void)
{destroy_foo_obj(baz_obj);destroy_foo_obj(bar_obj);destroy_foo_obj(foo_obj);kset_unregister(example_kset);
}MODULE_AUTHOR("LUKEKE");        // 作者
MODULE_DESCRIPTION("kset test"); // 描述
MODULE_ALIAS("kset Learn");   // 别名module_init(example_init);
module_exit(example_exit);

测试解果如下

[root@qemu_imx6ul:/sys/kernel/kset_example]# ls
bar  baz  foo
[root@qemu_imx6ul:/sys/kernel/kset_example]# ls baz
depth_attr   derict_attr  len_attr
[root@qemu_imx6ul:/sys/kernel/kset_example/foo]# cd foo & ls
depth_attr   derict_attr  len_attr
[root@qemu_imx6ul:/sys/kernel/kset_example/foo]# echo 5 > depth_attr 
[root@qemu_imx6ul:/sys/kernel/kset_example/foo]# cat depth_attr 
5
[root@qemu_imx6ul:/sys/kernel/kset_example/foo]# echo 3 > derict_attr 
[root@qemu_imx6ul:/sys/kernel/kset_example/foo]# cat derict_attr 
3

除了上述通过kobject_init_and_add设置ktype的方式创建属性文件,也可以在kobject_create_and_add("my_kobject", kernel_kobj)之后使用sysfs_create_files(my_kobject, attrs)创建属性文件。

属性读写的简单描述如下图所示:
sysfs属性文件读写

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/417816.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

机床导轨误差对机械加工品质的影响

机床导轨是确定机床各部件相对位置关系的基准&#xff0c;也是相对运动的基准&#xff0c;其精度直接决定了机械加工产品的精度&#xff0c;机床导轨要确保精准,才能保证机械加工产品的良好&#xff0c;否则&#xff0c;会影响加工品质&#xff0c;主要体现在以下几方面&#x…

JAVA和C++ SECS/GEM300开发和概念

编译SECS示例程序 1. 示例程序使用默认路径&#xff1a; D:\SECS 稳定版\SECS Debug\ 2. 该操作分为俩步 ① 将C#的Secs库编译成设备相同Net版本。 如.net3.5、4.0、4.5等等 ② 编译金南瓜SECS demo程序 编译C#的SecsEquip.dll 1. 找到SecsEquip项目 项目文件 使用Visua…

第35集《佛法修学概要》

己四 、 精进度 分三&#xff1a;庚一、 精进自性。庚 二、趣入修习精进方便。 庚三、修习精进差别内容 请大家打开讲义第九十四页&#xff0c;我们看己四&#xff0c;精进度。 当我们从人天乘一个好人的阶段提升到一种菩萨道的修学&#xff0c;我们就要注意两个重点了。在我…

JAVA开发环境配置(保姆式教程)

目录 安装JDK 配置环境 下载IntelliJ IDEA 使用 IntelliJ IDEA 安装JDK JDK可以在电脑自带的应用商店下载 也可以如下方式下载 首先在网页中搜索ORACLE官网 然后点击产品找到Java 往下滑点击Java SE&#xff0c;点击Download Java now进入下载 或者直接点击Download Java下载 这…

常用界面设计组件 —— 窗体(QT)

二、常用界面设计组件2.1 窗体2.1.1 设置窗体位置、大小及背景颜色2.1.2 设置窗体标题2.1.3 多窗体调用 二、常用界面设计组件 组件是GUI的基本元素&#xff0c;也称为UI控件。它接受来自底层平台的不同用户事件&#xff0c;如鼠标和键盘事件&#xff08;以及其它事件&#xf…

【MySQL】一文总结MVCC多版本并发控制

目录 MVCC 介绍当前读和快照读当前读快照读 MVCC 原理解析隐式字段Undo Log版本链Read ViewRead View 可见性原则 RC 和 RR 下的 Read ViewRC 下的 Read ViewRR 下的 Read View小结RR 级别下能否防止幻读总结 MVCC 介绍 在当今高度并发的数据库环境中&#xff0c;有效的并发控…

汇川PLC如何连接电脑进行数据通信和远程上下载

准备工作 一台可联网操作的电脑一台单网口的远程透传网关及博达远程透传配置工具网线一条&#xff0c;用于实现网络连接和连接PLC一台汇川PLC及其编程软件InoPrShop一张4G卡或WIFI天线实现通讯(使用4G联网则插入4G SIM卡&#xff0c;WIFI联网则将WIFI天线插入USB口&#xff09…

慢查询定位

慢查询 使用工具 mysql自带慢日志 默认没有开启需要手动开启 查看慢日志中的文件 总结

玩《Minecraft》的贝贝

贝贝通过指令获得了 n 个木棍和 m 个钻石&#xff0c;以及一个工作台&#xff0c;他想要制造尽可能多的工具。 一共有五种钻石工具&#xff0c;下面是每种钻石工具的合成方案&#xff1a; 输入 第一行&#xff0c;一个整数&#xff0c;为数组组数 T (1 ≤ T ≤ 1e5)。 接下来的…

【软件测试】学习笔记-统一测试数据平台

这篇文章主要探讨全球大型电商企业中关于准备测试数据的最佳实践&#xff0c;从全球大型电商企业早期的测试数据准备实践谈起&#xff0c;分析这些测试数据准备方法在落地时遇到的问题&#xff0c;以及如何在实践中解决这些问题。其实&#xff0c;这种分析问题、解决问题的思路…

储能变流器(PCS)构网型和跟网型区别

跟网型本质为电流源&#xff0c;自身无法提供电压与频率支撑&#xff0c;必须依赖电网电压和频率&#xff0c;无法支撑系统&#xff1b;构网型本质为电压源&#xff0c;内部设定电压参数信号输出电压与频率&#xff0c;既可并网也可离网运行&#xff0c;对电网支撑能力强。因新…

日志记录logging

文章目录 1. logging基础使用1.1 日志的6个级别1.2 logging.basicConfig1.3 案例 2. logging的高级应用2.1 记录器Logger2.2 处理器- Handler2.3 格式器- Formatter2.4 创建关联2.4 案例 3.在项目中的应用3.1 定义全局使用的logger对象3.2 使用案例 参考 1. logging基础使用 1…