-
-
Notifications
You must be signed in to change notification settings - Fork 221
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Explore linux behaviour when grabbing stack.
- Loading branch information
Showing
7 changed files
with
240 additions
and
6 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,210 @@ | ||
module std::os::linux @if(env::LINUX); | ||
import libc; | ||
import std::os::posix; | ||
import std::io; | ||
import std::collections::list; | ||
|
||
extern fn isz readlink(ZString path, char* buf, usz bufsize); | ||
|
||
def BacktraceList = List(<Backtrace>); | ||
|
||
const PT_PHDR = 6; | ||
const EI_NIDENT = 16; | ||
def Elf32_Half = ushort; | ||
def Elf32_Word = uint; | ||
def Elf32_Addr = uint; | ||
def Elf32_Off = uint; | ||
|
||
struct Elf32_Ehdr | ||
{ | ||
char[EI_NIDENT] e_ident; | ||
Elf32_Half e_type; | ||
Elf32_Half e_machine; | ||
Elf32_Word e_version; | ||
Elf32_Addr e_entry; | ||
Elf32_Off e_phoff; | ||
Elf32_Off e_shoff; | ||
Elf32_Word e_flags; | ||
Elf32_Half e_ehsize; | ||
Elf32_Half e_phentsize; | ||
Elf32_Half e_phnum; | ||
Elf32_Half e_shentsize; | ||
Elf32_Half e_shnum; | ||
Elf32_Half e_shstrndx; | ||
} | ||
|
||
struct Elf32_Phdr | ||
{ | ||
Elf32_Word p_type; | ||
Elf32_Off p_offset; | ||
Elf32_Addr p_vaddr; | ||
Elf32_Addr p_paddr; | ||
Elf32_Word p_filesz; | ||
Elf32_Word p_memsz; | ||
Elf32_Word p_flags; | ||
Elf32_Word p_align; | ||
} | ||
|
||
def Elf64_Addr = ulong; | ||
def Elf64_Half = ushort; | ||
def Elf64_Off = ulong; | ||
def Elf64_Word = uint; | ||
def Elf64_Sword = int; | ||
def Elf64_Sxword = long; | ||
def Elf64_Lword = ulong; | ||
def Elf64_Xword = ulong; | ||
|
||
struct Elf64_Ehdr | ||
{ | ||
char[EI_NIDENT] e_ident; | ||
Elf64_Half e_type; | ||
Elf64_Half e_machine; | ||
Elf64_Word e_version; | ||
Elf64_Addr e_entry; | ||
Elf64_Off e_phoff; | ||
Elf64_Off e_shoff; | ||
Elf64_Word e_flags; | ||
Elf64_Half e_ehsize; | ||
Elf64_Half e_phentsize; | ||
Elf64_Half e_phnum; | ||
Elf64_Half e_shentsize; | ||
Elf64_Half e_shnum; | ||
Elf64_Half e_shstrndx; | ||
} | ||
|
||
struct Elf64_Phdr | ||
{ | ||
Elf64_Word p_type; | ||
Elf64_Word p_flags; | ||
Elf64_Off p_offset; | ||
Elf64_Addr p_vaddr; | ||
Elf64_Addr p_paddr; | ||
Elf64_Xword p_filesz; | ||
Elf64_Xword p_memsz; | ||
Elf64_Xword p_align; | ||
} | ||
|
||
extern fn CInt dladdr(void* addr, Linux_Dl_info* info); | ||
|
||
struct Linux_Dl_info | ||
{ | ||
ZString dli_fname; /* Pathname of shared object */ | ||
void* dli_fbase; /* Base address of shared object */ | ||
ZString dli_sname; /* Name of nearest symbol */ | ||
void* dli_saddr; /* Address of nearest symbol */ | ||
} | ||
|
||
fn bool! resolve_addr(void* addr, ZString* obj_path, ZString* name, void** obj_address) | ||
{ | ||
Linux_Dl_info info; | ||
io::printfn("Trying to resolve %p", addr); | ||
if (dladdr(addr, &info) == 0) return false; | ||
io::printn("Success"); | ||
*obj_path = info.dli_fname; | ||
*obj_address = addr - (uptr)info.dli_fbase + elf_module_image_base(info.dli_fname.str_view())!; | ||
*name = info.dli_sname; | ||
return true; | ||
} | ||
|
||
fn uptr! elf_module_image_base(String path) @local | ||
{ | ||
File file = file::open(path, "rb")!; | ||
char[4] buffer; | ||
io::read_all(&file, &buffer)!; | ||
if (buffer != char[4]{ 0x7f, 'E', 'L', 'F'}) return BacktraceFault.IMAGE_NOT_FOUND?; | ||
bool is_64 = file.read_byte()! == 2; | ||
bool is_little_endian = file.read_byte()! == 1; | ||
// Actually, not supported. | ||
if (!is_little_endian) return BacktraceFault.IMAGE_NOT_FOUND?; | ||
file.seek(0)!; | ||
if (is_64) | ||
{ | ||
Elf64_Ehdr file_header; | ||
io::read_any(&file, &file_header)!; | ||
if (file_header.e_ehsize != Elf64_Ehdr.sizeof) return BacktraceFault.IMAGE_NOT_FOUND?; | ||
for (isz i = 0; i < file_header.e_phnum; i++) | ||
{ | ||
Elf64_Phdr header; | ||
file.seek(file_header.e_phoff + file_header.e_phentsize * i)!; | ||
io::read_any(&file, &header)!; | ||
if (header.p_type == PT_PHDR) return header.p_vaddr - header.p_offset; | ||
} | ||
return 0; | ||
} | ||
Elf32_Ehdr file_header; | ||
io::read_any(&file, &file_header)!; | ||
if (file_header.e_ehsize != Elf32_Ehdr.sizeof) return BacktraceFault.IMAGE_NOT_FOUND?; | ||
for (isz i = 0; i < file_header.e_phnum; i++) | ||
{ | ||
Elf32_Phdr header; | ||
file.seek(file_header.e_phoff + file_header.e_phentsize * i)!; | ||
io::read_any(&file, &header)!; | ||
if (header.p_type == PT_PHDR) return (ulong)header.p_vaddr - header.p_offset; | ||
} | ||
return 0; | ||
} | ||
|
||
fn BacktraceList! backtrace_load(Allocator* allocator) | ||
{ | ||
void*[256] bt_buffer; | ||
CInt size = posix::backtrace(&bt_buffer, 256); | ||
ZString* symbols = posix::backtrace_symbols(&bt_buffer, size); | ||
BacktraceList list; | ||
list.init_new(size, allocator); | ||
defer catch | ||
{ | ||
foreach (trace : list) | ||
{ | ||
trace.free(); | ||
} | ||
list.free(); | ||
} | ||
bool has_addr = true; | ||
@pool() | ||
{ | ||
char[] buf = mem::temp_array(char, 1024); | ||
while (size > 0) | ||
{ | ||
String symbol = symbols[size - 1].str_view(); | ||
io::printn(symbol); | ||
size--; | ||
String file; | ||
String! hex = {| | ||
usz index = symbol.index_of("(")!; | ||
file = symbol[:index]; | ||
String after = symbol[index + 1..]; | ||
after = after[after.index_of("[0x")! + 1..]; | ||
return after[:after.index_of("]")!]; | ||
|}; | ||
if (try hex) | ||
{ | ||
ZString obj_path; | ||
ZString name; | ||
void* obj_address; | ||
if (resolve_addr(bt_buffer[size], &obj_path, &name, &obj_address)!) | ||
{ | ||
io::printfn("Got %s %s %s", obj_path, name, obj_address); | ||
hex = string::tformat("0x%x", obj_address); | ||
} | ||
else | ||
{ | ||
io::printfn("Resolve failed"); | ||
} | ||
(void)io::printfn("Image module offset %s", elf_module_image_base(file)); | ||
io::printfn("%s %s %p", hex, file, bt_buffer[size]); | ||
String! s = process::execute_stdout_to_buffer(buf, { "addr2line", "-p", "-i", "-C", "-f", "-e", file, hex }); | ||
if (try s) | ||
{ | ||
io::printn(s); | ||
continue; | ||
} | ||
has_addr = false; | ||
} | ||
else | ||
{ | ||
io::printn(symbol); | ||
} | ||
} | ||
}; | ||
return list; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,10 @@ | ||
module test; | ||
import std::io; | ||
import std::collections::map; | ||
import std::os; | ||
|
||
fn void main() | ||
{ | ||
int x = 2; | ||
(void)linux::backtrace_load(mem::heap()); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters