mirror of
https://github.com/mirror/tinycc.git
synced 2025-01-07 04:40:08 +08:00
853a498f2c
The BOUNDS_CHECKING_ON/BOUNDS_CHECKING_OFF is not working for signal/sigaction/fork. The reason is that the code stops bound checking for the whole application. This result in wrong handling of __bound_local_new/__bound_local_delete and malloc/calloc/realloc/free. Consider the following code: void tst(int n) { int i, arr[n]; for (i = 0; i < n; i++) arr[i] = 0; } void *some_thread(void *dummy) { while (running) { tst(10); tst(20); } } void signal_handler(int sig) { ... } When the signal handler is called the some_thread code can be interrupted when is just registered the arr[10] data. When the signal handler is leaved the arr[10] is still registered and did not see the call to deregister arr[10] and then register arr[20]. The code resumes when tst(20) is running. This results in a bound checking error when i >= 10. To solve the above problem I changed the bound checking code to use tls (thread local storage) for the no_checking variable. This also makes it now possible to redirect signal/sigaction/fork code through the bound checking library and disable checking when a signal is running and to correct the bounds_sem for the fork child process. The BOUNDS_CHECKING_ON/BOUNDS_CHECKING_OFF is not needed any more for signal/sigaction/fork. In fact I could remove them from all my applications. The use of the tls function code slows down the code by about 10%. So if the slowdown due to bound checking was 5. It is now 5.5 times slower. For x86_64/i386 I also allowed to use __thread variable in bcheck.c when compiled with gcc with: make x86_64-libtcc1-usegcc=yes make i386-libtcc1-usegcc=yes This makes code run faster due to use of gcc and __thread variable. With the __thread variable there is no 10% slowdown. For other targets this does not work because stabs is not supported. Changes: lib/bcheck.c: - Add TRY_SEM - Add HAVE_SIGNAL/HAVE_SIGACTION/HAVE_FORK/HAVE_TLS_FUNC/HAVE_TLS_VAR - HAVE_SIGNAL: redirect signal() call if set. - HAVE_SIGACTION: redirect sigaction() call if set. - HAVE_FORK: redirect fork() call if set. - HAVE_TLS_FUNC: If target has tls function calls. - HAVE_TLS_VAR: If target has __thread tls support. - Replace all no_checking refecrences to NO_CHECKING_SET/NO_CHECKING_GET macros tcc-doc.texi: - Remove examples for signal/sigaction/fork code. - Add some explanation for signal/sigaction/fork code. - Add documentaion for __bounds_checking(). tccelf.c: - Add support for SHF_TLS tests/tests2/114_bound_signal.c: - Remove BOUNDS_CHECKING_ON/BOUNDS_CHECKING_OFF - Add code to trigger failure when tls is not working. x86_64-link.c: - Add support for R_X86_64_TLSGD/R_X86_64_TLSLD/R_X86_64_DTPOFF32/R_X86_64_TPOFF32 i386-link.c: - Add support for R_386_TLS_GD/R_386_TLS_LDM/R_386_TLS_LDO_32/R_386_TLS_LE
99 lines
1.8 KiB
C
99 lines
1.8 KiB
C
#include <stdio.h>
|
|
#include <pthread.h>
|
|
#include <semaphore.h>
|
|
#include <signal.h>
|
|
#include <string.h>
|
|
#include <sys/types.h>
|
|
#include <unistd.h>
|
|
#include <errno.h>
|
|
#include <setjmp.h>
|
|
|
|
static volatile int run = 1;
|
|
static sem_t sem;
|
|
|
|
static void
|
|
add (int n)
|
|
{
|
|
int i;
|
|
int arr[n];
|
|
|
|
for (i = 0; i < n; i++) {
|
|
arr[i]++;
|
|
}
|
|
memset (&arr[0], 0, n * sizeof(int));
|
|
}
|
|
|
|
static void *
|
|
high_load (void *unused)
|
|
{
|
|
while (run) {
|
|
add(10);
|
|
add(20);
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
static void *
|
|
do_signal (void *unused)
|
|
{
|
|
while (run) {
|
|
kill (getpid(), SIGUSR1);
|
|
while (sem_wait(&sem) < 0 && errno == EINTR);
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
static void signal_handler(int sig)
|
|
{
|
|
add(10);
|
|
add(20);
|
|
sem_post (&sem);
|
|
}
|
|
|
|
int
|
|
main (void)
|
|
{
|
|
int i;
|
|
pthread_t id1, id2;
|
|
struct sigaction act;
|
|
sigjmp_buf sj;
|
|
sigset_t m;
|
|
time_t end;
|
|
|
|
memset (&act, 0, sizeof (act));
|
|
act.sa_handler = signal_handler;
|
|
act.sa_flags = 0;
|
|
sigemptyset (&act.sa_mask);
|
|
sigaction (SIGUSR1, &act, NULL);
|
|
|
|
sem_init (&sem, 1, 0);
|
|
pthread_create(&id1, NULL, high_load, NULL);
|
|
pthread_create(&id2, NULL, do_signal, NULL);
|
|
|
|
printf ("start\n");
|
|
/* sleep does not work !!! */
|
|
end = time(NULL) + 2;
|
|
while (time(NULL) < end) ;
|
|
run = 0;
|
|
printf ("end\n");
|
|
|
|
pthread_join(id1, NULL);
|
|
pthread_join(id2, NULL);
|
|
sem_destroy (&sem);
|
|
|
|
sigemptyset (&m);
|
|
sigprocmask (SIG_SETMASK, &m, NULL);
|
|
if (sigsetjmp (sj, 0) == 0)
|
|
{
|
|
sigaddset (&m, SIGUSR1);
|
|
sigprocmask (SIG_SETMASK, &m, NULL);
|
|
siglongjmp (sj, 1);
|
|
printf ("failed");
|
|
return 1;
|
|
}
|
|
sigprocmask (SIG_SETMASK, NULL, &m);
|
|
if (!sigismember (&m, SIGUSR1))
|
|
printf ("failed");
|
|
return 0;
|
|
}
|