一,task_struct和文件系统相关的一些信息
<sched.h>
struct task_struct {
...
/* file system info */
int link_count, total_link_count;
...
/* filesystem information */
struct fs_struct *fs;
/* open file information */
struct files_struct *files;
/* namespaces */
struct nsproxy *nsproxy;
...
}
进程的文件系统相关的数据保存在fs中,这些数据包含当前的工作目录
fs_struct主要用于管理特定进程本身的一些信息,
struct fs_struct {
atomic_t count;
int umask; //表示标准的掩码,用于设置新文件的权限
struct dentry * root, * pwd, * altroot; //root表示该进程所在的根目录,pwd表示当前进程所在的目录,shell的cd命令使用时会改变
struct vfsmount * rootmnt, * pwdmnt, * altrootmnt; //rootmnt表示根目录所在的文件系统,
};
struct files_struct {
atomic_t count; //共享该文件的进程的数目
struct fdtable *fdt;
struct fdtable fdtab;
int next_fd; //该进程下一次打开新文件的时候使用的文件描述符id
struct embedded_fd_set close_on_exec_init;
struct embedded_fd_set open_fds_init;
struct file * fd_array[NR_OPEN_DEFAULT]; 每个成员都是一个指针,指向每个打开文件struct file实例
};
默认情况下内核运行每个进程打开NR_OPEN_DEFAULT个文件,默认值是BITS_PER_LONG,32位系统上该值为32
struct fdtable {
unsigned int max_fds; //该进程当前可以处理的文件对象和文件描述符的最大数目
struct file ** fd; /* current fd array */
fd_set *close_on_exec;
fd_set *open_fds;
struct rcu_head rcu;
struct files_struct *free_files;
struct fdtable *next;
};
struct file指向了真正文件的信息,
struct file {
struct list_head fu_list;
struct path f_path; //指定了文件名和inode之间的关联,文件所在文件系统相关信息
#define f_dentry f_path.dentry
#define f_vfsmnt f_path.mnt
const struct file_operations *f_op; //文件操作所调用到的各个函数
atomic_t f_count;
unsigned int f_flags;
mode_t f_mode;
loff_t f_pos; //表示进程对文件操作的位置
struct fown_struct f_owner; //处理该文件的进程有关信息
unsigned int f_uid, f_gid; //用户的UID GID
struct file_ra_state f_ra; //预读取特征,指定在实际读取数据之前是否预读取
unsigned long f_version;
...
struct address_space *f_mapping; //指向属于文件相关的inode实例的地址空间映射
...
};
struct path {
struct vfsmount *mnt;
struct dentry *dentry;
};
在硬盘上并不存在一个文件结构,进程打开一个文件,内核就动态创建一个文件对象,同一个文件在不同的进程中有不同的文件对象
附录:
进程打开一个文件的过程:
1,用户层的open()函数最终调用了内核里面的sys_open()函数(fs/open.c),sys_open()函数主要借助do_flip_open()函数来完成查找文件的inode,do_flip_open()函数首先调用open_namei()函数,open_namei()函数调用path_lookup()函数查找文件的inode并执行额外几个检查动作,如果是创建新的文件系统项,该函数还需要应用存储在进程umask(current->fs->umask)中的权限位的默认设置。do_flip_open()函数然后调用nameidata_to_flip()函数初始化预读结构,将新创建的file实例放置到超级快的s_files链上并调用底层文件系统的file_operations结构中的open()函数。
最终控制权转回用户进程,返回文件描述符之前,fd_install必须将file实例放置到进程task_struct的files->fd数组中