我的内核学习笔记5:proc目录文件创建及读写

上一篇内核学习笔记《我的内核学习笔记4:sysfs学习》是2013年写的,彼时至今,随着工作的展开和安排,内核方面的知识可谓突飞猛进,当然,其它方面亦是如此。关于内核方面,积累的笔记大大小小有几十篇了,但只是笔记形式或代码片段,无法形成文章,不敢献艺,怕贻笑大方。

最近研究网络子系统,想在内核层打印调试信息,但网络数据十分频繁,所以需要使用手段来控制调试信息的输出。以前一直使用sysfs,但它与具体的驱动有关,不方便,这次使用procfs,直接在系统的/proc目录下创建文件,使用echo写入不同等级,以显示不同信息。


完整代码如下:

/**
/proc创建文件示例

*/

#include <linux/module.h>
#include <linux/init.h>
#include <linux/kernel.h>

#include <linux/fs.h>
#include <linux/proc_fs.h>
#include <linux/seq_file.h>

#include <linux/uaccess.h>

#define MY_VERSION "foo"

static int ver = 0;

// 写入到proc目录指定文件
static int my_version_proc_show(struct seq_file *m, void *v)
{
     seq_printf(m, "%s %d\n",MY_VERSION, ver);
     return 0;
}

static int my_version_proc_open(struct inode *inode, struct file *file)
{
     return single_open(file, my_version_proc_show, NULL);
}

static ssize_t my_write(struct file *filp, const char __user *buf, size_t count, loff_t *f_ops)
{
    // 简单数字形式,只能0~9
#if 1
    unsigned char tmp = 0;
    int value = 0;

    get_user(tmp, buf);
    value = simple_strtol(&tmp, NULL, 10);
    printk(KERN_ERR "got user info: %d....\n", value);
    
    ver = value;
    
    return count;
#else
    unsigned char buffer[32] = {0};
    int value = 0;

    if (copy_from_user(buffer, buf, (count > 32 ? 32: count)))
        goto out;

    // 如果只是输入简单的数字,可以:get_user(foo, buf);
    value = simple_strtol(buffer, NULL, 10);
    printk(KERN_ERR "got user info: %d....\n", value);
    
    ver = value;
out:
    return count;
#endif
}

static const struct file_operations my_debug_proc_fops = {
    .open      = my_version_proc_open,
    .read      = seq_read,
    .write     = my_write,
    .llseek    = seq_lseek,
    .release   = single_release,
};

static struct proc_dir_entry* my_proc_entry = NULL; 

void create_debugproc(void)
{
    if (!my_proc_entry)
    {
        my_proc_entry = proc_create("myversion4", 0, NULL, &my_debug_proc_fops); // 在/proc目录下创建
    }
}

void delete_debugproc(void)
{
    if (my_proc_entry)
        proc_remove(my_proc_entry);
}

/
static int __init procfs_init(void)
{
    printk(KERN_ERR "in %s\n", __func__);
    
    create_debugproc();
    create_debugproc(); // note:多次调用,但该函数已经确保只注册一次

    return 0;
}

static void __exit procfs_exit(void)
{
    printk(KERN_ERR "in %s\n", __func__);

    delete_debugproc();
}

module_init(procfs_init);
module_exit(procfs_exit);

MODULE_LICENSE("GPL");
MODULE_DESCRIPTION("proc fs entry test driver by Late Lee");


PS:对于内核的跟踪,推荐使用stack_dump,在需要的函数中直接调用stack_dump()即可,当内核运行到对应的函数会将调用栈打印出来,就可以一窥函数调用过程了。

李迟 2016.10.13 周四 晚上

  • 0
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Linuxproc文件系统是一种虚拟文件系统,它将系统内核的信息以文件的形式呈现给用户空间。用户可以通过读取和写入这些文件来与内核进行通信。proc文件系统文件包含了系统的各种信息,如进程信息、内存信息、网络信息等。用户可以通过读取这些文件来获取系统的状态信息,也可以通过写入这些文件来修改系统的配置参数。因此,proc文件系统是实现用户和内核通信的一种重要方式。 ### 回答2: Linuxproc文件系统是一种特殊的文件系统,它将系统的进程信息以文件的形式暴露出来,使用户空间和内核空间可以通过文件访问的方式进行通信。 在proc文件系统,每个进程都被视为一个目录目录的名称为进程的PID(进程标识符)。每个PID目录下都包含了一系列的文件,这些文件用于描述进程的不同属性和状态。 用户空间可以通过读取proc文件系统文件来获取进程的信息。例如,/proc/<PID>/status文件包含了关于进程的各种状态信息,如进程的PID、父进程的PID、进程运行状态等等。用户可以通过读取这个文件来了解进程的当前状态。 此外,proc文件系统的一些文件可以被用户写入,以改变进程的一些属性或执行一些操作。例如,/proc/<PID>/oom_adj文件可以被用户写入一个整数值,以调整进程在系统内存不足时被杀死的优先级。用户可以通过写入这个文件来改变进程的oom_adj优先级。 而在内核空间内核可以通过proc文件系统来与用户空间通信。内核可以通过创建和修改proc文件系统文件,向用户空间提供系统的信息或接收用户空间的请求。 总之,proc文件系统通过文件的形式提供了用户空间和内核空间之间的通信接口。用户空间可以通过读取和写入proc文件系统文件来获取和控制进程的信息,而内核空间则可以通过创建和修改proc文件系统文件来与用户空间进行通信。这种基于文件的通信方式使得用户和内核之间的通信更加灵活和方便。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值