创建自己的/proc文件——processinfo

2024-06-18 09:59
文章标签 创建 proc processinfo

本文主要是介绍创建自己的/proc文件——processinfo,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Cited from http://www.lupaworld.com/home-space-uid-401174-do-blog-id-149701.html

前面写过一个模块,创建内核进程之kernel_thread获取到了进程的一些信息。结合最近看得/proc文件系统,我写了个模块,将获取的信息读入到/proc文件中。
        主要思想是:利用proc_mkdir()创建一个mydir,再利用create_proc_read_entry()函数创建一个processinfo文件。我们从模块里面获取的信息都将写入到processinfo文件中。
        还是现看看具体的代码吧。
#include <linux/module.h>
#include <linux/list.h>
#include <linux/init.h>
#include <linux/kernel.h>
#include <linux/sched.h>
#include <linux/types.h>
#include <linux/kthread.h>
#include <linux/proc_fs.h>
//#define tasklist_lock_address 0xc04f1980 
static struct proc_dir_entry *tempdir, *processinfo_file;
static int proc_read_processinfo(char *page, char **start, off_t offset,int count, int *eof, void *data)  
{
    struct task_struct *pos, *task;
    static char buf[1024*8] = {0};
    char tmp[128] = {0};
    task = &init_task; 
       if(offset  > 0){
            return 0;
    }
      memset(buf, 0, sizeof(buf));
//    read_lock(tasklist_lock_address);
        list_for_each_entry(pos, &task->tasks, tasks){
            sprintf(tmp,"%d\t\t%s\n",pos->pid, pos->comm);
        strcat(buf, tmp);
       memset(tmp, 0 sizeof(tmp));
    }
//    read_unlock(tasklist_lock_address);
        *start=buf;
        return strlen(buf);
}
static int __init hello_list_process(void)
{
    int rv = 0;
    printk("kernel,I am coming!...........\n");
    tempdir = proc_mkdir("mydir", NULL);
    if(tempdir == NULL){
        rv = -ENOMEM;
        return rv;
    }
    
    processinfo_file = create_proc_read_entry("processinfo", 0444, tempdir, proc_read_processinfo, NULL);
        if(processinfo_file == NULL){
                rv = -ENOMEM;
                remove_proc_entry("mydir", NULL);
        }
    printk("%s initialised\n", "mydir");
    printk("create processinfo success!\n");
    return 0;
}
static void __exit bye_list_process(void)
{    
    remove_proc_entry("processinfo", tempdir);
    remove_proc_entry("mydir", NULL);
    printk("bye kernel.......\n");
}
module_init(hello_list_process);
module_exit(bye_list_process);
MODULE_LICENSE("GPL");
        代码思路很简单。创建文件成功后,利用宏list_for_each_entry()获取进程信息,并全部导入到processinfo中。若创建失败则 remove_proc_entry()。
        首先看一下结构体proc_dir_entry
        在/include/linux/proc_fs.h中:
struct proc_dir_entry {
unsigned int low_ino;
        unsigned short namelen;
        const char *name;
        mode_t mode;
        nlink_t nlink;
        uid_t uid;
        gid_t gid;
        loff_t size;
        const struct inode_operations *proc_iops;
        /*
         * NULL ->proc_fops means "PDE is going away RSN" or
         * "PDE is just created". In either case, e.g. ->read_proc won't be
         * called because it's too late or too early, respectively.
         *
         * If you're allocating ->proc_fops dynamically, save a pointer
         * somewhere.
         */
        const struct file_operations *proc_fops;
        struct proc_dir_entry *next, *parent, *subdir;
        void *data;
        read_proc_t *read_proc;
        write_proc_t *write_proc;
        atomic_t count;         /* use count */
        int pde_users;  /* number of callers into module in progress */
        spinlock_t pde_unload_lock; /* proc_fops checks and pde_users bumps *

        常用字段: read_proc_t *read_proc;,write_proc_t *write_proc.proc文件的读函数,写函数。read_proc_t 类型将在后面介绍。
        具体的我们来看看下面几个函数。
        proc_mkdir()主要功能是创建一个目录.函数原型为:
        struct proc_dir_entry *proc_mkdir(const char *name,struct proc_dir_entry *parent);
        参数说明如下:name就是你定义的目录名称。parent就是你定义的这个目录的父目录。若为NULL,表示为/proc。
        create_proc_read_entry()这个函数主要功能是创建一个只读文件。关于使用这个函数,我是然了很久关于只读只写这个问题的。因为内核里面还有create_proc_entry()函数可以创建可读可写文件。所以没想明白这个文件的权限时,我没发确定用哪个函数。
        那么我们来看看这个文件究竟应该是怎么样的权限了?对于内核的话,对于这个文件就应该是可写的,可是对于用户应该对这个文件只是可读的,那这个文件究竟是可读还是可写还是可读又可写。最后我然清楚了两个概念,才算是明白了。用户态数据,内核态数据。其实这个文件应该是对用户可读的,其实这也即使说对内核是可写的。那么我们要创建的就是一个可读文件。用户读的过程其实就是内核写的过程。那么我们就可以用create_proc_read_entry()这个函数直接创建一个可读的文件即可。
        看看create_proc_read_entry()的原型。static inline struct proc_dir_entry *create_proc_read_entry(const char *name, mode_t mode, struct proc_dir_entry *base,  read_proc_t *read_proc, void * data)
        参数描述如下:name为定义的文件名,mode为对该proc节点的访问权限, base指定建立的proc节点所在的目录。如果为NULL则指/proc目录,read_proc是建立的文件的读函数指针,参数data为该文件的专用数据,它将保存在该文件对应的struct file结构的private_data字段中。
        在介绍我自己写的/proc文件读函数proc_read_processinfo()函数之前,先介绍一下:
        typedef int (read_proc_t)(char *page, char **start, off_t off,
                          int count, int *eof, void *data);
        page:把需要返回给信息写入page(最好不要超过PAGE_SIZE(通常4K),否则比较麻烦)
        start:顾名思义,就是开始的指针。不经常使用。当然个人觉得这也暗示某个关键时候要用它。当我们写的数据的大小大于一个页面的时候,需要*start指针控制写入page的起始地址。例如在本模块中,数据的大小是1024*8,是8K,就需要用到*start指针。否则会出现乱码。
        off:page的偏移量,表示page已有的数据。例如本模块,如果off > 0,page原来就有数据,我们则跳出read函数,即不再往page里面写数据。
        count:用户所要读取的字节数目。
        peof:当读到文件尾时,置peof为1。
        data:当这个函数被多个proc文件定义为“读”函数时,我们可以通过这个指针传递参数。
        proc_read_processinfo()函数就是这样类型的函数。
我们写的这个函数,是将buf数组里面的数据写入page中。里面的memset函数主要只是起一个清洁数组的作用。不让这次写入的数据残留到下次的数据。
        remove_proc_entry()函数就是删除节点。函数原型为: void remove_proc_entry(const char *name, struct proc_dir_entry *parent).参数描述如下:name为你要删除的文件或者目录,是字符串型的。parent是它的父目录,注意它是struct proc_dir_entry指针型。  
        关于程序中注释掉的部分//#define tasklist_lock_address 0xc04f1980 ,//    read_lock(tasklist_lock_address);//    read_unlock(tasklist_lock_address);刚开始使用tasklist_lock时,是考虑到读取的时候需要锁上,保护读的过程。tasklist_lock读写自旋锁,保护以init_task为头节点的进程链表。那么就使用它了,可是发现一个问题,就是加载模块后提示:Undefined Symbol。dmesg一下发现提示是:Unknown symbol tasklist_lock。打开/proc/kallsyms里面发现c04f1980 D tasklist_lock然后就这样给用上了。这样应该感觉不对,虽然没有了错误,但是提示有警告。
        整个程序下来运行结果是:
susu@corechen-desktop:/proc$ cd mydir/
susu@corechen-desktop:/proc/mydir$ ls
processinfo
susu@corechen-desktop:/proc/mydir$ cat processinfo 
1        init
2        kthreadd
3        migration/0
4        ksoftirqd/0
5        watchdog/0
6        migration/1
7        ksoftirqd/1
8        watchdog/1
9        events/0
10        events/1
11        cpuset
12        khelper
17        async/mgr
67        sync_supers
69        bdi-default
71        kblockd/0
72        kblockd/1
74        kacpid
75        kacpi_notify
76        kacpi_hotplug
162        kseriod
218        khungtaskd
219        kswapd0
264        aio/0
265        aio/1
268        crypto/0
269        crypto/1
1534        ksuspend_usbd
1542        khubd
1672        ata/0
1673        ata/1
1674        ata_aux
1685        scsi_eh_0
1687        scsi_eh_1
2663        kjournald
2848        udevd
3049        kpsmoused
4590        getty
4591        getty
4593        getty
4594        getty
4595        getty
4717        acpid
4741        kconservative/0
4742        kconservative/1
4756        kondemand/0
4757        kondemand/1
4826        syslogd
4884        dd
4886        klogd
4908        dbus-daemon
4924        NetworkManager
4938        NetworkManagerD
4951        system-tools-ba
4973        sshd
5003        avahi-daemon
5004        avahi-daemon
5035        flush-8:0
5056        mysqld_safe
5098        mysqld
5099        logger
5192        cupsd
5209        nginx
5210        nginx
5305        winbindd
5352        winbindd
5353        dhcdbd
5372        hald
5375        console-kit-dae
5376        hald-runner

这篇关于创建自己的/proc文件——processinfo的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

【Python编程】Linux创建虚拟环境并配置与notebook相连接

1.创建 使用 venv 创建虚拟环境。例如,在当前目录下创建一个名为 myenv 的虚拟环境: python3 -m venv myenv 2.激活 激活虚拟环境使其成为当前终端会话的活动环境。运行: source myenv/bin/activate 3.与notebook连接 在虚拟环境中,使用 pip 安装 Jupyter 和 ipykernel: pip instal

在cscode中通过maven创建java项目

在cscode中创建java项目 可以通过博客完成maven的导入 建立maven项目 使用快捷键 Ctrl + Shift + P 建立一个 Maven 项目 1 Ctrl + Shift + P 打开输入框2 输入 "> java create"3 选择 maven4 选择 No Archetype5 输入 域名6 输入项目名称7 建立一个文件目录存放项目,文件名一般为项目名8 确定

Java 创建图形用户界面(GUI)入门指南(Swing库 JFrame 类)概述

概述 基本概念 Java Swing 的架构 Java Swing 是一个为 Java 设计的 GUI 工具包,是 JAVA 基础类的一部分,基于 Java AWT 构建,提供了一系列轻量级、可定制的图形用户界面(GUI)组件。 与 AWT 相比,Swing 提供了许多比 AWT 更好的屏幕显示元素,更加灵活和可定制,具有更好的跨平台性能。 组件和容器 Java Swing 提供了许多

顺序表之创建,判满,插入,输出

文章目录 🍊自我介绍🍊创建一个空的顺序表,为结构体在堆区分配空间🍊插入数据🍊输出数据🍊判断顺序表是否满了,满了返回值1,否则返回0🍊main函数 你的点赞评论就是对博主最大的鼓励 当然喜欢的小伙伴可以:点赞+关注+评论+收藏(一键四连)哦~ 🍊自我介绍   Hello,大家好,我是小珑也要变强(也是小珑),我是易编程·终身成长社群的一名“创始团队·嘉宾”

Maven创建项目中的groupId, artifactId, 和 version的意思

文章目录 groupIdartifactIdversionname groupId 定义:groupId 是 Maven 项目坐标的第一个部分,它通常表示项目的组织或公司的域名反转写法。例如,如果你为公司 example.com 开发软件,groupId 可能是 com.example。作用:groupId 被用来组织和分组相关的 Maven artifacts,这样可以避免

批处理以当前时间为文件名创建文件

批处理以当前时间为文件名创建文件 批处理创建空文件 有时候,需要创建以当前时间命名的文件,手动输入当然可以,但是有更省心的方法吗? 假设我是 windows 操作系统,打开命令行。 输入以下命令试试: echo %date:~0,4%_%date:~5,2%_%date:~8,2%_%time:~0,2%_%time:~3,2%_%time:~6,2% 输出类似: 2019_06

ORACLE 11g 创建数据库时 Enterprise Manager配置失败的解决办法 无法打开OEM的解决办法

在win7 64位系统下安装oracle11g,在使用Database configuration Assistant创建数据库时,在创建到85%的时候报错,错误如下: 解决办法: 在listener.ora中增加对BlueAeri-PC或ip地址的侦听,具体步骤如下: 1.启动Net Manager,在“监听程序”--Listener下添加一个地址,主机名写计

PHP7扩展开发之类的创建

本篇文章主要将如何在扩展中创建一个对象。创建的对象的过程,其实和一个小孩出生,成长的过程有些类似。 第一步,办准生证 生孩子第一步,先办准生证。声明我要生孩子了。对象创建的时候,如何办准生证呢?只要定义一个zend_class_entry变量即可。代码如下: zend_class_entry ce; zend_class_entry 是啥?可以认为它使一个原型,定义了一些对象应该有哪些东西

创建表时添加约束

查询表中的约束信息: SHOW KEYS FROM 表名; 示例: 创建depts表包含department_id该列为主键自动增长,department_name列不允许重复,location_id列不允许有空值。 create table depts(department_id int primary key auto_increment,department_name varcha

UML- 统一建模语言(Unified Modeling Language)创建项目的序列图及类图

陈科肇 ============= 1.主要模型 在UML系统开发中有三个主要的模型: 功能模型:从用户的角度展示系统的功能,包括用例图。 对象模型:采用对象、属性、操作、关联等概念展示系统的结构和基础,包括类图、对象图、包图。 动态模型:展现系统的内部行为。 包括序列图、活动图、状态图。 因为要创建个人空间项目并不是一个很大的项目,我这里只须关注两种图的创建就可以了,而在开始创建UML图