Allow on_exit to be used with -run
Added on_exit support for tcc -run. Also fixed the last atexit code. Updated testcase 128 Fixed test/test2/Mafile for testcase 126
This commit is contained in:
parent
60d52f0d68
commit
fef701b57f
4 changed files with 112 additions and 37 deletions
94
tccrun.c
94
tccrun.c
|
@ -54,9 +54,6 @@ static rt_context g_rtctxt;
|
||||||
static void set_exception_handler(void);
|
static void set_exception_handler(void);
|
||||||
static int _rt_error(void *fp, void *ip, const char *fmt, va_list ap);
|
static int _rt_error(void *fp, void *ip, const char *fmt, va_list ap);
|
||||||
static void rt_exit(int code);
|
static void rt_exit(int code);
|
||||||
static void init_atexit(void);
|
|
||||||
static void run_atexit(void);
|
|
||||||
static int rt_atexit(void (*function)(void));
|
|
||||||
#endif /* CONFIG_TCC_BACKTRACE */
|
#endif /* CONFIG_TCC_BACKTRACE */
|
||||||
|
|
||||||
/* defined when included from lib/bt-exe.c */
|
/* defined when included from lib/bt-exe.c */
|
||||||
|
@ -148,6 +145,58 @@ static void run_cdtors(TCCState *s1, const char *start, const char *end,
|
||||||
((void(*)(int, char **, char **))*a++)(argc, argv, envp);
|
((void(*)(int, char **, char **))*a++)(argc, argv, envp);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#define NR_AT_EXIT 32
|
||||||
|
|
||||||
|
static int exit_called = 0;
|
||||||
|
static int nr_exit = 0;
|
||||||
|
static void (*exitfunc[NR_AT_EXIT])(int, void *);
|
||||||
|
static void *exitarg[NR_AT_EXIT];
|
||||||
|
#ifndef CONFIG_TCC_BACKTRACE
|
||||||
|
static jmp_buf run_jmp_buf;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
static void init_exit(void)
|
||||||
|
{
|
||||||
|
exit_called = 0;
|
||||||
|
nr_exit = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void call_exit(int ret)
|
||||||
|
{
|
||||||
|
while (nr_exit--)
|
||||||
|
exitfunc[nr_exit](ret, exitarg[nr_exit]);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int rt_atexit(void (*function)(void))
|
||||||
|
{
|
||||||
|
if (nr_exit < NR_AT_EXIT) {
|
||||||
|
exitfunc[nr_exit] = (void (*)(int, void *))function;
|
||||||
|
exitarg[nr_exit++] = NULL;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int rt_on_exit(void (*function)(int, void *), void *arg)
|
||||||
|
{
|
||||||
|
if (nr_exit < NR_AT_EXIT) {
|
||||||
|
exitfunc[nr_exit] = function;
|
||||||
|
exitarg[nr_exit++] = arg;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void run_exit(int code)
|
||||||
|
{
|
||||||
|
exit_called = 1;
|
||||||
|
#ifdef CONFIG_TCC_BACKTRACE
|
||||||
|
longjmp(g_rtctxt.jmp_buf, code ? code : 256);
|
||||||
|
#else
|
||||||
|
longjmp(run_jmp_buf, code ? code : 256);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
/* launch the compiled program with the given arguments */
|
/* launch the compiled program with the given arguments */
|
||||||
LIBTCCAPI int tcc_run(TCCState *s1, int argc, char **argv)
|
LIBTCCAPI int tcc_run(TCCState *s1, int argc, char **argv)
|
||||||
{
|
{
|
||||||
|
@ -168,11 +217,9 @@ LIBTCCAPI int tcc_run(TCCState *s1, int argc, char **argv)
|
||||||
s1->runtime_main = s1->nostdlib ? "_start" : "main";
|
s1->runtime_main = s1->nostdlib ? "_start" : "main";
|
||||||
if ((s1->dflag & 16) && (addr_t)-1 == get_sym_addr(s1, s1->runtime_main, 0, 1))
|
if ((s1->dflag & 16) && (addr_t)-1 == get_sym_addr(s1, s1->runtime_main, 0, 1))
|
||||||
return 0;
|
return 0;
|
||||||
#ifdef CONFIG_TCC_BACKTRACE
|
tcc_add_symbol(s1, "exit", run_exit);
|
||||||
if (s1->do_debug)
|
|
||||||
tcc_add_symbol(s1, "exit", rt_exit);
|
|
||||||
#endif
|
|
||||||
tcc_add_symbol(s1, "atexit", rt_atexit);
|
tcc_add_symbol(s1, "atexit", rt_atexit);
|
||||||
|
tcc_add_symbol(s1, "on_exit", rt_on_exit);
|
||||||
if (tcc_relocate(s1, TCC_RELOCATE_AUTO) < 0)
|
if (tcc_relocate(s1, TCC_RELOCATE_AUTO) < 0)
|
||||||
return -1;
|
return -1;
|
||||||
prog_main = (void*)get_sym_addr(s1, s1->runtime_main, 1, 1);
|
prog_main = (void*)get_sym_addr(s1, s1->runtime_main, 1, 1);
|
||||||
|
@ -219,19 +266,23 @@ LIBTCCAPI int tcc_run(TCCState *s1, int argc, char **argv)
|
||||||
errno = 0; /* clean errno value */
|
errno = 0; /* clean errno value */
|
||||||
fflush(stdout);
|
fflush(stdout);
|
||||||
fflush(stderr);
|
fflush(stderr);
|
||||||
init_atexit();
|
init_exit();
|
||||||
/* These aren't C symbols, so don't need leading underscore handling. */
|
/* These aren't C symbols, so don't need leading underscore handling. */
|
||||||
run_cdtors(s1, "__init_array_start", "__init_array_end", argc, argv, envp);
|
run_cdtors(s1, "__init_array_start", "__init_array_end", argc, argv, envp);
|
||||||
#ifdef CONFIG_TCC_BACKTRACE
|
#ifdef CONFIG_TCC_BACKTRACE
|
||||||
if (!rc->do_jmp || !(ret = setjmp(rc->jmp_buf)))
|
if (!(ret = setjmp(rc->jmp_buf)))
|
||||||
|
#else
|
||||||
|
if (!(ret = setjmp(run_jmp_buf)))
|
||||||
#endif
|
#endif
|
||||||
{
|
{
|
||||||
ret = prog_main(argc, argv, envp);
|
ret = prog_main(argc, argv, envp);
|
||||||
}
|
}
|
||||||
run_cdtors(s1, "__fini_array_start", "__fini_array_end", 0, NULL, NULL);
|
run_cdtors(s1, "__fini_array_start", "__fini_array_end", 0, NULL, NULL);
|
||||||
run_atexit();
|
call_exit(ret);
|
||||||
if ((s1->dflag & 16) && ret)
|
if ((s1->dflag & 16) && ret)
|
||||||
fprintf(s1->ppfp, "[returns %d]\n", ret), fflush(s1->ppfp);
|
fprintf(s1->ppfp, "[returns %d]\n", ret), fflush(s1->ppfp);
|
||||||
|
if ((s1->dflag & 16) == 0 && exit_called)
|
||||||
|
exit(ret);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1028,29 +1079,6 @@ static void rt_exit(int code)
|
||||||
exit(code);
|
exit(code);
|
||||||
}
|
}
|
||||||
|
|
||||||
#define NR_AT_EXIT 32
|
|
||||||
|
|
||||||
static int nr_atexit = 0;
|
|
||||||
static void (*at_exitfunc[NR_AT_EXIT])(void);
|
|
||||||
|
|
||||||
static void init_atexit(void)
|
|
||||||
{
|
|
||||||
nr_atexit = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void run_atexit(void)
|
|
||||||
{
|
|
||||||
while (nr_atexit)
|
|
||||||
at_exitfunc[--nr_atexit]();
|
|
||||||
}
|
|
||||||
|
|
||||||
static int rt_atexit(void (*function)(void))
|
|
||||||
{
|
|
||||||
if (nr_atexit < NR_AT_EXIT)
|
|
||||||
at_exitfunc[nr_atexit++] = function;
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* ------------------------------------------------------------- */
|
/* ------------------------------------------------------------- */
|
||||||
|
|
||||||
#ifndef _WIN32
|
#ifndef _WIN32
|
||||||
|
|
|
@ -1,5 +1,7 @@
|
||||||
#include <stdio.h>
|
int printf(const char *format, ...);
|
||||||
#include <stdlib.h>
|
int atexit(void (*function)(void));
|
||||||
|
int on_exit(void (*function)(int, void *), void *arg);
|
||||||
|
void exit(int status);
|
||||||
|
|
||||||
void cleanup1(void)
|
void cleanup1(void)
|
||||||
{
|
{
|
||||||
|
@ -11,10 +13,40 @@ void cleanup2(void)
|
||||||
printf ("cleanup2\n");
|
printf ("cleanup2\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
int main(int argc, char **argv)
|
void cleanup3(int ret, void *arg)
|
||||||
|
{
|
||||||
|
printf ("%d %s\n", ret, (char *) arg);
|
||||||
|
}
|
||||||
|
|
||||||
|
void cleanup4(int ret, void *arg)
|
||||||
|
{
|
||||||
|
printf ("%d %s\n", ret, (char *) arg);
|
||||||
|
}
|
||||||
|
|
||||||
|
void __attribute__((destructor)) cleanup5(void)
|
||||||
|
{
|
||||||
|
printf ("cleanup5\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
void test(void)
|
||||||
{
|
{
|
||||||
atexit(cleanup1);
|
atexit(cleanup1);
|
||||||
atexit(cleanup2);
|
atexit(cleanup2);
|
||||||
return 0;
|
on_exit(cleanup3, "cleanup3");
|
||||||
|
on_exit(cleanup4, "cleanup4");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if defined test_128_return
|
||||||
|
int main(int argc, char **argv)
|
||||||
|
{
|
||||||
|
test();
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
#elif defined test_128_exit
|
||||||
|
int main(int argc, char **argv)
|
||||||
|
{
|
||||||
|
test();
|
||||||
|
exit(2);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
|
@ -1,2 +1,15 @@
|
||||||
|
[test_128_return]
|
||||||
|
cleanup5
|
||||||
|
1 cleanup4
|
||||||
|
1 cleanup3
|
||||||
cleanup2
|
cleanup2
|
||||||
cleanup1
|
cleanup1
|
||||||
|
[returns 1]
|
||||||
|
|
||||||
|
[test_128_exit]
|
||||||
|
cleanup5
|
||||||
|
2 cleanup4
|
||||||
|
2 cleanup3
|
||||||
|
cleanup2
|
||||||
|
cleanup1
|
||||||
|
[returns 2]
|
||||||
|
|
|
@ -36,6 +36,7 @@ ifeq ($(CONFIG_bcheck),no)
|
||||||
SKIP += 115_bound_setjmp.test
|
SKIP += 115_bound_setjmp.test
|
||||||
SKIP += 116_bound_setjmp2.test
|
SKIP += 116_bound_setjmp2.test
|
||||||
SKIP += 117_builtins.test
|
SKIP += 117_builtins.test
|
||||||
|
SKIP += 126_bound_global.test
|
||||||
endif
|
endif
|
||||||
ifeq ($(CONFIG_dll),no)
|
ifeq ($(CONFIG_dll),no)
|
||||||
SKIP += 113_btdll.test # no shared lib support yet
|
SKIP += 113_btdll.test # no shared lib support yet
|
||||||
|
@ -126,6 +127,7 @@ endif
|
||||||
124_atomic_counter.test: FLAGS += -pthread
|
124_atomic_counter.test: FLAGS += -pthread
|
||||||
126_bound_global.test: FLAGS += -b
|
126_bound_global.test: FLAGS += -b
|
||||||
126_bound_global.test: NORUN = true
|
126_bound_global.test: NORUN = true
|
||||||
|
128_run_atexit.test: FLAGS += -dt
|
||||||
|
|
||||||
# Filter source directory in warnings/errors (out-of-tree builds)
|
# Filter source directory in warnings/errors (out-of-tree builds)
|
||||||
FILTER = 2>&1 | sed -e 's,$(SRC)/,,g'
|
FILTER = 2>&1 | sed -e 's,$(SRC)/,,g'
|
||||||
|
|
Loading…
Add table
Reference in a new issue