2019-11-13 09:05:06 +01:00
|
|
|
// Copyright (c) 2019 Alexander Medvednikov. All rights reserved.
|
|
|
|
// Use of this source code is governed by an MIT license
|
|
|
|
// that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
module builtin
|
|
|
|
|
|
|
|
fn print_backtrace_skipping_top_frames_msvc(skipframes int) bool {
|
|
|
|
println('not implemented, see builtin_windows.v')
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
fn print_backtrace_skipping_top_frames_mingw(skipframes int) bool {
|
|
|
|
println('not implemented, see builtin_windows.v')
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
fn print_backtrace_skipping_top_frames_nix(xskipframes int) bool {
|
|
|
|
skipframes := xskipframes + 2
|
2019-12-03 14:29:24 +01:00
|
|
|
$if macos { return print_backtrace_skipping_top_frames_mac(skipframes) }
|
2019-11-13 09:05:06 +01:00
|
|
|
$if linux { return print_backtrace_skipping_top_frames_linux(skipframes) }
|
2019-11-25 11:54:56 +01:00
|
|
|
$if freebsd { return print_backtrace_skipping_top_frames_freebsd(skipframes) }
|
2019-11-13 09:05:06 +01:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
// the functions below are not called outside this file,
|
|
|
|
// so there is no need to have their twins in builtin_windows.v
|
|
|
|
fn print_backtrace_skipping_top_frames_mac(skipframes int) bool {
|
2019-12-03 14:29:24 +01:00
|
|
|
$if macos {
|
2019-11-13 09:05:06 +01:00
|
|
|
buffer := [100]byteptr
|
|
|
|
nr_ptrs := C.backtrace(*voidptr(buffer), 100)
|
|
|
|
C.backtrace_symbols_fd(*voidptr(&buffer[skipframes]), nr_ptrs-skipframes, 1)
|
2019-11-14 22:46:40 +01:00
|
|
|
}
|
2019-11-13 09:05:06 +01:00
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2019-11-25 11:54:56 +01:00
|
|
|
fn print_backtrace_skipping_top_frames_freebsd(skipframes int) bool {
|
|
|
|
$if freebsd {
|
|
|
|
buffer := [100]byteptr
|
|
|
|
nr_ptrs := C.backtrace(*voidptr(buffer), 100)
|
|
|
|
C.backtrace_symbols_fd(*voidptr(&buffer[skipframes]), nr_ptrs-skipframes, 1)
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2019-11-13 09:05:06 +01:00
|
|
|
fn print_backtrace_skipping_top_frames_linux(skipframes int) bool {
|
|
|
|
$if tinyc {
|
|
|
|
println('TODO: print_backtrace_skipping_top_frames_linux $skipframes with tcc fails tests with "stack smashing detected" .')
|
|
|
|
return false
|
|
|
|
}
|
2019-11-25 06:38:00 +01:00
|
|
|
$if !android { // backtrace is not available on Android.
|
2019-11-13 09:05:06 +01:00
|
|
|
$if glibc {
|
|
|
|
buffer := [100]byteptr
|
|
|
|
nr_ptrs := C.backtrace(*voidptr(buffer), 100)
|
|
|
|
nr_actual_frames := nr_ptrs-skipframes
|
|
|
|
mut sframes := []string
|
|
|
|
csymbols := *byteptr(C.backtrace_symbols(*voidptr(&buffer[skipframes]), nr_actual_frames))
|
|
|
|
for i in 0..nr_actual_frames { sframes << tos2(csymbols[i]) }
|
|
|
|
for sframe in sframes {
|
|
|
|
executable := sframe.all_before('(')
|
|
|
|
addr := sframe.all_after('[').all_before(']')
|
2019-11-23 12:22:55 +01:00
|
|
|
beforeaddr := sframe.all_before('[')
|
2019-11-13 09:05:06 +01:00
|
|
|
cmd := 'addr2line -e $executable $addr'
|
|
|
|
|
|
|
|
// taken from os, to avoid depending on the os module inside builtin.v
|
|
|
|
f := C.popen(cmd.str, 'r')
|
|
|
|
if isnil(f) {
|
|
|
|
println(sframe) continue
|
|
|
|
}
|
|
|
|
buf := [1000]byte
|
|
|
|
mut output := ''
|
|
|
|
for C.fgets(voidptr(buf), 1000, f) != 0 {
|
|
|
|
output += tos(buf, vstrlen(buf))
|
|
|
|
}
|
|
|
|
output = output.trim_space()+':'
|
|
|
|
if 0 != int(C.pclose(f)) {
|
|
|
|
println(sframe) continue
|
|
|
|
}
|
2019-11-23 12:22:55 +01:00
|
|
|
if output in ['??:0:','??:?:'] { output = '' }
|
|
|
|
println( '${output:-46s} | ${addr:14s} | $beforeaddr')
|
2019-11-13 09:05:06 +01:00
|
|
|
}
|
|
|
|
//C.backtrace_symbols_fd(*voidptr(&buffer[skipframes]), nr_actual_frames, 1)
|
|
|
|
return true
|
|
|
|
}$else{
|
|
|
|
C.printf('backtrace_symbols_fd is missing, so printing backtraces is not available.\n')
|
|
|
|
C.printf('Some libc implementations like musl simply do not provide it.\n')
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|