您好,登錄后才能下訂單哦!
這篇文章給大家分享的是有關用php讀取elf結構的方法的內容。小編覺得挺實用的,因此分享給大家做個參考。一起跟隨小編過來看看吧。
UNIX系統的可執行文件都采用ELF格式,類型分為目標文件、可執行文件和共享庫
ELF格式探析之三:sections
本例基于64位的小端序linux機器
#include <stdio.h> void say_hello(char *who) { printf("hello, %s!\n", who); } char *my_name = "wb"; int man() { say_hello(my_name); return 0; } // 執行gcc -c hello.c生成hello.o
ELF header,位于文件的0~64字節,存儲文件的描述信息,Section header table的起始位置
N個Section
Section header table,每個條目64字節,對應一個Section的信息
Program header table,可執行文件需要,本例的hello.o沒有
首先用readelf命令讀取elf信息:readelf -h hello.o。總結如下:
ELF header占用64字節
N個Section占用6488-64-1472=4952字節
Section header table占用23*64=1472字節
readelf -h hello.o ELF Header: Class: ELF64 Data: 2's complement, little endian OS/ABI: UNIX - System V Type: REL (Relocatable file) Machine: Advanced Micro Devices X86-64 Start of program headers: 0 (bytes into file) Start of section headers: 5016 (bytes into file) //Section header table的起始位置 Size of this header: 64 (bytes) //ELF header的占用大小 Size of program headers: 0 (bytes) //hello.o沒有program header table Number of program headers: 0 //hello.o沒有program header table Size of section headers: 64 (bytes) //Section header table每個條目占用大小 Number of section headers: 23 //Section header table條目個數 Section header string table index: 22 //.shstrtab Section位于Section header table第22個條目
.shstrtab Section其實是存儲的所有Section的名字
<?php $fp = fopen("hello.o", "rb"); fseek($fp, 40, SEEK_SET); $sh_off = fread($fp, 8); $sh_off = unpack("P", $sh_off); var_dump("section header offset in file: ". $sh_off[1]); fseek($fp, 10, SEEK_CUR); $sh_ent_size = fread($fp, 2); $sh_ent_size = unpack("v", $sh_ent_size); var_dump("section header entry size: ". $sh_ent_size[1]); $sh_num = fread($fp, 2); $sh_num = unpack("v", $sh_num); var_dump("section header number: ". $sh_num[1]); $sh_strtab_index = fread($fp, 2); $sh_strtab_index = unpack("v", $sh_strtab_index); var_dump("section header string table index: ". $sh_strtab_index[1]); fseek($fp, $sh_off[1] + $sh_strtab_index[1] * $sh_ent_size[1], SEEK_SET); fseek($fp, 24, SEEK_CUR); //sh_name(4) + sh_type(4) + sh_flags(8) + sh_addr(8) = 24 $str_table_off = fread($fp, 8); $str_table_off = unpack("P", $str_table_off); var_dump("section name string table offset: ". $str_table_off[1]); $str_table_size = fread($fp, 8); $str_table_size = unpack("P", $str_table_size); var_dump("section name string table size: ". $str_table_size[1]); fseek($fp, $str_table_off[1], SEEK_SET); $str = fread($fp, $str_table_size[1]); print_r(explode("\x00", trim($str, "\x00"))); // 讀取所有Section條目信息 for ($i =0;$i < $sh_num[1]; $i ++) { fseek($fp, $sh_off[1] + $i * $sh_ent_size[1], SEEK_SET); $sh_name = fread($fp, 4); $sh_name = unpack("V", $sh_name); fseek($fp, 20, SEEK_CUR); //sh_type(4) + sh_flags(8) + sh_addr(8) = 20 $sh_offset = fread($fp, 8); $sh_offset = unpack("P", $sh_offset); $sh_size = fread($fp, 8); $sh_size = unpack("P", $sh_size); printf("section: %2s name: %-24s offset: %12s size: %12s\n", $i, get_section_name($sh_name[1]), $sh_offset[1], $sh_size[1]); } function get_section_name($start) { global $str; $name = substr($str, $start); return strstr($name, "\x00", true); }
感謝各位的閱讀!關于用php讀取elf結構的方法就分享到這里了,希望以上內容可以對大家有一定的幫助,讓大家可以學到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到吧!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。