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.mzph.cn/news/636935.shtml

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

相关文章

nginx日志分割

日志切割是线上常见的操作&#xff0c;能够控制单个日志文件的大小&#xff0c;便于对日志进行管理 给nginx主进程发送一个重新打开的信号&#xff0c;让nginx重新生成新的日志文件 nginx -s reopen 这个命令等同于kill -USR1 cat nginx.pid 切割日志文件shell命令 #!/bin/bas…

网站转小程序系统,任意网址打包成小程序

源码介绍 将任意网站打包成小程序&#xff0c;只需简单修改域名&#xff0c;即可轻松实现&#xff01;这一创新技术让您的网站内容在小程序平台上焕发新生。通过智能转换工具&#xff0c;您可以将任意网站迅速转化为小程序&#xff0c;无需繁琐的编码和开发工作。只需简单修改…

如何结合antd design pro 5 结合express 上传多个文件

在Ant Design Pro 5&#xff08;基于React&#xff09;的前端界面结合Express后端实现上传整个文件夹的文件&#xff0c;实际上是在前端进行多文件选择&#xff0c;并通过POST请求将文件列表发送到后端&#xff0c;然后由后端处理上传。由于浏览器API限制&#xff0c;直接上传整…

qt学习:模仿qq界面+添加资源+无边框界面+修改样式

目录 一,创建登录ui界面类 LoginWidget 二,添加图片资源 三,通过样式的方法将图片设置成圆圈的背景 四,新建登录后的ui界面 MWindow 简陋的就可以,因为只为了学习,可以自己补充 五,新建三个嵌套ui界面类,ChatWidget聊天界面 FriendWiidget好友界面 CollectW…

vue3-模版引用

模版引用 ref 属性 场景&#xff1a;需要直接访问底层 DOM 元素。 方法&#xff1a;使用特殊的 ref 属性。 <input ref"input">ref 属性 允许我们在一个特定的 DOM 元素或子组件实例被挂载后&#xff0c;获得对它的直接引用。 访问模板引用 小 Demo: 当 i…

SpringBoot:详解Bean生命周期和作用域

&#x1f3e1;浩泽学编程&#xff1a;个人主页 &#x1f525; 推荐专栏&#xff1a;《深入浅出SpringBoot》《java项目分享》 《RabbitMQ》《Spring》《SpringMVC》 &#x1f6f8;学无止境&#xff0c;不骄不躁&#xff0c;知行合一 文章目录 前言一、生命周期二…

SQL笔记 -- 锁

1. 概述 在数据库中&#xff0c;除传统的计算资源&#xff08;如CPU、RAM、I/O等&#xff09;的争用以外&#xff0c;数据也是一种供许多用户共享的资源。为保证数据的一致性&#xff0c;需要对并发操作进行控制&#xff0c;因此产生了锁 。同时锁机制也为实现MySQL 的各个隔离…

继电器开关电路图大全

继电器是一种电控制器件&#xff0c;是当输入量&#xff08;激励量&#xff09;的变化达到规定要求时&#xff0c;在电气输出电路中使被控量发生预定的阶跃变化的一种电器。它具有控制系统&#xff08;又称输入回路&#xff09;和被控制系统&#xff08;又称输出回路&#xff0…

vue-Mixin-复用代码片段

1. 介绍 在软件开发中&#xff0c;代码复用是一个重要的原则&#xff0c;它可以帮助我们避免重复的代码&#xff0c;提高代码的可维护性和可读性。在Vue.js中&#xff0c;我们可以使用mixins来复用代码片段。Mixin可以包含任何组件选项。当组件使用Mixin时&#xff0c;所有Mix…

Python笔记10-数据可视化练习折线图

文章目录 JSON数据Python数据和Json数据的相互转化pyecharts模块构建折线图全局配置绘制疫情数据折线图 JSON数据 JSON是一种轻量级的数据交互格式。可以按照JSON指定的格式去组织和封装数据 。本质上是一个带有特定格式的字符串 主要功能&#xff1a;可以在各个编程语言中流通…

rust 各种骚操作

前言 rust 代码确实简介&#xff0c;但是各种操作做层出不穷&#xff0c;这里记录一下一些难以理解的晦涩语法。 正文 奇怪的省略值。 fn main() {let numbers (2, 4, 8, 16, 32);match numbers {(first, .., last) > {println!("Some numbers: {}, {}", fir…

零日漏洞:威胁与应对

一、引言 随着信息技术的迅猛发展&#xff0c;网络安全问题日益凸显。其中&#xff0c;零日漏洞已成为当今网络安全领域最受关注的问题之一。本文将深入探讨零日漏洞的威胁、产生原因以及应对策略&#xff0c;以期提高人们对这一问题的认识和防范意识。 二、零日漏洞的威胁 …

892. 台阶-Nim游戏

#include<bits/stdc.h>using namespace std;int main() {int n;cin>>n;int res0;for(int i1;i<n;i){int x;cin>>x;if(i%2){res^x;}}if(res) puts("Yes");else puts("No");return 0; }最优游戏策略是&#xff1a;如果另一个人操作的…

PostgreSQL 的对象层次

所有的数据库离开数据量来谈性能都是耍流氓。 就你那几万条的数据库&#xff0c;用啥都行&#xff0c;典型的就是怎么方便怎么来。 不过 PostgreSQL 上手确实比 MySQL 概念更多。 PostgreSQL 比 MySQL 多了一层。 PostgreSQL 是从PostgreSQL 是从 Database&#xff0c;到 S…

C++---string类

一.string类&#x1f357; C支持C风格的字符串&#xff0c;另外还提供了一种 字符串数据类型&#xff1a; string是定义在头文件string中的类&#xff0c;使用前需要包含头文件string。 #include<string> C语言中的字符串需要引用头文件#include<string.h> #includ…

RabbitMQ-业务的幂等性

一、幂等性处理方式 1、使用唯一id 生产者和消费者都需要添加配置类&#xff1a; Beanpublic MessageConverter jsonMessageConverter() {Jackson2JsonMessageConverter jackson2JsonMessageConverter new Jackson2JsonMessageConverter();jackson2JsonMessageConverter.set…

SpringMVC传递数据给前台

SpringMVC有三种方式将数据提供给前台 第一种 使用Request域 第二种 使用Model&#xff08;数据默认是存放在Request域中&#xff09; 与第一种方式其实是一致的 第三种 使用Map集合&#xff08;数据默认是存放在Request域中&#xff09;

20234.1.20 使用idea进行Java的helloworld程序开发

20234.1.20 使用idea进行Java的helloworld程序开发 idea毕竟是jtbrain的产品&#xff0c;整体和pycharm相同&#xff0c;初步使用感受比eclipse更亲切 一、程序结构 project&#xff08;项目&#xff0c;工程&#xff09; module&#xff08;模块&#xff09; package&…

回调地狱与解决方案

什么是回调地狱&#xff1f; 简单理解就是回调函数嵌套回调 示例&#xff1a; setTimeout(() > {console.log(1);setTimeout(() > {console.log(2);setTimeout(() > {console.log(3);}, 1000);}, 2000)}, 3000)如上代码所示&#xff0c;回调函数嵌套回调&#xff0c;就…

Django从入门到精通(一)

目录 一、Django环境搭建与命令 1.1、安装 1.2、命令行 创建项目 编写代码 运行 app概念 1.3、Pycharm创建项目 1.4、虚拟环境 创建虚拟环境 - 命令行 介绍 操作 基本问题 Pycharm 项目虚拟环境 django虚拟环境【安装django最新版本】 django虚拟环境【安装指…