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了


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM