2020-01-18 05:58:39 +08:00
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
/* function to get a stack backtrace on demand with a message */
|
|
|
|
|
2020-04-15 10:13:05 +08:00
|
|
|
#include <stdarg.h>
|
2020-01-18 05:58:39 +08:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
int (*__rt_error)(void*, void*, const char *, va_list);
|
2023-08-30 22:10:39 +08:00
|
|
|
void (*__rt_location)(void*, void*, const char *);
|
2020-01-18 05:58:39 +08:00
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
# define DLL_EXPORT __declspec(dllexport)
|
|
|
|
#else
|
|
|
|
# define DLL_EXPORT
|
|
|
|
#endif
|
|
|
|
|
2022-12-17 16:37:37 +08:00
|
|
|
/* Needed when using ...libtcc1-usegcc=yes in lib/Makefile */
|
|
|
|
#if (defined(__GNUC__) && (__GNUC__ >= 6)) || defined(__clang__)
|
|
|
|
#pragma GCC diagnostic push
|
|
|
|
#pragma GCC diagnostic ignored "-Wframe-address"
|
|
|
|
#endif
|
|
|
|
|
2020-01-18 05:58:39 +08:00
|
|
|
DLL_EXPORT int tcc_backtrace(const char *fmt, ...)
|
|
|
|
{
|
|
|
|
va_list ap;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (__rt_error) {
|
|
|
|
void *fp = __builtin_frame_address(1);
|
|
|
|
void *ip = __builtin_return_address(0);
|
|
|
|
va_start(ap, fmt);
|
|
|
|
ret = __rt_error(fp, ip, fmt, ap);
|
|
|
|
va_end(ap);
|
|
|
|
} else {
|
|
|
|
const char *p;
|
|
|
|
if (fmt[0] == '^' && (p = strchr(fmt + 1, fmt[0])))
|
|
|
|
fmt = p + 1;
|
|
|
|
va_start(ap, fmt);
|
|
|
|
ret = vfprintf(stderr, fmt, ap);
|
|
|
|
va_end(ap);
|
|
|
|
fprintf(stderr, "\n"), fflush(stderr);
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
2022-12-17 16:37:37 +08:00
|
|
|
|
2023-08-30 22:10:39 +08:00
|
|
|
DLL_EXPORT void tcc_location(const char *fmt)
|
|
|
|
{
|
|
|
|
if (__rt_location) {
|
|
|
|
void *fp = __builtin_frame_address(1);
|
|
|
|
void *ip = __builtin_return_address(0);
|
|
|
|
__rt_location(fp, ip, fmt);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-17 16:37:37 +08:00
|
|
|
#if (defined(__GNUC__) && (__GNUC__ >= 6)) || defined(__clang__)
|
|
|
|
#pragma GCC diagnostic pop
|
|
|
|
#endif
|