• linux 获得当前进程 对应的可执行文件的 绝对路径


     先说在应用层怎么搞这个事,很简单:

     #include <stdio.h>

    #include <unistd.h>

    int main() {
            char link[100], path[100];
            sprintf(link, "/proc/%d/exe", getpid());/////////////
            readlink(link, path, sizeof(path));//////////////
            printf("%s/n", path);
            return 0;
    }

    或者 

     #include <stdio.h>

    #include <unistd.h>
    char * get_exe_path( char * buf, int count)
    {
        int i;
        int rslt = readlink("/proc/self/exe", buf, count - 1);/////注意这里使用的是self
        if (rslt < 0 || (rslt >= count - 1))
        {
            return NULL;
        }
        buf[rslt] = '\0';
        for (i = rslt; i >= 0; i--)
        {
            printf("buf[%d] %c\n", i, buf[i]);
            if (buf[i] == '/')
            {
                buf[i + 1] = '\0';
                break;
            }
        }
        return buf;
    }

    int main(int argc, char ** argv)
    {
        char path[1024];
        printf("%s\n", get_exe_path(path, 1024));
        return 0;
    }

     在内核层,当前进程的所有信息都包含在current中。current里边有几个相关的变量:

     

    1、current->comm:是一个16字节大小的char数组,记录的是当前进程的段路径。就是说,如果当前进程的可执行文件的全路径是/home/yaog/test,那么comm的内容就是“test”,但是由comm是不能得到全路径的。

     

    2、current->mm/current->active_mm:这两个变量时mm_struct结构的,mm_struct用于描述虚拟内存。而current->mm

    /current->active_mm就是用于描述当前进程所在的页面信息的。这个看上去有点儿靠谱。果然顺藤摸瓜:

     #if LINUX_VERSION_CODE >= 0x020616

    p->mm->mmap->vm_file->f_path.dentry->d_name.name
    #else
    p->mm->mmap->vm_file->f_dentry->d_name.name
    #endif

     

    就可以得到“/home/yaog/test”中的“test”,再根据

     #if LINUX_VERSION_CODE >=0x020616
    p->mm->mmap->vm_file->f_path.dentry->d_parent
    #else
    p->mm->mmap->vm_file->f_dentry->d_parent
    #endif

    得到父亲节点,就是“/home/yaog/test”中的“yaog”,这样依次往上找。一直找到“/”为止。

    这还不算完:这个可执行文件可能是被挂载上去的,还要得到它的挂载点信息:

    #if LINUX_VERSION_CODE >= 0x020616
    current->mm->mmap->vm_file->f_path.mnt->mnt_mountpoint->d_name.name
    #else
    current->mm->mmap->vm_file->f_vfsmnt->mnt_mountpoint->d_name.name
    #endif

    然后把两截给拼起来。

    比如说:可执行文件在其文件系统中的路径是“/yaog/test”,而这个文件系统又挂载在系统的“/home”节点上。

    那么这个可执行文件的路径就是由“/home”和“/yaog/test”拼起来-->“/home/yaog/test”。

     

    3、很不幸的是,上面的方法有问题。在有些linux系统上工作的很好,但是在有些系统中,按照上述方法,得到的是“/lib/XXXX.so”之类的路径,而不是我们想要的“/home/yaog/test”。(/home/yaog/test调用了这个so)

     

    最后的解决方案又绕回到应用层的思路:

    filp = filp_open("/proc/(current->pid)/exe",0,0)  

    得到这个filp之后,再按照2中的方法找一遍,就OK了

  • 相关阅读:
    九九乘法表
    获取1-10 的和,并打印 2.1-100 之间 7 的倍数的个数,并打印
    1-100之间 7 的倍数的个数,并打印
    无限循环的最简单的表现形式
    for 和 while 区别
    do while 和 while 区别
    if 和 switch 比较
    星期和月份
    【运筹优化】求职与工作思考 -求职需要的技能(一)
    python 在线笔试的OJ输入输出总结
  • 原文地址:https://www.cnblogs.com/xuxm2007/p/2437954.html
Copyright © 2020-2023  润新知