背景
在Linux内核中,已知一个进程的pid和其打开文件的文件描述符fd,如何获取该文件的绝对路径?基本思路是先获取该文件在内核中的file结构体,再通过d_path()获取到整个文件的绝对路径。
方法一
如果理解了进程和文件系统数据结构之间的关系,那么这种方法可以采用。基本的方法如下:
1.通过进程pid获取进程描述符task_struct;
2.通过task_struct获取该进程打开文件结构files_struct,从而获取文件描述符表;
3.以fd为索引在文件描述符表中获取对应文件的结构体file;
4.通过file获取对应path结构,该结构封装当前文件对应的dentry和挂载点;
5.通过内核函数d_path()获取该文件的绝对路径;
通过进程pid获取进程描述符demo:
struct task_struct *get_proc(pid_t pid) { struct pid *pid_struct = NULL; struct task_struct *mytask = NULL; pid_struct = find_get_pid(pid); if (!pid_struct) return NULL; mytask = pid_task(pid_struct, PIDTYPE_PID); return mytask; }
通过fd以及d_path()获取绝对路径demo:
int get_path(struct task_struct *mytask, int fd) { struct file *myfile = NULL; struct files_struct *files = NULL; char path[100] = {'\0'}; char *ppath = path; files = mytask->files; if (!files) { printk("files is null..\n"); return -1; } myfile = files->fdt->fd[fd]; if (!myfile) { printk("myfile is null..\n"); return -1; } ppath = d_path(&(myfile->f_path), ppath, 100); printk("path:%s\n", ppath); return 0; }
从上面的代码可以看出,从fd到file结构的获取均通过各个数据结构之间的指向关系获取。
方法二
与方法一的思路相同,但是可以直接使用内核提供的函数fget()进行fd到file的获取。这种方法使用比较简单,程序更加安全,不过就是少了对数据结构关系的思考过程。其实也可以将fget()函数的实现过程作为参考,欣赏内核中代码实现的严谨性。