X-Git-Url: http://repo.macrolet.net/gitweb/?a=blobdiff_plain;f=src%2Fruntime%2Fthread.c;h=86626826916b3d75c38d258e9369c21b5259734c;hb=095564c28a259002c7e34fd1d861f5bbd0a959b6;hp=0d03ae265e42247e3a9dae11806e1d974935964e;hpb=a7c2f2622f1ceeeb3459cb6bbcf261bda1ff2327;p=sbcl.git diff --git a/src/runtime/thread.c b/src/runtime/thread.c index 0d03ae2..8662682 100644 --- a/src/runtime/thread.c +++ b/src/runtime/thread.c @@ -8,8 +8,8 @@ #include #include -#include "runtime.h" #include "sbcl.h" +#include "runtime.h" #include "validate.h" /* for CONTROL_STACK_SIZE etc */ #include "thread.h" #include "arch.h" @@ -19,6 +19,9 @@ #include "dynbind.h" #include "genesis/cons.h" #include "genesis/fdefn.h" +#include "interr.h" /* for lose() */ +#include "gc-internal.h" + #define ALIEN_STACK_SIZE (1*1024*1024) /* 1Mb size chosen at random */ int dynamic_values_bytes=4096*sizeof(lispobj); /* same for all threads */ @@ -26,8 +29,6 @@ struct thread *all_threads; volatile lispobj all_threads_lock; extern struct interrupt_data * global_interrupt_data; -void get_spinlock(volatile lispobj *word,int value); - int initial_thread_trampoline(struct thread *th) { @@ -159,13 +160,13 @@ struct thread * create_thread_struct(lispobj initial_function) { * we use the appropriate SymbolValue macros to access any of the * variable quantities from the C runtime. It's not quite OAOOM, * it just feels like it */ - SetSymbolValue(BINDING_STACK_START,th->binding_stack_start,th); - SetSymbolValue(CONTROL_STACK_START,th->control_stack_start,th); - SetSymbolValue(CONTROL_STACK_END,th->control_stack_end,th); + SetSymbolValue(BINDING_STACK_START,(lispobj)th->binding_stack_start,th); + SetSymbolValue(CONTROL_STACK_START,(lispobj)th->control_stack_start,th); + SetSymbolValue(CONTROL_STACK_END,(lispobj)th->control_stack_end,th); #ifdef LISP_FEATURE_X86 - SetSymbolValue(BINDING_STACK_POINTER,th->binding_stack_pointer,th); - SetSymbolValue(ALIEN_STACK,th->alien_stack_pointer,th); - SetSymbolValue(PSEUDO_ATOMIC_ATOMIC,th->pseudo_atomic_atomic,th); + SetSymbolValue(BINDING_STACK_POINTER,(lispobj)th->binding_stack_pointer,th); + SetSymbolValue(ALIEN_STACK,(lispobj)th->alien_stack_pointer,th); + SetSymbolValue(PSEUDO_ATOMIC_ATOMIC,(lispobj)th->pseudo_atomic_atomic,th); SetSymbolValue(PSEUDO_ATOMIC_INTERRUPTED,th->pseudo_atomic_interrupted,th); #else current_binding_stack_pointer=th->binding_stack_pointer; @@ -277,6 +278,32 @@ void destroy_thread (struct thread *th) 32*SIGSTKSZ); } +struct thread *find_thread_by_pid(pid_t pid) +{ + struct thread *th; + for_each_thread(th) + if(th->pid==pid) return th; + return 0; +} + +#if defined LISP_FEATURE_SB_THREAD +/* This is not needed unless #+SB-THREAD, as there's a trivial null + * unithread definition. */ + +void mark_dead_threads() +{ + pid_t kid; + int status; + while(1) { + kid=waitpid(-1,&status,__WALL|WNOHANG); + if(kid<=0) break; + if(WIFEXITED(status) || WIFSIGNALED(status)) { + struct thread *th=find_thread_by_pid(kid); + if(th) th->state=STATE_DEAD; + } + } +} + void reap_dead_threads() { struct thread *th,*next,*prev=0; @@ -304,19 +331,9 @@ void reap_dead_threads() } } - -struct thread *find_thread_by_pid(pid_t pid) -{ - struct thread *th; - for_each_thread(th) - if(th->pid==pid) return th; - return 0; -} - /* These are not needed unless #+SB-THREAD, and since sigwaitinfo() * doesn't seem to be easily available everywhere (OpenBSD...) it's * more trouble than it's worth to compile it when not needed. */ -#if defined LISP_FEATURE_SB_THREAD void block_sigcont(void) { /* don't allow ourselves to receive SIGCONT while we're in the @@ -343,9 +360,12 @@ void unblock_sigcont_and_sleep(void) int interrupt_thread(pid_t pid, lispobj function) { union sigval sigval; + struct thread *th; sigval.sival_int=function; - - return sigqueue(pid, SIG_INTERRUPT_THREAD, sigval); + for_each_thread(th) + if((th->pid==pid) && (th->state != STATE_DEAD)) + return sigqueue(pid, SIG_INTERRUPT_THREAD, sigval); + errno=EPERM; return -1; } int signal_thread_to_dequeue (pid_t pid) @@ -372,7 +392,12 @@ void gc_stop_the_world() if(p==th) continue; if(p->state==STATE_RUNNING) { p->state=STATE_STOPPING; - kill(p->pid,SIG_STOP_FOR_GC); + if(kill(p->pid,SIG_STOP_FOR_GC)==-1) { + /* we can't kill the process; assume because it + * died already (and its parent is dead so never + * saw the SIGCHLD) */ + p->state=STATE_DEAD; + } } if((p->state!=STATE_STOPPED) && (p->state!=STATE_DEAD)) {