ci: fix vlib/dl/example/use_test.v in windows-tcc

pull/8347/head^2
Delyan Angelov 2021-01-25 21:19:15 +02:00
parent dbaccd4272
commit 4d0dab826b
No known key found for this signature in database
GPG Key ID: 66886C0F12D595ED
1 changed files with 111 additions and 93 deletions

View File

@ -1,71 +1,80 @@
// Copyright (c) 2019-2021 Alexander Medvednikov. All rights reserved. // Copyright (c) 2019-2021 Alexander Medvednikov. All rights reserved.
// Use of this source code is governed by an MIT license // Use of this source code is governed by an MIT license
// that can be found in the LICENSE file. // that can be found in the LICENSE file.
module builtin module builtin
// dbghelp.h is already included in cheaders.v // dbghelp.h is already included in cheaders.v
#flag windows -l dbghelp #flag windows -l dbghelp
// SymbolInfo is used by print_backtrace_skipping_top_frames_msvc
pub struct SymbolInfo { pub struct SymbolInfo {
pub mut: pub mut:
f_size_of_struct u32 // must be 88 to be recognised by SymFromAddr f_size_of_struct u32 // must be 88 to be recognised by SymFromAddr
f_type_index u32 // Type Index of symbol f_type_index u32 // Type Index of symbol
f_reserved [2]u64 f_reserved [2]u64
f_index u32 f_index u32
f_size u32 f_size u32
f_mod_base u64 // Base Address of module comtaining this symbol f_mod_base u64 // Base Address of module comtaining this symbol
f_flags u32 f_flags u32
f_value u64 // Value of symbol, ValuePresent should be 1 f_value u64 // Value of symbol, ValuePresent should be 1
f_address u64 // Address of symbol including base address of module f_address u64 // Address of symbol including base address of module
f_register u32 // register holding value or pointer to value f_register u32 // register holding value or pointer to value
f_scope u32 // scope of the symbol f_scope u32 // scope of the symbol
f_tag u32 // pdb classification f_tag u32 // pdb classification
f_name_len u32 // Actual length of name f_name_len u32 // Actual length of name
f_max_name_len u32 // must be manually set f_max_name_len u32 // must be manually set
f_name byte // must be calloc(f_max_name_len) f_name byte // must be calloc(f_max_name_len)
} }
pub struct SymbolInfoContainer { pub struct SymbolInfoContainer {
pub mut: pub mut:
syminfo SymbolInfo syminfo SymbolInfo
f_name_rest [254]char f_name_rest [254]char
} }
pub struct Line64 { pub struct Line64 {
pub mut: pub mut:
f_size_of_struct u32 f_size_of_struct u32
f_key voidptr f_key voidptr
f_line_number u32 f_line_number u32
f_file_name byteptr f_file_name byteptr
f_address u64 f_address u64
} }
fn C.SymSetOptions(symoptions u32) u32 // returns the current options mask // returns the current options mask
fn C.GetCurrentProcess() voidptr // returns handle fn C.SymSetOptions(symoptions u32) u32
// returns handle
fn C.GetCurrentProcess() voidptr
fn C.SymInitialize(h_process voidptr, p_user_search_path byteptr, b_invade_process int) int fn C.SymInitialize(h_process voidptr, p_user_search_path byteptr, b_invade_process int) int
fn C.CaptureStackBackTrace(frames_to_skip u32, frames_to_capture u32, p_backtrace voidptr, p_backtrace_hash voidptr) u16 fn C.CaptureStackBackTrace(frames_to_skip u32, frames_to_capture u32, p_backtrace voidptr, p_backtrace_hash voidptr) u16
fn C.SymFromAddr(h_process voidptr, address u64, p_displacement voidptr, p_symbol voidptr) int fn C.SymFromAddr(h_process voidptr, address u64, p_displacement voidptr, p_symbol voidptr) int
fn C.SymGetLineFromAddr64(h_process voidptr, address u64, p_displacement voidptr, p_line &Line64) int fn C.SymGetLineFromAddr64(h_process voidptr, address u64, p_displacement voidptr, p_line &Line64) int
// Ref - https://docs.microsoft.com/en-us/windows/win32/api/dbghelp/nf-dbghelp-symsetoptions // Ref - https://docs.microsoft.com/en-us/windows/win32/api/dbghelp/nf-dbghelp-symsetoptions
const ( const (
symopt_undname = 0x00000002 symopt_undname = 0x00000002
symopt_deferred_loads = 0x00000004 symopt_deferred_loads = 0x00000004
symopt_no_cpp = 0x00000008 symopt_no_cpp = 0x00000008
symopt_load_lines = 0x00000010 symopt_load_lines = 0x00000010
symopt_include_32bit_modules = 0x00002000 symopt_include_32bit_modules = 0x00002000
symopt_allow_zero_address = 0x01000000 symopt_allow_zero_address = 0x01000000
symopt_debug = 0x80000000 symopt_debug = 0x80000000
) )
// g_original_codepage - used to restore the original windows console code page when exiting // g_original_codepage - used to restore the original windows console code page when exiting
__global ( g_original_codepage = u32(0) ) __global ( g_original_codepage = u32(0))
// utf8 to stdout needs C.SetConsoleOutputCP(C.CP_UTF8) // utf8 to stdout needs C.SetConsoleOutputCP(C.CP_UTF8)
fn C.GetConsoleOutputCP() u32 fn C.GetConsoleOutputCP() u32
fn C.SetConsoleOutputCP(wCodePageID u32) bool fn C.SetConsoleOutputCP(wCodePageID u32) bool
fn restore_codepage() { fn restore_codepage() {
C.SetConsoleOutputCP( g_original_codepage ) C.SetConsoleOutputCP(g_original_codepage)
} }
fn builtin_init() { fn builtin_init() {
@ -98,65 +107,68 @@ fn print_backtrace_skipping_top_frames(skipframes int) bool {
} }
fn print_backtrace_skipping_top_frames_msvc(skipframes int) bool { fn print_backtrace_skipping_top_frames_msvc(skipframes int) bool {
$if msvc { $if msvc {
mut offset := u64(0) mut offset := u64(0)
backtraces := [100]voidptr{} backtraces := [100]voidptr{}
sic := SymbolInfoContainer{} sic := SymbolInfoContainer{}
mut si := &sic.syminfo mut si := &sic.syminfo
si.f_size_of_struct = sizeof(SymbolInfo) // Note: C.SYMBOL_INFO is 88 si.f_size_of_struct = sizeof(SymbolInfo) // Note: C.SYMBOL_INFO is 88
si.f_max_name_len = sizeof(SymbolInfoContainer) - sizeof(SymbolInfo) - 1 si.f_max_name_len = sizeof(SymbolInfoContainer) - sizeof(SymbolInfo) - 1
fname := charptr( &si.f_name ) fname := charptr(&si.f_name)
mut sline64 := Line64{} mut sline64 := Line64{}
sline64.f_size_of_struct = sizeof(Line64) sline64.f_size_of_struct = sizeof(Line64)
handle := C.GetCurrentProcess() handle := C.GetCurrentProcess()
defer { C.SymCleanup(handle) } defer {
C.SymCleanup(handle)
}
C.SymSetOptions(symopt_debug | symopt_load_lines | symopt_undname) C.SymSetOptions(symopt_debug | symopt_load_lines | symopt_undname)
syminitok := C.SymInitialize( handle, 0, 1) syminitok := C.SymInitialize(handle, 0, 1)
if syminitok != 1 { if syminitok != 1 {
eprintln('Failed getting process: Aborting backtrace.\n') eprintln('Failed getting process: Aborting backtrace.\n')
return false return false
} }
frames := int(C.CaptureStackBackTrace(skipframes + 1, 100, backtraces, 0)) frames := int(C.CaptureStackBackTrace(skipframes + 1, 100, backtraces, 0))
if frames < 2 { if frames < 2 {
eprintln('C.CaptureStackBackTrace returned less than 2 frames') eprintln('C.CaptureStackBackTrace returned less than 2 frames')
return false return false
} }
for i in 0..frames { for i in 0 .. frames {
frame_addr := backtraces[i] frame_addr := backtraces[i]
if C.SymFromAddr(handle, frame_addr, &offset, si) == 1 { if C.SymFromAddr(handle, frame_addr, &offset, si) == 1 {
nframe := frames - i - 1 nframe := frames - i - 1
mut lineinfo := '' mut lineinfo := ''
if C.SymGetLineFromAddr64(handle, frame_addr, &offset, &sline64) == 1 { if C.SymGetLineFromAddr64(handle, frame_addr, &offset, &sline64) == 1 {
file_name := tos3(sline64.f_file_name) file_name := tos3(sline64.f_file_name)
lineinfo = '${file_name}:${sline64.f_line_number}' lnumber := sline64.f_line_number
lineinfo = '$file_name:$lnumber'
} else {
addr:
lineinfo = '?? : address = 0x${(&frame_addr):x}'
}
sfunc := tos3(fname)
eprintln('${nframe:-2d}: ${sfunc:-25s} $lineinfo')
} else { } else {
addr : // https://docs.microsoft.com/en-us/windows/win32/debug/system-error-codes
lineinfo = '?? : address = 0x${(&frame_addr):x}' cerr := int(C.GetLastError())
} if cerr == 87 {
sfunc := tos3(fname) eprintln('SymFromAddr failure: $cerr = The parameter is incorrect)')
eprintln('${nframe:-2d}: ${sfunc:-25s} $lineinfo') } else if cerr == 487 {
} else { // probably caused because the .pdb isn't in the executable folder
// https://docs.microsoft.com/en-us/windows/win32/debug/system-error-codes eprintln('SymFromAddr failure: $cerr = Attempt to access invalid address (Verify that you have the .pdb file in the right folder.)')
cerr := int(C.GetLastError()) } else {
if cerr == 87 { eprintln('SymFromAddr failure: $cerr (see https://docs.microsoft.com/en-us/windows/win32/debug/system-error-codes)')
eprintln('SymFromAddr failure: $cerr = The parameter is incorrect)') }
} else if cerr == 487 {
// probably caused because the .pdb isn't in the executable folder
eprintln('SymFromAddr failure: $cerr = Attempt to access invalid address (Verify that you have the .pdb file in the right folder.)')
} else {
eprintln('SymFromAddr failure: $cerr (see https://docs.microsoft.com/en-us/windows/win32/debug/system-error-codes)')
} }
} }
return true
} $else {
eprintln('print_backtrace_skipping_top_frames_msvc must be called only when the compiler is msvc')
return false
} }
return true
} $else {
eprintln('print_backtrace_skipping_top_frames_msvc must be called only when the compiler is msvc')
return false
}
} }
fn print_backtrace_skipping_top_frames_mingw(skipframes int) bool { fn print_backtrace_skipping_top_frames_mingw(skipframes int) bool {
@ -165,43 +177,48 @@ fn print_backtrace_skipping_top_frames_mingw(skipframes int) bool {
} }
fn C.tcc_backtrace(fmt charptr, other ...charptr) int fn C.tcc_backtrace(fmt charptr, other ...charptr) int
fn print_backtrace_skipping_top_frames_tcc(skipframes int) bool { fn print_backtrace_skipping_top_frames_tcc(skipframes int) bool {
$if tinyc { $if tinyc {
C.tcc_backtrace("Backtrace") $if no_backtrace ? {
return false eprintln('backtraces are disabled')
} $else {
C.tcc_backtrace('Backtrace')
return false
}
} $else { } $else {
eprintln('print_backtrace_skipping_top_frames_tcc must be called only when the compiler is tcc') eprintln('print_backtrace_skipping_top_frames_tcc must be called only when the compiler is tcc')
return false return false
} }
} }
//TODO copypaste from os // TODO copypaste from os
// we want to be able to use this here without having to `import os` // we want to be able to use this here without having to `import os`
struct ExceptionRecord { struct ExceptionRecord {
pub: pub:
// status_ constants // status_ constants
code u32 code u32
flags u32 flags u32
record &ExceptionRecord
record &ExceptionRecord address voidptr
address voidptr
param_count u32 param_count u32
// params []voidptr // params []voidptr
} }
struct ContextRecord { struct ContextRecord {
// TODO // TODO
} }
struct ExceptionPointers { struct ExceptionPointers {
pub: pub:
exception_record &ExceptionRecord exception_record &ExceptionRecord
context_record &ContextRecord context_record &ContextRecord
} }
type VectoredExceptionHandler = fn(&ExceptionPointers)int type VectoredExceptionHandler = fn (&ExceptionPointers) int
fn C.AddVectoredExceptionHandler(int, C.PVECTORED_EXCEPTION_HANDLER) fn C.AddVectoredExceptionHandler(int, C.PVECTORED_EXCEPTION_HANDLER)
fn add_vectored_exception_handler(handler VectoredExceptionHandler) { fn add_vectored_exception_handler(handler VectoredExceptionHandler) {
C.AddVectoredExceptionHandler(1, C.PVECTORED_EXCEPTION_HANDLER(handler)) C.AddVectoredExceptionHandler(1, C.PVECTORED_EXCEPTION_HANDLER(handler))
} }
@ -228,6 +245,7 @@ fn add_unhandled_exception_handler() {
} }
fn C.IsDebuggerPresent() bool fn C.IsDebuggerPresent() bool
fn C.__debugbreak() fn C.__debugbreak()
fn break_if_debugger_attached() { fn break_if_debugger_attached() {