linux 獲得當前進程 對應的可執行檔的 絕對路徑

來源:互聯網
上載者:User

 

 先說在應用程式層怎麼搞這個事,很簡單:

 #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 >=0x020616p->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了

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.