debugfs_create_fileLinux中debugfs的解析debugfs_remove_recursiveDEFINE_SIMPLE_ATTRIBUTE

本文主要是介绍debugfs_create_fileLinux中debugfs的解析debugfs_remove_recursiveDEFINE_SIMPLE_ATTRIBUTE,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Linux中debugfs的解析

2016年06月27日 17:56:17

阅读数:1088

debugfs_create_dir和debugfs_create_file定义在fs/debugfs/inode.c,函数原型声明在include/linux/fs.h。

struct dentry *debugfs_create_dir(const char *name, struct dentry *parent);
struct dentry *debugfs_create_file(const char *name, umode_t mode, struct dentry *parent, void *data, const struct file_operations *fops);
name是创建的文件或目录的名字,mode是文件的权限,parent是父目录,data通常指向驱动程序分配的数据,fops指向驱动分配的file_operations结构体,该结构体指定了应用程序读写时对调用的驱动程序函数。
对应的remove函数:
void debugfs_remove_recursive(struct dentry *dentry);
参数是调用debugfs_create_dir函数返回的指针。

配置CONFIG_DEBUG_FS=y让内核支持debugfs,也可以在menuconfig中指定:
Kernel hacking  --->
    [*] Debug Filesystem
内核默认会将debugfs挂载在/sys/kernel/debug目录下,也可以指定挂载目录:mount -t debugfs none /mnt
使用debugfs文件系统调试,先用宏DEFINE_SIMPLE_ATTRIBUTE指定使用读写函数和可变参数的格式,然后调用debugfs_create_dir和debugfs_create_file在文件系统下面创建文件,执行cat和echo时,会调用驱动的读写函数。

DEFINE_SIMPLE_ATTRIBUTE宏定义在include/linux/fs.h。
#define DEFINE_SIMPLE_ATTRIBUTE(__fops, __get, __set, __fmt)        \
static int __fops ## _open(struct inode *inode, struct file *file)    \
{                                    \
    __simple_attr_check_format(__fmt, 0ull);            \
    return simple_attr_open(inode, file, __get, __set, __fmt);    \
}                                    \
static const struct file_operations __fops = {                \
    .owner     = THIS_MODULE,                        \
    .open     = __fops ## _open,                    \
    .release = simple_attr_release,                    \
    .read     = simple_attr_read,                    \
    .write     = simple_attr_write,                    \
    .llseek     = generic_file_llseek,                    \
};

DEFINE_SIMPLE_ATTRIBUTE定义了一个file_operations结构体,并指定了open,read,write等函数,simple_attr_open函数关联了驱动的read,write函数,应用程序read,write时,通过simple_attr_read和simple_attr_write来调用驱动程序的read,write函数。
int simple_attr_open(struct inode *inode, struct file *file, int (*get)(void *, u64 *), int (*set)(void *, u64), const char *fmt)
{
    struct simple_attr *attr;

    attr = kmalloc(sizeof(*attr), GFP_KERNEL);
    if (!attr)
        return -ENOMEM;

    attr->get = get;
    attr->set = set;
    attr->data = inode->i_private;
    attr->fmt = fmt;
    mutex_init(&attr->mutex);

    file->private_data = attr;

    return nonseekable_open(inode, file);
}

attr->get和attr->set分别指向驱动的get和set函数。attr->data指向了驱动分配的数据,在调用debugfs_create_file时,该函数会inode->i_private指正指向第四个参数,这个参数一般是驱动程序的数据指针。
ssize_t simple_attr_read(struct file *file, char __user *buf,
             size_t len, loff_t *ppos)
{
    struct simple_attr *attr;
    size_t size;
    ssize_t ret;

    attr = file->private_data;

    if (!attr->get)
        return -EACCES;

    ret = mutex_lock_interruptible(&attr->mutex);
    if (ret)
        return ret;

    if (*ppos) {        /* continued read */
        size = strlen(attr->get_buf);
    } else {        /* first read */
        u64 val;
        ret = attr->get(attr->data, &val);
        if (ret)
            goto out;

        size = scnprintf(attr->get_buf, sizeof(attr->get_buf),
                 attr->fmt, (unsigned long long)val);
    }

    ret = simple_read_from_buffer(buf, len, ppos, attr->get_buf, size);
out:
    mutex_unlock(&attr->mutex);
    return ret;
}

通过attr = file->private_data;获得在simple_attr_open函数分配的attr结构体地址,然后ret = attr->get(attr->data, &val);用来调用驱动的get函数,读取到的值存放在val中,最后调用scnprintf函数将val按照DEFINE_SIMPLE_ATTRIBUTE中定义的字符串格式返回到simple_attr结构体的get_buf字符数组。
ssize_t simple_attr_write(struct file *file, const char __user *buf,
              size_t len, loff_t *ppos)
{
    struct simple_attr *attr;
    u64 val;
    size_t size;
    ssize_t ret;

    attr = file->private_data;
    if (!attr->set)
        return -EACCES;

    ret = mutex_lock_interruptible(&attr->mutex);
    if (ret)
        return ret;

    ret = -EFAULT;
    size = min(sizeof(attr->set_buf) - 1, len);
    if (copy_from_user(attr->set_buf, buf, size))
        goto out;

    attr->set_buf[size] = '\0';
    val = simple_strtoll(attr->set_buf, NULL, 0);
    ret = attr->set(attr->data, val);
    if (ret == 0)
        ret = len; /* on success, claim we got the whole input */
out:
    mutex_unlock(&attr->mutex);
    return ret;
}
首先和simple_attr_read一样获得attr的地址,然后调用copy_from_user获得echo传入的参数,接着利用simple_strtoll将字符传格式的数字转换为整型,最后调用驱动的set函数,将值写入attr->data中。

这篇关于debugfs_create_fileLinux中debugfs的解析debugfs_remove_recursiveDEFINE_SIMPLE_ATTRIBUTE的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/985018

相关文章

Java图片压缩三种高效压缩方案详细解析

《Java图片压缩三种高效压缩方案详细解析》图片压缩通常涉及减少图片的尺寸缩放、调整图片的质量(针对JPEG、PNG等)、使用特定的算法来减少图片的数据量等,:本文主要介绍Java图片压缩三种高效... 目录一、基于OpenCV的智能尺寸压缩技术亮点:适用场景:二、JPEG质量参数压缩关键技术:压缩效果对比

关于WebSocket协议状态码解析

《关于WebSocket协议状态码解析》:本文主要介绍关于WebSocket协议状态码的使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录WebSocket协议状态码解析1. 引言2. WebSocket协议状态码概述3. WebSocket协议状态码详解3

CSS Padding 和 Margin 区别全解析

《CSSPadding和Margin区别全解析》CSS中的padding和margin是两个非常基础且重要的属性,它们用于控制元素周围的空白区域,本文将详细介绍padding和... 目录css Padding 和 Margin 全解析1. Padding: 内边距2. Margin: 外边距3. Padd

Oracle数据库常见字段类型大全以及超详细解析

《Oracle数据库常见字段类型大全以及超详细解析》在Oracle数据库中查询特定表的字段个数通常需要使用SQL语句来完成,:本文主要介绍Oracle数据库常见字段类型大全以及超详细解析,文中通过... 目录前言一、字符类型(Character)1、CHAR:定长字符数据类型2、VARCHAR2:变长字符数

使用Jackson进行JSON生成与解析的新手指南

《使用Jackson进行JSON生成与解析的新手指南》这篇文章主要为大家详细介绍了如何使用Jackson进行JSON生成与解析处理,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1. 核心依赖2. 基础用法2.1 对象转 jsON(序列化)2.2 JSON 转对象(反序列化)3.

Springboot @Autowired和@Resource的区别解析

《Springboot@Autowired和@Resource的区别解析》@Resource是JDK提供的注解,只是Spring在实现上提供了这个注解的功能支持,本文给大家介绍Springboot@... 目录【一】定义【1】@Autowired【2】@Resource【二】区别【1】包含的属性不同【2】@

SpringCloud动态配置注解@RefreshScope与@Component的深度解析

《SpringCloud动态配置注解@RefreshScope与@Component的深度解析》在现代微服务架构中,动态配置管理是一个关键需求,本文将为大家介绍SpringCloud中相关的注解@Re... 目录引言1. @RefreshScope 的作用与原理1.1 什么是 @RefreshScope1.

Java并发编程必备之Synchronized关键字深入解析

《Java并发编程必备之Synchronized关键字深入解析》本文我们深入探索了Java中的Synchronized关键字,包括其互斥性和可重入性的特性,文章详细介绍了Synchronized的三种... 目录一、前言二、Synchronized关键字2.1 Synchronized的特性1. 互斥2.

Java的IO模型、Netty原理解析

《Java的IO模型、Netty原理解析》Java的I/O是以流的方式进行数据输入输出的,Java的类库涉及很多领域的IO内容:标准的输入输出,文件的操作、网络上的数据传输流、字符串流、对象流等,这篇... 目录1.什么是IO2.同步与异步、阻塞与非阻塞3.三种IO模型BIO(blocking I/O)NI

Python 中的异步与同步深度解析(实践记录)

《Python中的异步与同步深度解析(实践记录)》在Python编程世界里,异步和同步的概念是理解程序执行流程和性能优化的关键,这篇文章将带你深入了解它们的差异,以及阻塞和非阻塞的特性,同时通过实际... 目录python中的异步与同步:深度解析与实践异步与同步的定义异步同步阻塞与非阻塞的概念阻塞非阻塞同步