Java開源工具在linux上的源碼分析(六):符號表的讀取
通常我們使用jmap,jstack 去檢查堆棧信息的時候,是不會使用-f參數(shù)的,但有的時候系統(tǒng)在無法打印出堆棧信息的時候,會建議你使用參數(shù)-F。
關(guān)于-F參數(shù)與非-F參數(shù)的區(qū)別筆者已經(jīng)在前面的博客中講述(http://blog.csdn.net/raintungli/article/details/7023092),簡單的說也就是一種是讓jvm進(jìn)程自己打印出堆棧信息,另有一種是直接訪問jvm的堆棧區(qū)通過固定的結(jié)構(gòu)找出我們需要的信息。
1. Linux-F參數(shù)的實(shí)現(xiàn)
在linux中可以使用ptrace的系統(tǒng)調(diào)用去訪問運(yùn)行中的進(jìn)程的內(nèi)存信息,具體如何實(shí)現(xiàn)可以參考筆者的博客(http://blog.csdn.net/raintungli/article/details/6563867)
在java中使用動態(tài)加載的方式加載jvm自己的鏈接共享庫,jvm的核心鏈接共享庫是libjvm.so,linux中如何動態(tài)加載可以參考(http://www.ibm.com/developerworks/cn/linux/l-dynamic-libraries/#dynamiclinking)
因?yàn)槭莿討B(tài)共享庫,當(dāng)想查找具體的參數(shù)的值,內(nèi)存的信息的時候,就需要計(jì)算出正確的參數(shù)或者函數(shù)的地址。
2. 共享庫中的符號相對地址偏移
可運(yùn)行程序,共享庫使用ELF格式,當(dāng)運(yùn)行一個程序的時候,內(nèi)核會把ELF加載到用戶空間,里面記錄了程序的函數(shù)和數(shù)據(jù)的地址和內(nèi)容,elf文件格式就不具體描述了。
在linux 中可以使用結(jié)構(gòu)體ELF_EHDR,ELF_PHDR,ELF_SHDR讀出elf 的program header, section header, section data.
在jvm中源碼具體實(shí)現(xiàn)請參考 /hotspot/agent/os/linux/salibelf.c
在linux中本身就自帶一個讀取elf格式的工具,readelf 你可以使用不同的參數(shù)讀取不同的內(nèi)容。
- readelf -s libjvm.so
顯示共享庫中的方法參數(shù)的虛擬地址,類型,名字
- readelf -l libjvm.so
讀取program headers,其中出現(xiàn)2個LOAD的類型,***個是程序的指令虛擬的起始地址,另一個是程序數(shù)據(jù)的起始地址。
通過2個地址我們就能找到共享庫中的參數(shù),函數(shù)的相對地址的偏移。
3. 進(jìn)程中的符號地址
在第二章節(jié)中,得到的只是相對的地址偏移,并不是真實(shí)運(yùn)行中的進(jìn)程的符號地址,如何得到真實(shí)的地址在linux中就相對比較簡單。
- cat /proc/$processid/maps
在maps里詳細(xì)記錄了進(jìn)程的堆棧分配的地址,包括共享庫的地址,那么起始地址就是這個庫分配的最小地址
- 進(jìn)程中共享庫分配的最小地址+相對地址的偏移 =真實(shí)的進(jìn)程中該函數(shù)或變量的真實(shí)地址
4. Java tool 保存的符號表
在jmap/jstack 中,為了提高讀取符號地址的性能,避免每一次要找符號的地址從elf文件中查找,只是在初始話的時候?qū)⒎柋肀4娉晒1恚渲衚ey是符號的名字,內(nèi)容是符號的地址,長度。
具體實(shí)現(xiàn)可以參考 /hotspot/src/os/linux/symtab.c build_symtab_internal 函數(shù)。
原文鏈接:http://blog.csdn.net/raintungli/article/details/7289639
【系列文章】























