module builtin // print_backtrace_skipping_top_frames prints the backtrace skipping N top frames. pub fn print_backtrace_skipping_top_frames(xskipframes int) bool { $if no_backtrace ? { return false } $else { skipframes := xskipframes + 2 $if macos || freebsd || openbsd || netbsd { return print_backtrace_skipping_top_frames_bsd(skipframes) } $else $if linux { return print_backtrace_skipping_top_frames_linux(skipframes) } $else { println('print_backtrace_skipping_top_frames is not implemented. skipframes: ${skipframes}') } } return false } // the functions below are not called outside this file, // so there is no need to have their twins in builtin_windows.v @[direct_array_access] fn print_backtrace_skipping_top_frames_bsd(skipframes int) bool { $if no_backtrace ? { return false } $else { $if macos || freebsd || netbsd { buffer := [100]voidptr{} nr_ptrs := C.backtrace(&buffer[0], 100) if nr_ptrs < 2 { eprintln('C.backtrace returned less than 2 frames') return false } C.backtrace_symbols_fd(&buffer[skipframes], nr_ptrs - skipframes, 2) } return true } } fn C.tcc_backtrace(fmt &char) int @[direct_array_access] fn print_backtrace_skipping_top_frames_linux(skipframes int) bool { $if android { eprintln('On Android no backtrace is available.') return false } $if !glibc { eprintln('backtrace_symbols is missing => printing backtraces is not available.') eprintln('Some libc implementations like musl simply do not provide it.') return false } $if native { eprintln('native backend does not support backtraces yet.') return false } $else $if no_backtrace ? { return false } $else { $if linux && !freestanding { $if tinyc { C.tcc_backtrace(c'Backtrace') return false } $else { buffer := [100]voidptr{} nr_ptrs := C.backtrace(&buffer[0], 100) if nr_ptrs < 2 { eprintln('C.backtrace returned less than 2 frames') return false } nr_actual_frames := nr_ptrs - skipframes //////csymbols := backtrace_symbols(*voidptr(&buffer[skipframes]), nr_actual_frames) csymbols := C.backtrace_symbols(voidptr(&buffer[skipframes]), nr_actual_frames) for i in 0 .. nr_actual_frames { sframe := unsafe { tos2(&u8(csymbols[i])) } executable := sframe.all_before('(') addr := sframe.all_after('[').all_before(']') beforeaddr := sframe.all_before('[') cmd := 'addr2line -e ' + executable + ' ' + addr // taken from os, to avoid depending on the os module inside builtin.v f := C.popen(&char(cmd.str), c'r') if f == unsafe { nil } { eprintln(sframe) continue } buf := [1000]u8{} mut output := '' unsafe { bp := &buf[0] for C.fgets(&char(bp), 1000, f) != 0 { output += tos(bp, vstrlen(bp)) } } output = output.trim_chars(' \t\n', .trim_both) + ':' if C.pclose(f) != 0 { eprintln(sframe) continue } if output in ['??:0:', '??:?:'] { output = '' } // See http://wiki.dwarfstd.org/index.php?title=Path_Discriminators // Note: it is shortened here to just d. , just so that it fits, and so // that the common error file:lineno: line format is enforced. output = output.replace(' (discriminator', ': (d.') eprint(output) eprint_space_padding(output, 55) eprint(' | ') eprint(addr) eprint(' | ') eprintln(beforeaddr) } if nr_actual_frames > 0 { unsafe { C.free(csymbols) } } } } } return true }