Skip to content

Commit

Permalink
Provide C backtraces on exceptions (and via API) when available.
Browse files Browse the repository at this point in the history
  • Loading branch information
viega committed Jul 9, 2024
1 parent 59a1029 commit 08436b8
Show file tree
Hide file tree
Showing 18 changed files with 379 additions and 32 deletions.
1 change: 1 addition & 0 deletions dev
Original file line number Diff line number Diff line change
Expand Up @@ -55,6 +55,7 @@ function meson_build {

if [[ ${OS} = "Darwin" ]] ; then
meson compile
dsymutil c4test
else
CC=musl-gcc meson compile
fi
Expand Down
4 changes: 2 additions & 2 deletions include/compiler/datatypes/cfg.h
Original file line number Diff line number Diff line change
Expand Up @@ -36,8 +36,8 @@ typedef struct {
c4m_cfg_node_t *exit_node;
c4m_cfg_node_t **branch_targets;
c4m_utf8_t *label; // For loops
int32_t num_branches;
int32_t next_to_process;
int64_t num_branches;
int64_t next_to_process;
} c4m_cfg_branch_info_t;

typedef struct {
Expand Down
6 changes: 5 additions & 1 deletion include/con4m.h
Original file line number Diff line number Diff line change
@@ -1,9 +1,11 @@
#pragma once

// #define C4M_FULL_MEMCHECK
// #define C4M_DEBUG
#define C4M_DEBUG
// #define C4M_TRACE_GC

// #define C4M_GC_SHOW_COLLECT_STACK_TRACES

// #define C4M_GCT_MOVE 1
// #define C4M_GCT_PTR_TO_MOVE 1

Expand Down Expand Up @@ -161,6 +163,8 @@
// For functions we need to wrap to use through the FFI.
#include "con4m/wrappers.h"

#include "con4m/cbacktrace.h"

// The compiler.
#include "compiler/ast_utils.h"
#include "compiler/compile.h"
Expand Down
12 changes: 12 additions & 0 deletions include/con4m/cbacktrace.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
#pragma once
#include "con4m.h"

#ifdef BACKTRACE_SUPPORTED
void c4m_backtrace_init(char *);
void c4m_print_c_backtrace();
#else
#define c4m_backtrace_init(x)
#define c4m_print_c_backtrace()
#endif

c4m_grid_t *c4m_get_c_backtrace();
2 changes: 1 addition & 1 deletion include/con4m/datatypes/types.h
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@ typedef struct c4m_type_info_t {
c4m_dt_info_t *base_type;
c4m_list_t *items;
void *tsi; // Type-specific info.
uint8_t flags;
uint64_t flags;
} c4m_type_info_t;

#define C4M_FN_TY_VARARGS 1
Expand Down
29 changes: 20 additions & 9 deletions include/con4m/exception.h
Original file line number Diff line number Diff line change
Expand Up @@ -131,15 +131,26 @@
#define C4M_FINALLY C4M_LFINALLY(default_label)
#define C4M_TRY_END C4M_LTRY_END(default_label)

#define C4M_CRAISE(s, ...) c4m_exception_raise( \
c4m_alloc_exception(s, __VA_OPT__(, ) __VA_ARGS__), \
__FILE__, \
__LINE__)

#define C4M_RAISE(s, ...) c4m_exception_raise( \
c4m_alloc_str_exception(s __VA_OPT__(, ) __VA_ARGS__), \
__FILE__, \
__LINE__)
#if defined(C4M_DEBUG) && defined(BACKTRACE_SUPPORTED)
extern void c4m_print_c_backtrace();
#define c4m_trace() c4m_print_c_backtrace()
#else
#define c4m_trace()
#endif

#define C4M_CRAISE(s, ...) \
c4m_trace(); \
c4m_exception_raise( \
c4m_alloc_exception(s, __VA_OPT__(, ) __VA_ARGS__), \
__FILE__, \
__LINE__)

#define C4M_RAISE(s, ...) \
c4m_trace(); \
c4m_exception_raise( \
c4m_alloc_str_exception(s __VA_OPT__(, ) __VA_ARGS__), \
__FILE__, \
__LINE__)

#define C4M_RERAISE() \
_c4x_exception_state = C4M_EXCEPTION_NOT_HANDLED; \
Expand Down
1 change: 1 addition & 0 deletions include/vendor.h
Original file line number Diff line number Diff line change
Expand Up @@ -3,3 +3,4 @@
#include "vendor/unibreak.h"
#include "vendor/utf8proc.h"
#include "vendor/md4c.h"
#include "vendor/backtrace.h"
173 changes: 173 additions & 0 deletions include/vendor/backtrace.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,173 @@
/* backtrace.h -- Public header file for stack backtrace library.
Copyright (C) 2012-2021 Free Software Foundation, Inc.
Written by Ian Lance Taylor, Google.
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are
met:
(1) Redistributions of source code must retain the above copyright
notice, this list of conditions and the following disclaimer.
(2) Redistributions in binary form must reproduce the above copyright
notice, this list of conditions and the following disclaimer in
the documentation and/or other materials provided with the
distribution.
(3) The name of the author may not be used to
endorse or promote products derived from this software without
specific prior written permission.
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT,
INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
POSSIBILITY OF SUCH DAMAGE. */

#ifndef BACKTRACE_H
#define BACKTRACE_H

#include <stddef.h>
#include <stdint.h>
#include <stdio.h>

#ifdef __cplusplus
extern "C" {
#endif

/* The backtrace state. This struct is intentionally not defined in
the public interface. */

struct backtrace_state;

/* The type of the error callback argument to backtrace functions.
This function, if not NULL, will be called for certain error cases.
The DATA argument is passed to the function that calls this one.
The MSG argument is an error message. The ERRNUM argument, if
greater than 0, holds an errno value. The MSG buffer may become
invalid after this function returns.
As a special case, the ERRNUM argument will be passed as -1 if no
debug info can be found for the executable, or if the debug info
exists but has an unsupported version, but the function requires
debug info (e.g., backtrace_full, backtrace_pcinfo). The MSG in
this case will be something along the lines of "no debug info".
Similarly, ERRNUM will be passed as -1 if there is no symbol table,
but the function requires a symbol table (e.g., backtrace_syminfo).
This may be used as a signal that some other approach should be
tried. */

typedef void (*backtrace_error_callback)(void *data, const char *msg, int errnum);

/* Create state information for the backtrace routines. This must be
called before any of the other routines, and its return value must
be passed to all of the other routines. FILENAME is the path name
of the executable file; if it is NULL the library will try
system-specific path names. If not NULL, FILENAME must point to a
permanent buffer. If THREADED is non-zero the state may be
accessed by multiple threads simultaneously, and the library will
use appropriate atomic operations. If THREADED is zero the state
may only be accessed by one thread at a time. This returns a state
pointer on success, NULL on error. If an error occurs, this will
call the ERROR_CALLBACK routine.
Calling this function allocates resources that cannot be freed.
There is no backtrace_free_state function. The state is used to
cache information that is expensive to recompute. Programs are
expected to call this function at most once and to save the return
value for all later calls to backtrace functions. */

extern struct backtrace_state *backtrace_create_state(
const char *filename,
int threaded,
backtrace_error_callback error_callback,
void *data);

/* The type of the callback argument to the backtrace_full function.
DATA is the argument passed to backtrace_full. PC is the program
counter. FILENAME is the name of the file containing PC, or NULL
if not available. LINENO is the line number in FILENAME containing
PC, or 0 if not available. FUNCTION is the name of the function
containing PC, or NULL if not available. This should return 0 to
continuing tracing. The FILENAME and FUNCTION buffers may become
invalid after this function returns. */

typedef int (*backtrace_full_callback)(void *data, uintptr_t pc, const char *filename, int lineno, const char *function);

/* Get a full stack backtrace. SKIP is the number of frames to skip;
passing 0 will start the trace with the function calling
backtrace_full. DATA is passed to the callback routine. If any
call to CALLBACK returns a non-zero value, the stack backtrace
stops, and backtrace returns that value; this may be used to limit
the number of stack frames desired. If all calls to CALLBACK
return 0, backtrace returns 0. The backtrace_full function will
make at least one call to either CALLBACK or ERROR_CALLBACK. This
function requires debug info for the executable. */

extern int backtrace_full(struct backtrace_state *state, int skip, backtrace_full_callback callback, backtrace_error_callback error_callback, void *data);

/* The type of the callback argument to the backtrace_simple function.
DATA is the argument passed to simple_backtrace. PC is the program
counter. This should return 0 to continue tracing. */

typedef int (*backtrace_simple_callback)(void *data, uintptr_t pc);

/* Get a simple backtrace. SKIP is the number of frames to skip, as
in backtrace. DATA is passed to the callback routine. If any call
to CALLBACK returns a non-zero value, the stack backtrace stops,
and backtrace_simple returns that value. Otherwise
backtrace_simple returns 0. The backtrace_simple function will
make at least one call to either CALLBACK or ERROR_CALLBACK. This
function does not require any debug info for the executable. */

extern int backtrace_simple(struct backtrace_state *state, int skip, backtrace_simple_callback callback, backtrace_error_callback error_callback, void *data);

/* Print the current backtrace in a user readable format to a FILE.
SKIP is the number of frames to skip, as in backtrace_full. Any
error messages are printed to stderr. This function requires debug
info for the executable. */

extern void backtrace_print(struct backtrace_state *state, int skip, FILE *);

/* Given PC, a program counter in the current program, call the
callback function with filename, line number, and function name
information. This will normally call the callback function exactly
once. However, if the PC happens to describe an inlined call, and
the debugging information contains the necessary information, then
this may call the callback function multiple times. This will make
at least one call to either CALLBACK or ERROR_CALLBACK. This
returns the first non-zero value returned by CALLBACK, or 0. */

extern int backtrace_pcinfo(struct backtrace_state *state, uintptr_t pc, backtrace_full_callback callback, backtrace_error_callback error_callback, void *data);

/* The type of the callback argument to backtrace_syminfo. DATA and
PC are the arguments passed to backtrace_syminfo. SYMNAME is the
name of the symbol for the corresponding code. SYMVAL is the
value and SYMSIZE is the size of the symbol. SYMNAME will be NULL
if no error occurred but the symbol could not be found. */

typedef void (*backtrace_syminfo_callback)(void *data, uintptr_t pc, const char *symname, uintptr_t symval, uintptr_t symsize);

/* Given ADDR, an address or program counter in the current program,
call the callback information with the symbol name and value
describing the function or variable in which ADDR may be found.
This will call either CALLBACK or ERROR_CALLBACK exactly once.
This returns 1 on success, 0 on failure. This function requires
the symbol table but does not require the debug info. Note that if
the symbol table is present but ADDR could not be found in the
table, CALLBACK will be called with a NULL SYMNAME argument.
Returns 1 on success, 0 on error. */

extern int backtrace_syminfo(struct backtrace_state *state, uintptr_t addr, backtrace_syminfo_callback callback, backtrace_error_callback error_callback, void *data);

#ifdef __cplusplus
} /* End extern "C". */
#endif

#endif
10 changes: 9 additions & 1 deletion meson.build
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,13 @@ else
c_args = c_args + ['-D_GNU_SOURCE']
endif

backtrace = cc.find_library('backtrace',
required: false)

if backtrace.found()
c_args = c_args + ['-DBACKTRACE_SUPPORTED']
endif

fpty_code = '''
#include <stddef.h>
#include <pty.h>
Expand Down Expand Up @@ -141,6 +148,7 @@ c4m_src = ['src/con4m/style.c',
'src/con4m/ffi.c',
'src/con4m/watch.c',
'src/con4m/wrappers.c',
'src/con4m/ctrace.c',
'src/con4m/crypto/sha.c',
'src/con4m/compiler/compile.c',
'src/con4m/compiler/lex.c',
Expand Down Expand Up @@ -221,7 +229,7 @@ ssl = cc.find_library('ssl')
unibreak = dependency('libunibreak')
utf8proc = dependency('libutf8proc')

deps = [unibreak, utf8proc, threads, ffi, ssl, crypto]
deps = [unibreak, utf8proc, threads, ffi, ssl, crypto, backtrace]
opts = [math]
if using_glibc
opts = opts + [cc.find_library('atomic')]
Expand Down
4 changes: 4 additions & 0 deletions src/con4m/collect.c
Original file line number Diff line number Diff line change
Expand Up @@ -1045,6 +1045,10 @@ c4m_collect_arena(c4m_arena_t *from_space)
c4m_printf("[b]Average allocation size:[/] [em]{:,}[/] bytes",
c4m_box_u64((c4m_total_words * 8) / c4m_total_allocs));

#ifdef C4M_GC_SHOW_COLLECT_STACK_TRACES
c4m_c_backtrace();
#endif

#endif
return ctx.to_space;
}
Expand Down
18 changes: 14 additions & 4 deletions src/con4m/compiler/cfg_build.c
Original file line number Diff line number Diff line change
Expand Up @@ -39,12 +39,19 @@ c4m_cfg_enter_block(c4m_cfg_node_t *parent,

result->contents.block_entrance.inbound_links = c4m_new(
c4m_type_list(c4m_type_ref()));

exit->reference_location = treeloc;
exit->parent = result;
exit->starting_liveness_info = NULL;
exit->starting_sometimes = NULL;
exit->liveness_info = NULL;
exit->sometimes_live = NULL;
exit->kind = c4m_cfg_block_exit;
exit->contents.block_exit.next_node = NULL;
exit->contents.block_exit.entry_node = result;
exit->contents.block_exit.to_merge = NULL;
exit->contents.block_exit.inbound_links = c4m_new(
c4m_type_list(c4m_type_ref()));
exit->contents.block_exit.entry_node = result;

exit->parent = result;
exit->kind = c4m_cfg_block_exit;

if (parent != NULL) {
add_child(parent, result);
Expand Down Expand Up @@ -444,6 +451,8 @@ c4m_cfg_repr_internal(c4m_cfg_node_t *node,
c4m_kw("contents", c4m_ka(label)));
c4m_cfg_node_t *kid = node->contents.branches.branch_targets[i];

assert(kid != NULL);

c4m_tree_adopt_node(result, sub);
c4m_cfg_repr_internal(kid, sub, node, NULL);
}
Expand All @@ -452,6 +461,7 @@ c4m_cfg_repr_internal(c4m_cfg_node_t *node,
tree_parent,
node,
NULL);

return result;

case c4m_cfg_block_exit:
Expand Down
2 changes: 0 additions & 2 deletions src/con4m/compiler/check_pass.c
Original file line number Diff line number Diff line change
Expand Up @@ -1726,8 +1726,6 @@ handle_typeof_statement(pass2_ctx *ctx)

base_check_pass_dispatch(ctx);

c4m_print_parse_node(branch->children[1]);

ctx->cfg = c4m_cfg_exit_block(ctx->cfg, bstart, ctx->node);

if (c4m_list_len(tmp->sym_defs) > c4m_list_len(sym->sym_defs)) {
Expand Down
1 change: 0 additions & 1 deletion src/con4m/compiler/codegen.c
Original file line number Diff line number Diff line change
Expand Up @@ -907,7 +907,6 @@ gen_typeof(gen_ctx *ctx)
gen_one_tcase(ctx, switch_exit);
}
else {
emit(ctx, C4M_ZPop);
gen_one_node(ctx);
}
}
Expand Down
Loading

0 comments on commit 08436b8

Please sign in to comment.