linux内核源码分析之proc文件系统(三)

目录

一、数据结构分析

1、proc数据项的表示proc_dir_entry

2、proc inode

二、proc初始化

三、管理/proc数据项

1. 数据项的创建和注册

2. 查找proc数据项

三、读取和写入信息

四、进程相关的信息

1、创建目录inode

2、处理文件


一、数据结构分析

1、proc数据项的表示proc_dir_entry

proc文件系统中的每个数据项都由proc_dir_entry 的一个实例描述
struct proc_dir_entry {
	spinlock_t pde_unload_lock;
	struct completion *pde_unload_completion;
	const struct inode_operations *proc_iops;
	union {
		const struct proc_ops *proc_ops;
		const struct file_operations *proc_dir_ops;
	};
	const struct dentry_operations *proc_dops;
	proc_write_t write;
...
}

实例

struct proc_dir_entry proc_root = {
	.low_ino	= PROC_ROOT_INO, 
	.namelen	= 5, 
	.mode		= S_IFDIR | S_IRUGO | S_IXUGO, 
	.nlink		= 2, 
	.refcnt		= REFCOUNT_INIT(1),
	.proc_iops	= &proc_root_inode_operations, //对proc根目录几乎不能做什么
	.proc_dir_ops	= &proc_root_operations,//proc目录是特别的,因为其中包含了<pid>目录,因而不能对该目录使用通用处理函数
	.parent		= &proc_root,
	.subdir		= RB_ROOT,
	.name		= "/proc",
};

2、proc inode

内核提供了一个数据结构,称之为 proc_inode ,支持以面向 inode 的方式来查看 proc 文件系统的
数据项。
//proc的数据与VFS层的inode数据关联起来,
struct proc_inode {
	struct pid *pid;
	unsigned int fd;
	union proc_op op;
	struct proc_dir_entry *pde;//pde是一个指针,指向关联到proc数据项的proc_dir_entry实例
	struct ctl_table_header *sysctl;
	struct ctl_table *sysctl_entry;
	struct hlist_node sysctl_inodes;
	const struct proc_ns_operations *ns_ops;
	struct inode vfs_inode;
} __randomize_layout;

两个结构体之间的关系

proc_inode中的成员pde是一个指针,指向关联到proc数据项的proc_dir_entry实例

二、proc初始化

初始化,装载文件系统

三、管理/proc数据项

proc 文件系统投入使用之前,必须向其中添加数据项。内核提供了几个辅助例程来添加文件、 创建目录

1. 数据项的创建和注册

内核中有相关的宏定义
#define proc_create_seq_private(name, mode, parent, ops, size, data) ({NULL;})
#define proc_create_seq_data(name, mode, parent, ops, data) ({NULL;})
#define proc_create_seq(name, mode, parent, ops) ({NULL;})
#define proc_create_single(name, mode, parent, show) ({NULL;})
#define proc_create_single_data(name, mode, parent, show, data) ({NULL;})
#define proc_create(name, mode, parent, proc_ops) ({NULL;})
#define proc_create_data(name, mode, parent, proc_ops, data) ({NULL;})

在创建了数据项之后,绑定相关的操作函数
文件相关的 :proc_ops,  proc_file_inode_operations
目录相关的 :proc_dir_operations,proc_dir_inode_operations
连接相关的 :proc_link_inode_operations
使用 proc_register 将其注册到 proc 文件系统,5.6.18 内核先绑定相关的操作函数如
struct proc_dir_entry *proc_create_data(const char *name, umode_t mode,
		struct proc_dir_entry *parent,
		const struct proc_ops *proc_ops, void *data)
{
	struct proc_dir_entry *p;

	p = proc_create_reg(name, mode, &parent, data);
	if (!p)
		return NULL;
	p->proc_ops = proc_ops;
	return proc_register(parent, p);
}

2. 查找proc数据项

分为进程pid相关的查找,和普通查找
static const struct inode_operations proc_root_inode_operations = {
	.lookup		= proc_root_lookup,
	.getattr	= proc_root_getattr,
};


static struct dentry *proc_root_lookup(struct inode * dir, struct dentry * dentry, unsigned int flags)
{
	if (!proc_pid_lookup(dentry, flags))//PID
		return NULL;

	return proc_lookup(dir, dentry, flags);
}

三、读取和写入信息

使用proc_ops, 而不是file_operations

struct proc_ops {
	int	(*proc_open)(struct inode *, struct file *);
	ssize_t	(*proc_read)(struct file *, char __user *, size_t, loff_t *);
	ssize_t	(*proc_write)(struct file *, const char __user *, size_t, loff_t *);
	loff_t	(*proc_lseek)(struct file *, loff_t, int);
	int	(*proc_release)(struct inode *, struct file *);
	__poll_t (*proc_poll)(struct file *, struct poll_table_struct *);
	long	(*proc_ioctl)(struct file *, unsigned int, unsigned long);
#ifdef CONFIG_COMPAT
	long	(*proc_compat_ioctl)(struct file *, unsigned int, unsigned long);
#endif
	int	(*proc_mmap)(struct file *, struct vm_area_struct *);
	unsigned long (*proc_get_unmapped_area)(struct file *, unsigned long, unsigned long, unsigned long, unsigned long);
};

这里以读取CPU信息为例

extern const struct seq_operations cpuinfo_op;
static int cpuinfo_open(struct inode *inode, struct file *file)
{
	arch_freq_prepare_all();
	return seq_open(file, &cpuinfo_op);
}

//读写相关的操作函数 
static const struct proc_ops cpuinfo_proc_ops = {
	.proc_open	= cpuinfo_open,
	.proc_read	= seq_read,
	.proc_lseek	= seq_lseek,
	.proc_release	= seq_release,
};

//创建cpuinfo文件
static int __init proc_cpuinfo_init(void)
{
	proc_create("cpuinfo", 0, NULL, &cpuinfo_proc_ops);
	return 0;
}
fs_initcall(proc_cpuinfo_init);

四、进程相关的信息

1、创建目录inode

//查看进程信息
struct dentry *proc_pid_lookup(struct dentry *dentry, unsigned int flags)
{
	struct task_struct *task;
	unsigned tgid;
	struct pid_namespace *ns;
	struct dentry *result = ERR_PTR(-ENOENT);

	tgid = name_to_int(&dentry->d_name);
	if (tgid == ~0U)
		goto out;

	ns = dentry->d_sb->s_fs_info;
	rcu_read_lock();
	task = find_task_by_pid_ns(tgid, ns);
	if (task)
		get_task_struct(task);
	rcu_read_unlock();
	if (!task)
		goto out;

	result = proc_pid_instantiate(dentry, task, NULL);
	put_task_struct(task);
out:
	return result;
}

2、处理文件

因为特定于 PID 的目录,其内容总是同样的,内核源代码中定义了所有文件的静态列表以及其他一些信息,内核定义了静态参数.
static const struct pid_entry tgid_base_stuff[] = {
	REG("cmdline",    S_IRUGO, proc_pid_cmdline_ops),
	ONE("stat",       S_IRUGO, proc_tgid_stat),
	ONE("statm",      S_IRUGO, proc_pid_statm),
	REG("maps",       S_IRUGO, proc_pid_maps_operations),
...
}


参考:

 内核源码

《深入理解linux内核》
《Linux内核深度解析》

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

为了维护世界和平_

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值