home
***
CD-ROM
|
disk
|
FTP
|
other
***
search
/
Geek Gadgets 1
/
ADE-1.bin
/
ade-dist
/
gnat-2.06-src.tgz
/
tar.out
/
fsf
/
gnat
/
ada
/
threads
/
src
/
signal.c
< prev
next >
Wrap
C/C++ Source or Header
|
1996-09-28
|
59KB
|
2,341 lines
/* Copyright (C) 1992, the Florida State University
Distributed by the Florida State University under the terms of the
GNU Library General Public License.
This file is part of Pthreads.
Pthreads is free software; you can redistribute it and/or
modify it under the terms of the GNU Library General Public
License as published by the Free Software Foundation (version 2).
Pthreads is distributed "AS IS" in the hope that it will be
useful, but WITHOUT ANY WARRANTY; without even the implied
warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
See the GNU Library General Public License for more details.
You should have received a copy of the GNU Library General Public
License along with Pthreads; see the file COPYING. If not, write
to the Free Software Foundation, 675 Mass Ave, Cambridge,
MA 02139, USA.
Report problems and direct all questions to:
pthreads-bugs@ada.cs.fsu.edu
@(#)signal.c 2.5 4/12/95
*/
/*
* Functions for the handling of signals and timers.
*/
/*
* The DEBUG flag causes a message to be printed out during signal handling.
* The IO flag handles I/O requests asynchronously such that a signal is
* delivered to the process upon completion of the operation.
* If both flags are set at the same time, the signal handler would issue
* an I/O request for each invocation which in turns causes another signal
* to be delivered to yet another instance of the signal handler.
* To avoid this, messages are only printed if DEBUG is defined but not IO.
*/
#define PTHREAD_KERNEL
#include "pthread_setjmp.h"
#include "pthread_offsets.h"
#include "signal_internals.h"
#include "pthread_internals.h"
#include <sys/syscall.h>
#if defined(SOLARIS) && defined(IO)
#include <dlfcn.h>
#endif SOLARIS && IO
#ifdef NOERR_CHECK
#undef NOERR_CHECK
#include "mutex.h"
#define NOERR_CHECK
#else !NOERR_CHECK
#include "mutex.h"
#endif NOERR_CHECK
#ifdef STAND_ALONE
#include <sun4e/counter.h>
extern char heap_start;
#endif STAND_ALONE
#ifdef RAND_SWITCH
extern int pthread_n_ready;
#endif
#ifdef TIMER_DEBUG
static struct timeval last_alarm;
#endif
#if defined(STACK_CHECK) && defined(SIGNAL_STACK)
extern int pthread_page_size;
extern KERNEL_STACK pthread_tempstack;
#endif
#ifdef STAND_ALONE
pthread_timer_q_t pthread_timer; /* timer queue */
#else
pthread_timer_q pthread_timer; /* timer queue */
static struct itimerval it; /* timer structure */
#endif
struct sigaction pthread_user_handler[NNSIG]; /* user signal handlers */
volatile int new_code[NNSIG]; /* UNIX signal code (new sigs) */
static int pending_code[NNSIG]; /* UNIX signal code (pending) */
static sigset_t synchronous; /* set of synchronous signals */
static sigset_t sig_handling; /* set of signals being handled*/
/*------------------------------------------------------------*/
/*
* pthread_sigcpyset2set - add a signal set to another one
*/
void pthread_sigcpyset2set(dst, src)
int *dst, *src;
{
int i;
for (i = 0; i < sizeof(sigset_t); i += sizeof(int))
*(dst++) = *(src++);
}
/*------------------------------------------------------------*/
/*
* pthread_sigaddset2set - add a signal set to another one
*/
void pthread_sigaddset2set(dst, src)
int *dst, *src;
{
int i;
for (i = 0; i < sizeof(sigset_t); i += sizeof(int))
*(dst++) |= *(src++);
}
/*------------------------------------------------------------*/
/*
* pthread_sigdelset2set - delete a signal set from another one
*/
void pthread_sigdelset2set(dst, src)
int *dst, *src;
{
int i;
for (i = 0; i < sizeof(sigset_t); i += sizeof(int))
*(dst++) &= ~(*(src++));
}
/*------------------------------------------------------------*/
/*
* pthread_sigismemberset2set - check if two sets overlap
*/
int pthread_sigismemberset2set(dst, src)
int *dst, *src;
{
int i;
for (i = 0; i < sizeof(sigset_t); i += sizeof(int))
if (*(dst++) & *(src++))
return(TRUE);
return(FALSE);
}
/*------------------------------------------------------------*/
/*
* pthread_signonemptyset - check if a set has any bits set
*/
int pthread_signonemptyset(set)
int *set;
{
int i;
for (i = 0; i < sizeof(sigset_t); i += sizeof(int))
if (*(set++))
return(TRUE);
return(FALSE);
}
/*------------------------------------------------------------*/
/*
* pthread_siggeset2set - check if dst set is >= src set, i.e.
* dst has at least the bits set which src has set
*/
int pthread_siggeset2set(dst, src)
int *dst, *src;
{
int i;
for (i = 0; i < sizeof(sigset_t); i += sizeof(int))
if ((*(dst++) & *(src)) != *(src++))
return(FALSE);
return(TRUE);
}
/*------------------------------------------------------------*/
/*
* pthread_sigeqset2set - check if dst set is == src set
*/
int pthread_sigeqset2set(dst, src)
int *dst, *src;
{
int i;
for (i = 0; i < sizeof(sigset_t); i += sizeof(int))
if (*(dst++) != *(src++))
return(FALSE);
return(TRUE);
}
#ifndef STAND_ALONE
/*------------------------------------------------------------*/
/*
* pthread_clear_sighandler - get rid of universal signal handler for all
* signals except for those which cannot be masked;
* also invalidate the timer if still active
*/
void pthread_clear_sighandler()
{
struct sigaction vec;
register int sig;
struct itimerval it;
vec.sa_handler = SIG_DFL;
pthread_sigcpyset2set(&vec.sa_mask, &all_signals);
vec.sa_flags = 0;
for (sig = 1; sig < NSIG; sig++)
if (sig != SIGKILL && sig != SIGSTOP)
if (SIGACTION(sig, &vec, (struct sigaction *) NULL))
#ifdef DEBUG
fprintf(stderr,
"Pthreads: Could not install handler for signal %d\n", sig)
#endif
;
if (!getitimer(ITIMER_REAL, &it) && timerisset(&it.it_value)) {
it.it_value.tv_sec = it.it_value.tv_usec = 0;
it.it_interval.tv_sec = it.it_interval.tv_usec = 0;
setitimer(ITIMER_REAL, &it, (struct itimerval *) NULL);
}
}
#endif !STAND_ALONE
/*------------------------------------------------------------*/
/*
* default_action - take default action on process
* Notice: SIGSTOP and SIGKILL can never be received (unmaskable)
* but they are included anyway.
*/
static void default_action(sig)
int sig;
{
switch (sig) {
case SIGURG:
case SIGCONT:
case SIGCHLD:
case SIGIO:
case SIGWINCH:
break; /* ignore or continue */
case SIGSTOP:
case SIGTSTP:
case SIGTTIN:
case SIGTTOU:
#ifdef STAND_ALONE
exit();
#else
kill(getpid(), SIGSTOP); /* stop process -> UNIX may generate SIGCHLD */
#endif
break;
default:
#ifdef TIMER_DEBUG
fprintf(stderr, "last alarm is %d.%d\n",
last_alarm.tv_sec, last_alarm.tv_usec);
if (pthread_timer.head != NO_TIMER) {
fprintf(stderr, "wakeup time is %d.%d\n", pthread_timer.head->tp.tv_sec,
pthread_timer.head->tp.tv_usec);
gettimeofday(&pthread_timer.head->tp, (struct timezone *) NULL);
fprintf(stderr, "current time is %d.%d\n", pthread_timer.head->tp.tv_sec,
pthread_timer.head->tp.tv_usec);
}
#endif
#ifndef STAND_ALONE
pthread_clear_sighandler();
kill(getpid(), sig); /* reissue signal to terminate process: UNIX default */
{
sigset_t empty;
sigemptyset(&empty);
SIGPROCMASK(SIG_SETMASK, &empty, (struct sigset_t *) NULL);
}
#endif !STAND_ALONE
#ifdef DEBUG
#ifndef IO
fprintf(stderr, "RUNAWAY: should produce core dump now\n");
#endif DEBUG
#endif IO
#ifdef STAND_ALONE
exit();
#else !STAND_ALONE
pthread_process_exit(-1); /* abnormal termination, but no core dump */
#endif STAND_ALONE
}
}
/*------------------------------------------------------------*/
/*
* handle_thread_signal - try to handle one signal on a thread and
* return TRUE if it was handled, otherwise return FALSE
*/
static int handle_thread_signal(p, sig, code)
pthread_t p;
int sig;
int code;
{
register struct context_t *scp;
#ifdef C_CONTEXT_SWITCH
extern void pthread_fake_call_wrapper_wrapper();
#endif C_CONTEXT_SWITCH
#ifndef STAND_ALONE
/*
* handle timer signals
*/
if (sig == TIMER_SIG) {
#ifdef TIMER_DEBUG
fprintf(stderr, "handle_thread_signal: timer signal\n");
#endif
#ifdef DEF_RR
#ifdef REAL_TIME
if (pthread_timer.head->mode == DL_TIME)
pthread_cancel_timed_sigwait(p, TRUE, ANY_TIME, p->queue != &ready);
else {
#endif REAL_TIME
if (pthread_timer.head->mode == RR_TIME && p->queue == &ready) {
pthread_cancel_timed_sigwait(p, TRUE, ANY_TIME, FALSE);
pthread_q_deq(&ready, p, PRIMARY_QUEUE);
pthread_q_primary_enq(&ready, p);
} else
#endif DER_FF
pthread_cancel_timed_sigwait(p, TRUE, ANY_TIME, p->queue != &ready);
return(TRUE);
}
#ifdef REAL_TIME
}
#endif REAL_TIME
#endif !STAND_ALONE
/*
* handle signals for sigwait
*/
if (p->state & T_SIGWAIT && sigismember(&p->sigwaitset, sig)) {
pthread_q_wakeup_thread(NO_QUEUE, p, NO_QUEUE_INDEX);
p->state &= ~T_SIGWAIT;
pthread_sigaddset2set(&p->mask, &p->sigwaitset);
pthread_sigdelset2set(&p->mask, &cantmask);
sigdelset(&p->sigwaitset, sig);
return(TRUE);
}
/*
* handler set to ignore
*/
if (pthread_user_handler[sig].sa_handler == SIG_IGN && sig != SIGCANCEL)
return(TRUE);
/*
* handler set to default action
*/
if (pthread_user_handler[sig].sa_handler == SIG_DFL && sig != SIGCANCEL) {
default_action(sig);
return(TRUE);
}
/*
* handle signals for sigsuspend and user handlers
*/
if (sigismember(&handlerset, sig)) {
if (p->state & T_BLOCKED)
p->errno = EINTR;
if (!(p->state & T_RUNNING)) {
if (p->state & T_SYNCTIMER)
pthread_cancel_timed_sigwait(p, FALSE, SYNC_TIME, TRUE);
else {
pthread_q_wakeup_thread(p->queue, p, PRIMARY_QUEUE);
if (p->state & (T_SIGWAIT | T_SIGSUSPEND)) {
p->state &= ~(T_SIGWAIT | T_SIGSUSPEND);
sigemptyset(&p->sigwaitset);
}
}
}
p->sig_info[sig].si_signo = sig;
p->sig_info[sig].si_code = code;
#ifdef C_CONTEXT_SWITCH
if (pthread_not_called_from_sighandler(p->context[JB_PC]))
p->nscp = (struct context_t *) DIRECTED_AT_THREAD;
p->sig = sig;
p->osp = p->context[JB_SP];
p->opc = p->context[JB_PC];
p->context[JB_PC] = (int) pthread_fake_call_wrapper_wrapper;
#ifdef ASM_SETJMP
p->opc += RETURN_OFFSET;
p->context[JB_PC] -= RETURN_OFFSET;
#endif ASM_SETJMP
#else !C_CONTEXT_SWITCH
if (pthread_not_called_from_sighandler(p->context[JB_PC]))
scp = (struct context_t *) DIRECTED_AT_THREAD;
else
scp = p->nscp;
pthread_push_fake_call(p, pthread_user_handler[sig].sa_handler, sig,
scp, &pthread_user_handler[sig].sa_mask);
#endif !C_CONTEXT_SWITCH
return(TRUE);
}
/*
* handle cancel signal
*/
if (sig == SIGCANCEL) {
if (p->state & T_SYNCTIMER)
pthread_cancel_timed_sigwait(p, FALSE, ALL_TIME, TRUE);
else if (p->state & (T_SIGWAIT | T_SIGSUSPEND)) {
p->state &= ~(T_SIGWAIT | T_SIGSUSPEND);
sigemptyset(&p->sigwaitset);
}
if (p->queue && !(p->state & T_RUNNING))
pthread_q_deq(p->queue, p, PRIMARY_QUEUE);
pthread_q_deq(&all, p, ALL_QUEUE);
/*
* no more signals for this thread, not even cancellation signal
*/
pthread_sigcpyset2set(&p->mask, &all_signals);
sigaddset(&p->mask, SIGCANCEL);
#ifdef C_CONTEXT_SWITCH
p->nscp = (struct context_t *) DIRECTED_AT_THREAD;
p->sig = -1;
p->context[JB_PC] = (int) pthread_fake_call_wrapper_wrapper;
#ifdef ASM_SETJMP
p->context[JB_PC] -= RETURN_OFFSET;
#endif ASM_SETJMP
#else !C_CONTEXT_SWITCH
pthread_push_fake_call(p, pthread_exit, -1,
(struct context_t *) DIRECTED_AT_THREAD,
(struct sigset_t *) NULL);
#endif !C_CONTEXT_SWITCH
if (!(p->state & T_RUNNING))
pthread_q_wakeup_thread(NO_QUEUE, p, NO_QUEUE_INDEX);
return(TRUE);
}
return (FALSE);
}
/*------------------------------------------------------------*/
/*
* handle_one_signal - handle one signal on the process level
* assumes SET_KERNEL_FLAG
*/
static void handle_one_signal(sig, code)
int sig;
int code;
{
register pthread_t p = mac_pthread_self();
struct itimerval it;
struct timespec now;
extern pthread_t pthread_q_all_find_receiver();
static int aio_handle();
/*
* Determine who needs to get the signal (in the following order):
* (1) signal directed at specific thread: take this thread
* (2) signal at process level:
* (2a) synchronous signal: direct at current thread
* (2b) SIGALRM, timer queue not empty, timer expired: take head off timer q
* (2c) SIGIO, asynchronous I/O requested: determine receiver and make ready
* (2c) handler defined: take first thread in all queue with signal unmasked
* (3) no handler defined: pend signal on process till thread unmasks signal
* if signal already pending, it's lost
*/
if (p != NO_PTHREAD &&
(p->nscp == DIRECTED_AT_THREAD ||
pthread_not_called_from_sighandler(p->context[JB_PC])) &&
(p = (pthread_t) code))
code = 0;
else if (p != NO_PTHREAD && sigismember(&synchronous, sig))
/* p = p */;
#ifdef STAND_ALONE
else if (sig == SIGALRM) {
if ((p = pthread_timer) &&
!clock_gettime(CLOCK_REALTIME, &now) && GTEQ_NTIME(now, p->tp))
pthread_cancel_timed_sigwait(p, TRUE, ANY_TIME, p->queue != &ready);
return;
}
/* SIGIO is being used to invoke pthread_select_isr which takes care of
* suspending and waking up threads waiting on I/O.
* The parameter 2 is used instead of UART_ISR
*/
else if (sig == SIGIO) {
pthread_select_isr(SIGIO);
return;
}
#else !STAND_ALONE
else if (sig == SIGALRM &&
#ifdef DEF_RR
pthread_timer.head != NO_TIMER && (p = pthread_timer.head->thread) &&
#else !DEF_RR
(p = pthread_timer.head) &&
#endif DEF_RR
!getitimer(ITIMER_REAL, &it) && !timerisset(&it.it_value))
sig = TIMER_SIG;
#endif !STAND_ALONE
#ifdef IO
else if (sig == SIGIO && aio_handle())
return;
#endif
else if (!(p = pthread_q_all_find_receiver(&all, sig))) {
if (!sigismember(&pending_signals, sig)) {
sigaddset(&pending_signals, sig);
pending_code[sig] = code;
}
return;
}
if (p->state & T_RETURNED)
return;
/*
* Pend signal on thread if it's masked out OR
* if the signal is SIGCANCEL, the interrupt state CONTROLLED, and
* we are not at an interruption point.
*/
if (sigismember(&p->mask, sig) ||
sig == SIGCANCEL &&
p->state & T_CONTROLLED && !(p->state & T_INTR_POINT)) {
sigaddset(&p->pending, sig);
p->sig_info[sig].si_code = code;
return;
}
if (handle_thread_signal(p, sig, code))
return;
default_action(sig);
}
/*------------------------------------------------------------*/
/*
* pthread_handle_many_process_signals - determine pending signal(s).
* if no thread ready, suspend process;
* returns the head of the ready queue.
* assumes SET_KERNEL_FLAG
*/
pthread_t pthread_handle_many_process_signals()
{
register int sig;
do {
while (pthread_signonemptyset(&new_signals)) {
/*
* start critical section
*/
SIGPROCMASK(SIG_BLOCK, &all_signals, (struct sigset_t *) NULL);
pthread_sigcpyset2set(&sig_handling, &new_signals);
pthread_sigdelset2set(&new_signals, &sig_handling);
SIGPROCMASK(SIG_UNBLOCK, &all_signals, (struct sigset_t *) NULL);
/*
* end of critical section
*/
for (sig = 1; sig < NNSIG; sig++)
if (sigismember(&sig_handling, sig))
handle_one_signal(sig, new_code[sig]);
}
/*
* No thread, no action: suspend waiting for signal at process level
*/
if (ready.head == NO_PTHREAD) {
SIGPROCMASK(SIG_BLOCK, &all_signals, (struct sigset_t *) NULL);
if (!pthread_signonemptyset(&new_signals)) {
#ifdef DEBUG
#ifndef IO
fprintf(stderr, "suspending process waiting for signal\n");
#endif
#endif
#ifdef STAND_ALONE
while(!pthread_signonemptyset(&new_signals))
#endif
SIGSUSPEND(&no_signals);
}
SIGPROCMASK(SIG_UNBLOCK, &all_signals, (struct sigset_t *) NULL);
}
} while (ready.head == NO_PTHREAD);
return(ready.head);
}
/*------------------------------------------------------------*/
/*
* pthread_handle_one_process_signal - handle latest signal caught by
* universal handler while not in kernel
* returns the head of the ready queue.
* assumes SET_KERNEL_FLAG
*/
pthread_t pthread_handle_one_process_signal(sig, code)
int sig;
int code;
{
handle_one_signal(sig, code);
if (pthread_signonemptyset(&new_signals) ||
ready.head == NO_PTHREAD)
pthread_handle_many_process_signals();
}
/*------------------------------------------------------------*/
/*
* pthread_handle_pending_signals - handle unmasked pending signals of
* current thread assumes SET_KERNEL_FLAG
*/
void pthread_handle_pending_signals()
{
pthread_t p = mac_pthread_self();
int sig;
/*
* handle signals pending on threads if they are unmasked and
* SIGCANCEL only on an interruption point.
*/
if (!pthread_siggeset2set(&p->mask, &p->pending))
for (sig = 1; sig < NNSIG; sig++)
if (sigismember(&p->pending, sig) && !sigismember(&p->mask, sig) &&
(sig != SIGCANCEL || p->state & T_INTR_POINT)) {
sigdelset(&p->pending, sig);
handle_thread_signal(p, sig, p->sig_info[sig].si_code);
}
/*
* handle signals pending on process
*/
if (!pthread_siggeset2set(&p->mask, &pending_signals))
for (sig = 1; sig < NNSIG; sig++)
if (sigismember(&pending_signals, sig) && !sigismember(&p->mask, sig)) {
sigdelset(&pending_signals, sig);
handle_thread_signal(p, sig, pending_code[sig]);
}
}
#ifndef STAND_ALONE
/*------------------------------------------------------------*/
/*
* sighandler - wrapper for all signals, defers signals for later processing
* Notice: All maskable signals are caught and re-multiplexed by Pthreads.
*/
#ifdef SVR4
static void sighandler(sig, sip, scp)
struct siginfo *sip;
#else !SVR4
static void sighandler(sig, code, scp, addr)
int code;
char *addr;
#endif !SVR4
int sig;
struct context_t *scp;
{
register pthread_t p = mac_pthread_self();
#ifdef SVR4
sigset_t omask;
greg_t opc = scp->sc_pc;
greg_t osp = scp->sc_sp;
#endif SVR4
#if defined(STACK_CHECK) && defined(SIGNAL_STACK)
void pthread_io_end();
SIGSTACK_T ss;
if (sig == SIGILL || sig == SIGBUS || sig == SIGSEGV) {
if (p != NO_PTHREAD && !(p->state & T_LOCKED))
default_action(sig);
if (((sig == SIGILL && SCP_CODE == ILL_STACK) ||
#ifdef SVR4
(sig == SIGSEGV &&
(FC_CODE(SCP_CODE) == FC_HWERR || FC_CODE(SCP_CODE) == FC_ALIGN)) ||
#else
(sig == SIGSEGV && FC_CODE(SCP_CODE) == FC_PROT) ||
#endif
(sig == SIGBUS && BUS_CODE(SCP_CODE) == BUS_OBJERR)) &&
p != NO_PTHREAD &&
((scp->sc_sp < PA(p->stack_base+3*pthread_page_size))
#ifdef IO
|| (scp->sc_pc > (int)read && scp->sc_pc < (int)pthread_io_end)
#endif
))
pthread_unlock_stack(p);
switch_stacks(scp->sc_sp);
#ifndef SVR4
/*
* Let UNIX know that we are not on signal stack anymore. Does not work
* for SVR4, does work for BSD.
*/
if (SCP_ONSTACK)
scp->sc_onstack = FALSE;
else {
ss.ss_sp = (char *) SA((int) pthread_tempstack_top - STACK_OFFSET);
scp->sc_onstack = FALSE;
if (SIGSTACK(&ss, (SIGSTACK_T *) NULL))
#ifdef DEBUG
fprintf(stderr,
"Pthreads: Could not specify signal stack, errno %d\n", errno)
#endif DEBUG
;
}
#endif !SVR4
}
#endif STACK_CHECK && SIGNAL_STACK
#ifdef SVR4
pthread_sigcpyset2set(&omask, &scp->sc_mask);
#endif SVR4
/*
* add signal to queue of pending signals
*/
#ifdef DEBUG
#ifndef IO
#ifdef DEF_RR
if (sig != SIGALRM)
#endif
fprintf(stderr, "signal %d caught\n", sig);
#endif
#endif
if (!sigismember(&new_signals, sig))
if (!is_in_kernel && scp) {
SET_KERNEL_FLAG;
SIGPROCMASK(SIG_UNBLOCK, &all_signals, (struct sigset_t *) NULL);
/*
* Associate UNIX context with current thread
*/
p->scp = p->nscp;
p->nscp = scp;
/*
* A signal is pending so that the signal dispatcher calls
* pthread_handle_one_process_signals()
* and then switches to highest priority thread. Thus, we act as if no
* context switch is initiated, i.e. as if we switched back to thread
* which was running when the signal came in.
*/
#ifndef C_CONTEXT_SWITCH
pthread_sched_wrapper(sig, SCP_CODE);
#else
p->errno = errno;
pthread_sched_wrapper(sig, SCP_CODE, p);
errno = p->errno;
CLEAR_KERNEL_FLAG;
#endif
}
else {
sigaddset(&new_signals, sig);
new_code[sig] = SCP_CODE;
state_change = TRUE;
}
#ifdef SVR4
/*
* Solaris does not seem to have a _sigtramp() wrapper for signal handlers.
* In fact, it is not quite understood how the signal mask is reset after
* returning from the signal handler. This is fixed by explicitly setting
* the context.
*/
setcontext((struct ucontext *) scp);
#endif SVR4
}
#endif !STAND_ALONE
/*------------------------------------------------------------*/
/*
* pthread_init_signals - initialize signal package
*/
void pthread_init_signals()
{
int sig;
struct sigaction vec;
#if defined(STACK_CHECK) && defined(SIGNAL_STACK)
SIGSTACK_T ss;
ss.ss_sp = (char *) SA((int) pthread_tempstack_top - STACK_OFFSET);
CLR_SS_ONSTACK;
#ifndef STAND_ALONE
if (SIGSTACK(&ss, (SIGSTACK_T *) NULL))
#ifdef DEBUG
fprintf(stderr,
"Pthreads: Could not specify signal stack, errno %d\n", errno)
#endif DEBUG
;
#endif !STAND_ALONE
#endif
/*
* initialize kernel structure
*/
is_in_kernel = is_updating_timer = FALSE;
sigemptyset(&synchronous);
sigaddset(&synchronous, SIGILL);
sigaddset(&synchronous, SIGABRT);
sigaddset(&synchronous, SIGEMT);
sigaddset(&synchronous, SIGFPE);
sigaddset(&synchronous, SIGBUS);
sigaddset(&synchronous, SIGSEGV);
sigemptyset((sigset_t *) &new_signals);
sigemptyset(&pending_signals);
sigemptyset(&handlerset);
sigemptyset(&sig_handling);
sigemptyset(&cantmask);
sigaddset(&cantmask, SIGKILL);
sigaddset(&cantmask, SIGSTOP);
sigaddset(&cantmask, SIGCANCEL);
sigfillset(&all_signals);
sigdelset(&all_signals, SIGKILL);
sigdelset(&all_signals, SIGSTOP);
pthread_queue_init(&ready);
pthread_queue_init(&all);
#ifdef STAND_ALONE
pthread_timer_queue_init(pthread_timer);
#else
pthread_timer_queue_init(&pthread_timer);
#endif
set_warning = "CAUTION: entering kernel again\n";
clear_warning = "CAUTION: leaving kernel again\n";
prio_warning = "CAUTION: prio violation when leaving kernel\n";
#ifdef RAND_SWITCH
srandom(1);
pthread_n_ready = 0;
#endif
#ifdef STAND_ALONE
sigemptyset(&proc_mask);
cur_heap = (int)(&heap_start);
#else !STAND_ALONE
/*
* no signal requests
*/
for (sig = 0; sig < NNSIG; sig++) {
pthread_user_handler[sig].sa_handler = SIG_DFL;
sigemptyset(&pthread_user_handler[sig].sa_mask);
pthread_user_handler[sig].sa_flags = 0;
new_code[sig] = 0;
}
/*
* install universal signal handler for all signals
* except for those which cannot be masked
*/
vec.sa_handler = sighandler;
pthread_sigcpyset2set(&vec.sa_mask, &all_signals);
vec.sa_flags = SA_SIGINFO | SA_RESTART;
for (sig = 1; sig < NSIG; sig++)
if (!sigismember(&cantmask, sig)) {
#if defined(STACK_CHECK) && defined(SIGNAL_STACK)
if (sig == SIGBUS || sig == SIGILL || sig == SIGSEGV)
vec.sa_flags |= SA_ONSTACK;
#endif
if (SIGACTION(sig, &vec, (struct sigaction *) NULL))
#ifdef DEBUG
fprintf(stderr, "Pthreads (signal): \
Could not install handler for signal %d\n", sig)
#endif
;
#if defined(STACK_CHECK) && defined(SIGNAL_STACK)
if (sig == SIGBUS || sig == SIGILL || sig == SIGSEGV)
vec.sa_flags &= ~SA_ONSTACK;
#endif
}
#endif STAND_ALONE
#ifdef IO
gwidth = 0;
FD_ZERO(&greadfds);
FD_ZERO(&gwritefds);
FD_ZERO(&gexceptfds);
#endif IO
}
/*------------------------------------------------------------*/
/*
* sigwait - suspend thread until signal becomes pending
* Return: signal number if o.k., otherwise -1
* Notice: cannot mask SIGKILL, SIGSTOP, SIGCANCEL
*/
int sigwait(set)
sigset_t *set;
{
register int sig;
sigset_t new, more;
register pthread_t p = mac_pthread_self();
pthread_sigcpyset2set(&new, set);
pthread_sigdelset2set(&new, &cantmask);
pthread_sigcpyset2set(&more, &new);
SET_KERNEL_FLAG;
/*
* Are the signals in set blocked by the current thread?
*/
if (!pthread_siggeset2set(&p->mask, &more)) {
set_errno(EINVAL);
CLEAR_KERNEL_FLAG;
return(-1);
}
/*
* Any of the signals in set pending on thread?
*/
if (pthread_sigismemberset2set(&p->pending, &more))
for (sig = 1; sig < NNSIG; sig++)
if (sigismember(&p->pending, sig) && sigismember(&more, sig)) {
sigdelset(&p->pending, sig);
CLEAR_KERNEL_FLAG;
return(sig);
}
/*
* Any of the signals in set pending on process?
*/
if (pthread_sigismemberset2set(&pending_signals, &more))
for (sig = 1; sig < NNSIG; sig++)
if (sigismember(&pending_signals, sig) && sigismember(&more, sig)) {
sigdelset(&pending_signals, sig);
CLEAR_KERNEL_FLAG;
return(sig);
}
/*
* suspend thread and wait for any of the signals
*/
pthread_sigaddset2set(&p->sigwaitset, &new);
pthread_sigdelset2set(&p->mask, &new);
p->state &= ~T_RUNNING;
p->state |= T_SIGWAIT | T_BLOCKED | T_INTR_POINT;
if (sigismember(&p->pending, SIGCANCEL) && !sigismember(&p->mask, SIGCANCEL))
SIG_CLEAR_KERNEL_FLAG(TRUE);
else {
pthread_q_deq_head(&ready, PRIMARY_QUEUE);
SIM_SYSCALL(TRUE);
CLEAR_KERNEL_FLAG;
}
if (get_errno() == EINTR)
return(-1);
/*
* determine the received signal
*/
for (sig = 1; sig < NNSIG; sig++)
if (sigismember(&new, sig) && !sigismember(&p->sigwaitset, sig))
break;
/*
* Clear signal mask
*/
SET_KERNEL_FLAG;
sigemptyset(&p->sigwaitset);
CLEAR_KERNEL_FLAG;
/*
* If no signal woke us up directly, a user handler (hence interrupt) must
* have been activated via a different signal.
*/
if (sig < 0) {
set_errno(EINVAL);
return(-1);
}
return(sig);
}
/*------------------------------------------------------------*/
/*
* sigprocmask - change or examine signal mask of thread
* return old mask or -1 if error
* cannot mask SIGKILL, SIGSTOP, SIGCANCEL
*/
int sigprocmask(how, set, oset)
int how;
const sigset_t *set;
sigset_t *oset;
{
sigset_t new, old, pending;
register pthread_t p = mac_pthread_self();
register already_in_kernel = is_in_kernel;
#if defined(SOLARIS) && defined(IO)
if (pthread_get_retaddr() < 0) /* call from dynamic library */
return(pthread_p_sigprocmask(how, set, oset));
#endif SOLARIS && IO
if (!already_in_kernel)
SET_KERNEL_FLAG;
pthread_sigcpyset2set(&old, &p->mask);
if (oset)
pthread_sigcpyset2set(oset, &old);
if (!set) {
if (!already_in_kernel)
CLEAR_KERNEL_FLAG;
return(0);
}
pthread_sigcpyset2set(&new, set);
pthread_sigdelset2set(&new, &cantmask);
switch (how) {
case SIG_BLOCK:
pthread_sigaddset2set(&p->mask, &new);
break;
case SIG_UNBLOCK:
pthread_sigdelset2set(&p->mask, &new);
break;
case SIG_SETMASK:
pthread_sigcpyset2set(&p->mask, &new);
break;
default:
set_errno(EINVAL);
if (!already_in_kernel)
CLEAR_KERNEL_FLAG;
return(-1);
}
pthread_sigcpyset2set(&pending, &p->pending);
pthread_sigaddset2set(&pending, &pending_signals);
pthread_sigdelset2set(&old, &p->mask);
if (pthread_sigismemberset2set(&pending, &old))
SIG_CLEAR_KERNEL_FLAG(TRUE);
else {
SIM_SYSCALL(TRUE);
if (!already_in_kernel)
CLEAR_KERNEL_FLAG;
}
return(0);
}
/*------------------------------------------------------------*/
/*
* sigpending - inquire about pending signals which are blocked, i.e. applies
* to only those signals which are explicitly pending on the current thread
* return 0 if o.k., -1 otherwise
*/
int sigpending(set)
sigset_t *set;
{
if (!set) {
set_errno(EINVAL);
return(-1);
}
SIM_SYSCALL(TRUE);
pthread_sigcpyset2set(set, &mac_pthread_self()->pending);
pthread_sigaddset2set(set, &pending_signals);
pthread_sigdelset2set(set, &cantmask);
return(0);
}
/*------------------------------------------------------------*/
/*
* sigsuspend - suspend thread,
* set replaces the masked signals for the thread temporarily,
* suspends thread, and resumes execution when a user handler is invoked
* Return: -1 and EINTR if interrupted or EINVAL if wrong parameters
* Notice: cannot mask SIGKILL, SIGSTOP
*/
int sigsuspend(set)
SIGSUSPEND_CONST sigset_t *set;
{
register int sig;
sigset_t old, pending;
register pthread_t p = mac_pthread_self();
if (!set) {
set_errno(EINVAL);
return(-1);
}
SET_KERNEL_FLAG;
pthread_sigcpyset2set(&old, &p->mask);
pthread_sigcpyset2set(&p->mask, set);
pthread_sigdelset2set(&p->mask, &cantmask);
pthread_sigaddset2set(&p->sigwaitset, &p->mask);
p->state &= ~T_RUNNING;
p->state |= T_SIGSUSPEND | T_BLOCKED | T_INTR_POINT;
if (sigismember(&p->pending, SIGCANCEL) && !sigismember(&p->mask, SIGCANCEL))
SIG_CLEAR_KERNEL_FLAG(TRUE);
else {
pthread_q_deq_head(&ready, PRIMARY_QUEUE);
SIM_SYSCALL(TRUE);
CLEAR_KERNEL_FLAG;
}
/*
* restore the initial signal mask
*/
SET_KERNEL_FLAG;
pthread_sigcpyset2set(&p->mask, &old);
pthread_sigcpyset2set(&pending, &p->pending);
pthread_sigaddset2set(&pending, &pending_signals);
pthread_sigcpyset2set(&old, set);
pthread_sigdelset2set(&old, &cantmask);
pthread_sigdelset2set(&old, &p->mask);
if (pthread_sigismemberset2set(&pending, &old))
SIG_CLEAR_KERNEL_FLAG(TRUE);
else
CLEAR_KERNEL_FLAG;
return(-1);
}
/*------------------------------------------------------------*/
/*
* pause - suspend thread until any signal is caught,
* same as sigsuspend except that the signal mask doesn't change
*/
int pause()
{
register int sig;
register pthread_t p = mac_pthread_self();
SET_KERNEL_FLAG;
pthread_sigcpyset2set(&p->sigwaitset, &p->mask);
p->state &= ~T_RUNNING;
p->state |= T_SIGSUSPEND | T_BLOCKED | T_INTR_POINT;
if (sigismember(&p->pending, SIGCANCEL) && !sigismember(&p->mask, SIGCANCEL))
SIG_CLEAR_KERNEL_FLAG(TRUE);
else {
pthread_q_deq_head(&ready, PRIMARY_QUEUE);
SIM_SYSCALL(TRUE);
CLEAR_KERNEL_FLAG;
}
return(-1);
}
#ifdef STAND_ALONE
/*------------------------------------------------------------*/
/*
* pthread_timed_sigwait - suspend running thread until specified time
* Return -1 if error, 0 otherwise
* assumes SET_KERNEL_FLAG
*/
int pthread_timed_sigwait(p, timeout, mode, func, arg)
pthread_t p;
struct timespec *timeout;
int mode;
pthread_func_t func;
any_t arg;
{
struct timespec now;
if (!timeout || timeout->tv_sec < 0 ||
timeout->tv_nsec < 0 || timeout->tv_nsec >= 1000000000) {
set_errno(EINVAL);
return(-1);
}
clock_gettime(CLOCK_REALTIME, &now);
if (mode == ABS_TIME) {
if (GTEQ_NTIME(now, *timeout)) {
set_errno(EAGAIN);
return(-1);
}
}
else if (mode == REL_TIME) {
if (LE0_NTIME(*timeout)) {
set_errno(EAGAIN);
return(-1);
}
PLUS_NTIME(*timeout, now, *timeout);
}
else {
set_errno(EINVAL);
return(-1);
}
/*
* queue up current thread on the timer queue
*/
p->timer_func = func;
p->timer_arg = arg;
p->tp.tv_sec = timeout->tv_sec;
p->tp.tv_nsec = timeout->tv_nsec;
pthread_q_timed_enq(&pthread_timer, p);
return(0);
}
/*------------------------------------------------------------*/
/*
* pthread_cancel_timed_sigwait - dequeue thread waiting on timer only
* "signaled" indicates if the thread "p" received a SIGALRM
* Notice: set error in both thread structure and global UNIX errno
* since this may be called from pthread_handle_many_process_signals
* assumes SET_KERNEL_FLAG
*/
int pthread_cancel_timed_sigwait(first_p, signaled, mode, activate)
pthread_t first_p;
int signaled, mode, activate;
{
pthread_t p = first_p;
struct timespec now;
int time_read = 0;
if (pthread_timer == NO_PTHREAD) {
is_updating_timer = FALSE;
return(0);
}
do {
#ifdef TRASH
if (!p->timer_func) {
#endif
if (p->state & T_CONDTIMER) {
p->state &= ~(T_CONDTIMER | T_SYNCTIMER);
pthread_q_deq(p->queue, p, PRIMARY_QUEUE);
pthread_q_timed_wakeup_thread(&pthread_timer, p,
(p == first_p ? activate : TRUE));
if (p != first_p || signaled) {
p->errno = EAGAIN;
if (p == mac_pthread_self())
set_errno(EAGAIN);
}
}
else {
p->state &= ~T_SYNCTIMER;
pthread_q_timed_wakeup_thread(&pthread_timer, p,
(p == first_p ? activate : TRUE));
}
#ifdef TRASH
}
else
(*p->timer_func)(p->timer_arg);
#endif
p = pthread_timer;
if (p != NO_PTHREAD && !time_read) {
clock_gettime(CLOCK_REALTIME, &now);
time_read++;
}
} while (p && GTEQ_NTIME(now, p->tp));
is_updating_timer = FALSE;
/*
* timer signal received while cancelling => ignore it
*/
if (!signaled && (sigismember(&new_signals, SIGALRM)))
sigdelset(&new_signals, SIGALRM);
return(0);
}
#else !STAND_ALONE
/*------------------------------------------------------------*/
/*
* pthread_timed_sigwait - suspend running thread until specified time
* Return -1 if error, 0 otherwise
* assumes SET_KERNEL_FLAG
*/
int pthread_timed_sigwait(p, timeout, mode, func, arg)
pthread_t p;
struct timespec *timeout;
int mode;
pthread_func_t func;
any_t arg;
{
struct itimerval it;
struct timeval now, in;
register timer_ent_t phead;
#ifdef TIMER_DEBUG
fprintf(stderr, "timed_sigwait: enter\n");
#endif
#ifdef DEF_RR
if (p->num_timers >= TIMER_MAX) {
set_errno(EAGAIN);
return(-1);
}
if (mode != RR_TIME) {
#endif
if (!timeout || timeout->tv_nsec < 0 || timeout->tv_nsec >= 1000000000 ||
gettimeofday(&now, (struct timezone *) NULL)) {
set_errno(EINVAL);
return(-1);
}
P2U_TIME(in, (*timeout));
#ifdef DEF_RR
}
#endif
it.it_interval.tv_sec = it.it_interval.tv_usec = 0;
if (mode == ABS_TIME) {
/*
* time has already passed
*/
if (GTEQ_TIME(now, in)) {
set_errno(EAGAIN);
return(-1);
}
MINUS_TIME(it.it_value, in, now);
}
else if (mode == REL_TIME) {
/*
* time has already passed
*/
if (LE0_TIME(in)) {
set_errno(EAGAIN);
return(-1);
}
it.it_value.tv_sec = in.tv_sec;
it.it_value.tv_usec = in.tv_usec;
PLUS_TIME(in, in, now);
}
#ifdef DEF_RR
else if (mode == RR_TIME) {
p->state |= T_ASYNCTIMER;
if ((p->interval.tv_sec == 0) && (p->interval.tv_usec == 0)) {
in.tv_sec = it.it_value.tv_sec = 0;
in.tv_usec = it.it_value.tv_usec = TIME_SLICE;
}
else {
in.tv_sec = it.it_value.tv_sec = p->interval.tv_sec;
in.tv_usec = it.it_value.tv_usec = p->interval.tv_usec;
}
}
#endif
else {
set_errno(EINVAL);
return(-1);
}
#ifdef DEF_RR
p->num_timers++;
#endif
/*
* if no timer set, set timer to current request; otherwise
* overwrite timer if current request needs to be served next
*/
if (!(phead = pthread_timer.head) || GT_TIME(phead->tp, in)) {
if (setitimer(ITIMER_REAL, &it, (struct itimerval *) NULL)) {
#ifdef DEBUG
fprintf(stderr, "ERROR: setitimer in timed_sigwait\n");
#endif
set_errno(EINVAL);
return(-1);
}
#ifdef TIMER_DEBUG
fprintf(stderr, "timed_sigwait: setitimer %d.%d sec.usec\n",
it.it_value.tv_sec, it.it_value.tv_usec);
#endif
}
#ifdef TIMER_DEBUG
else
fprintf(stderr, "timed_sigwait: timer not set up, pthread_timer.head=%x\n", phead);
#endif
/*
* queue up current thread on the timer queue
*/
pthread_q_timed_enq(&pthread_timer,
in,
func == (pthread_func_t) pthread_exit ? DL_TIME : mode,
p);
return(0);
}
/*------------------------------------------------------------*/
/*
* pthread_cancel_timed_sigwait - dequeue thread waiting for alarm only
* "signaled" indicates if the thread "p" received a SIGALRM
* Notice: set error in both thread structure and global UNIX errno
* since this may be called from pthread_handle_many_process_signals
* assumes SET_KERNEL_FLAG
*/
int pthread_cancel_timed_sigwait(first_p, signaled, mode, activate)
pthread_t first_p;
int signaled, mode, activate;
{
pthread_t p = first_p;
timer_ent_t tmr;
timer_ent_t old_tmr_head = pthread_timer.head;
struct itimerval it;
struct timeval now;
int time_read = FALSE;
/*
* find the first instance of this thread in timer queue
*/
#ifdef DEF_RR
for (tmr = pthread_timer.head; tmr; tmr = tmr->next[TIMER_QUEUE])
if (tmr->thread == p && (tmr->mode & mode))
break;
#else
tmr = p;
#endif
if (!tmr) {
#ifdef TIMER_DEBUG
fprintf(stderr, "pthread_cancel_timed_sigwait: exit0\n");
#endif
return(0);
}
/*
* for each occurrence, remove the timer entry
*/
do {
if (p->state & T_CONDTIMER) {
p->state &= ~(T_CONDTIMER | T_SYNCTIMER);
pthread_q_deq(p->queue, p, PRIMARY_QUEUE);
pthread_q_timed_wakeup_thread(&pthread_timer, p,
(p == first_p ? activate : TRUE));
if (p != first_p || signaled) {
p->errno = EAGAIN;
if (p == mac_pthread_self())
set_errno(EAGAIN);
}
}
else {
#ifdef DEF_RR
p->state &= ~((tmr->mode & RR_TIME) ? T_ASYNCTIMER : T_SYNCTIMER);
#else
p->state &= ~T_SYNCTIMER;
#endif
pthread_q_timed_wakeup_thread(&pthread_timer, p,
(p == first_p ? activate : TRUE));
}
#ifdef DEF_RR
/*
* find next instance of this thread in timer queue
*/
if (mode == ALL_TIME) {
tmr = pthread_timer.head;
while (tmr && tmr->thread != p)
tmr = tmr->next[TIMER_QUEUE];
}
else
tmr = NO_TIMER;
/*
* check if head of timer queue can be woken up, i.e. now > tmr->tp
*/
if (tmr == NO_TIMER && !time_read)
#else
if (!time_read)
#endif
if (gettimeofday(&now, (struct timezone *) NULL)) {
#ifdef TIMER_DEBUG
fprintf(stderr, "pthread_cancel_timed_sigwait: exit1\n");
#endif
set_errno(EINVAL);
return(-1);
}
else
time_read = TRUE;
if (time_read) {
tmr = pthread_timer.head;
#ifdef DEF_RR
if (tmr)
p = tmr->thread;
#else
p = tmr;
#endif
}
} while (tmr && (!time_read || GTEQ_TIME(now, tmr->tp)));
/*
* if head of timer queue hasn't changed, no action
*/
if (tmr == old_tmr_head) {
#ifdef TIMER_DEBUG
fprintf(stderr, "pthread_cancel_timed_sigwait: exit2\n");
#endif
return(0);
}
/*
* overwrite timer if current request needs to be served next or invalidate
*/
if (tmr != NO_TIMER)
MINUS_TIME(it.it_value, tmr->tp, now);
else
it.it_value.tv_sec = it.it_value.tv_usec = 0;
it.it_interval.tv_sec = it.it_interval.tv_usec = 0;
if (setitimer(ITIMER_REAL, &it, (struct itimerval *) NULL)) {
#ifdef DEBUG
fprintf(stderr, "ERROR: setitimer in pthread_cancel_timed_sigwait\n");
#endif
set_errno(EINVAL);
return(-1);
}
#ifdef TIMER_DEBUG
last_alarm.tv_sec = it.it_value.tv_sec;
last_alarm.tv_usec = it.it_value.tv_usec;
#endif
#ifdef TIMER_DEBUG
fprintf(stderr, "pthread_cancel_timed_sigwait: setitimer %d.%d sec.usec\n",
it.it_value.tv_sec, it.it_value.tv_usec);
#endif
return(0);
}
#endif STAND_ALONE
/*------------------------------------------------------------*/
/*
* raise - send a signal to the current process;
* NOT directed to any particular thread,
* any thread waiting for the signal may pick it up.
*/
int raise(sig)
int sig;
{
SET_KERNEL_FLAG;
SIM_SYSCALL(TRUE);
#ifdef C_CONTEXT_SWITCH
if (!SAVE_CONTEXT(mac_pthread_self()))
#endif
pthread_signal_sched(sig, (int) NO_PTHREAD);
#ifdef C_CONTEXT_SWITCH
CLEAR_KERNEL_FLAG;
#endif
return(0);
}
/*------------------------------------------------------------*/
/*
* pthread_kill - send signal to thread
*/
int pthread_kill(thread, sig)
pthread_t thread;
int sig;
{
if (thread == NO_PTHREAD || thread->state & T_RETURNED ||
sigismember(&cantmask, sig) || !sigismember(&all_signals, sig)) {
set_errno(EINVAL);
return(-1);
}
/*
* queue up signal associated with thread
*/
SET_KERNEL_FLAG;
SIM_SYSCALL(TRUE);
#ifdef C_CONTEXT_SWITCH
if (!SAVE_CONTEXT(mac_pthread_self()))
#endif
pthread_signal_sched(sig, (int) thread);
#ifdef C_CONTEXT_SWITCH
CLEAR_KERNEL_FLAG;
#endif
return(0);
}
/*------------------------------------------------------------*/
/*
* pthread_cancel - cancel thread
* Open question: is this an interruption point if a thread cancels itself?
* As of now, it isn't!
*/
int pthread_cancel(thread)
pthread_t thread;
{
if (thread == NO_PTHREAD || thread->state & T_RETURNED) {
set_errno(EINVAL);
return(-1);
}
/*
* queue up signal associated with thread
*/
SET_KERNEL_FLAG;
SIM_SYSCALL(TRUE);
#ifdef C_CONTEXT_SWITCH
if (!SAVE_CONTEXT(mac_pthread_self()))
#endif
pthread_signal_sched(SIGCANCEL, (int) thread);
#ifdef C_CONTEXT_SWITCH
CLEAR_KERNEL_FLAG;
#endif
return(0);
}
/*------------------------------------------------------------*/
/*
* pthread_setintr - set interruptablility state for thread cancellation
*/
int pthread_setintr(state)
int state;
{
int old;
sigset_t new;
if (state != PTHREAD_INTR_ENABLE && state != PTHREAD_INTR_DISABLE) {
set_errno(EINVAL);
return(-1);
}
SIM_SYSCALL(TRUE);
old = (sigismember(&mac_pthread_self()->mask, SIGCANCEL) ?
PTHREAD_INTR_DISABLE : PTHREAD_INTR_ENABLE);
sigemptyset(&new);
sigaddset(&new, SIGCANCEL);
if (sigprocmask(state, &new, (sigset_t *) NULL) == 0)
return(old);
else
return(-1);
}
/*------------------------------------------------------------*/
/*
* pthread_setintrtype - set interruptablility type for thread cancellation
*/
int pthread_setintrtype(type)
int type;
{
register pthread_t p = mac_pthread_self();
int old;
old = (p->state & T_CONTROLLED ?
PTHREAD_INTR_CONTROLLED : PTHREAD_INTR_ASYNCHRONOUS);
switch (type) {
case PTHREAD_INTR_CONTROLLED:
SET_KERNEL_FLAG;
p->state |= T_CONTROLLED;
SIM_SYSCALL(TRUE);
CLEAR_KERNEL_FLAG;
return(old);
case PTHREAD_INTR_ASYNCHRONOUS:
SET_KERNEL_FLAG;
p->state &= ~T_CONTROLLED;
if (sigismember(&p->pending, SIGCANCEL) &&
!sigismember(&p->mask, SIGCANCEL)) {
p->state |= T_INTR_POINT;
SIG_CLEAR_KERNEL_FLAG(TRUE);
}
else {
SIM_SYSCALL(TRUE);
CLEAR_KERNEL_FLAG;
}
return(old);
default:
set_errno(EINVAL);
return(-1);
}
}
/*------------------------------------------------------------*/
/*
* pthread_testintr - act upon pending cancellation (creates interruption point)
*/
void pthread_testintr()
{
register pthread_t p = mac_pthread_self();
SET_KERNEL_FLAG;
if (sigismember(&p->pending, SIGCANCEL) &&
!sigismember(&p->mask, SIGCANCEL)) {
p->state |= T_INTR_POINT;
SIG_CLEAR_KERNEL_FLAG(TRUE);
}
else {
SIM_SYSCALL(TRUE);
CLEAR_KERNEL_FLAG;
}
}
/*------------------------------------------------------------*/
/*
* sigaction - install interrupt handler for a thread on a signal
* return 0 if o.k., -1 otherwise
* Notice: cannot mask SIGKILL, SIGSTOP, SIGCANCEL
*/
int sigaction(sig, act, oact)
int sig;
const struct sigaction *act;
struct sigaction *oact;
{
register pthread_t p = mac_pthread_self();
struct sigaction vec;
#if defined(SOLARIS) && defined(IO)
/*
* Handling for asynchronous process level IO under Solaris.
* Under certain conditions, call sigaction() in libaio.so.
* The aio library seems to contain some bugs. When stdin and stdout
* are pending, the stdin seems to lock stdout at some point, until
* it gets more input. This may be a result of mutually exclusive file
* access under Solaris through mutexes. As a result, the Pthreads library
* can no longer guarantee that the process continues to execute while
* there is a runnable thread. It tries to do its best until Solaris
* gets the upper hand and blocks an otherwise runnable LWP performing IO.
* This problem may not occur when the accessed devices are distict.
* (Notice that stdin requires an echo, this accessing stdout. Thus, the
* two devices are not necessarily distinct.)
* This problem does not occur under SunOS 4.1.x.
*/
void *handle;
extern int pthread_started;
int (*clib_sigaction)
_C_PROTOTYPE((int __sig,
const struct sigaction *__act,
struct sigaction *__oact));
if (!pthread_started && sig == SIGIO || /* called from pthread_init_signals */
pthread_get_retaddr() < 0) { /* call from dynamic library */
handle = dlopen("libaio.so", RTLD_LAZY);
clib_sigaction = (int (*) _C_PROTOTYPE((int __sig,
const struct sigaction *__act,
struct sigaction *__oact)))
dlsym(handle, "sigaction");
return((*clib_sigaction)(sig, act, oact));
}
#endif SOLARIS && IO
if (sigismember(&cantmask, sig) || !sigismember(&all_signals, sig)) {
set_errno(EINVAL);
return(-1);
}
if (!act) {
if (oact)
*oact = pthread_user_handler[sig];
return(0);
}
if (pthread_sigismemberset2set(&act->sa_mask, &cantmask)) {
set_errno(EINVAL);
return(-1);
}
SET_KERNEL_FLAG;
if (oact)
*oact = pthread_user_handler[sig];
pthread_user_handler[sig] = *act;
/*
* queue up mac_pthread_self() in the signal queue indicated
*/
if (!sigismember(&handlerset, sig))
sigaddset(&handlerset, sig);
/*
* dequeue pending signals on process and threads if to be ignored
* or perform default action on process if default action chosen
*/
if (act->sa_handler == SIG_IGN || act->sa_handler == SIG_DFL) {
if (sigismember(&pending_signals, sig)) {
sigdelset(&pending_signals, sig);
if (act->sa_handler == SIG_DFL)
default_action(sig);
}
for (p = all.head; p; p = p->next[ALL_QUEUE])
if (sigismember(&p->pending, sig)) {
sigdelset(&p->pending, sig);
if (act->sa_handler == SIG_DFL)
default_action(sig);
}
}
#ifndef STAND_ALONE
/*
* let UNIX know about sa_flags by reinstalling process handler
* for signals that have a defined sa_flags bit
*/
if (sig == SIGCHLD) {
vec.sa_handler = act->sa_handler;
vec.sa_mask = act->sa_mask;
vec.sa_flags =
#ifdef SA_NOCLDSTOP
(act->sa_flags & SA_NOCLDSTOP ? SA_NOCLDSTOP : 0);
#else
0;
#endif
SIGACTION(sig, &vec, (struct sigaction *) NULL);
}
#endif !STAND_ALONE
SIM_SYSCALL(TRUE);
CLEAR_KERNEL_FLAG;
return(0);
}
/*------------------------------------------------------------*/
/*
* pthread_absnanosleep - suspend until abs. time interval specified by "rqtp"
*/
int pthread_absnanosleep(rqtp)
const struct timespec *rqtp;
{
register pthread_t p = mac_pthread_self();
struct timespec rmtp;
#ifdef STAND_ALONE
struct timespec now;
#else
timer_ent_t tmr;
struct timeval now;
#endif
rmtp.tv_sec = rqtp->tv_sec;
rmtp.tv_nsec = rqtp->tv_nsec;
do {
SET_KERNEL_FLAG;
if (pthread_timed_sigwait(p, rmtp, ABS_TIME, NULL, NULL) == -1) {
CLEAR_KERNEL_FLAG;
return(-1);
}
/*
* clear error number before suspending
*/
set_errno(0);
p->state &= ~T_RUNNING;
p->state |= T_BLOCKED | T_SYNCTIMER | T_INTR_POINT;
if (sigismember(&p->pending, SIGCANCEL) &&
!sigismember(&p->mask, SIGCANCEL))
SIG_CLEAR_KERNEL_FLAG(TRUE);
else {
pthread_q_deq_head(&ready, PRIMARY_QUEUE);
SIM_SYSCALL(TRUE);
CLEAR_KERNEL_FLAG;
}
/*
* Check if condition was signaled or time-out was exceeded.
*/
if (get_errno() == EINTR) {
#ifdef STAND_ALONE
clock_gettime(CLOCK_REALTIME, &now);
if (pthread_timer = p)
if (GT_NTIME(p->tp, now))
MINUS_NTIME(rmtp, p->tp, now);
#else !STAND_ALONE
if (gettimeofday(&now, (struct timezone *) NULL)) {
set_errno(EINVAL);
return(-1);
}
#ifdef DEF_RR
if (tmr = pthread_timer.head) {
while (tmr->thread != p)
tmr = tmr->next[TIMER_QUEUE];
#else
if (tmr = p) {
#endif
if (GT_TIME(tmr->tp, now)) {
MINUS_TIME(now, tmr->tp, now);
U2P_TIME((rmtp), now);
}
}
#endif !STAND_ALONE
}
else
NTIMERCLEAR(rmtp);
} while (!LE0_NTIME(rmtp));
return(0);
}
/*------------------------------------------------------------*/
/*
* nanosleep - suspend until time interval specified by "rqtp" has passed
* or a user handler is invoked or the thread is canceled
*/
int nanosleep(rqtp, rmtp)
const struct timespec *rqtp;
struct timespec *rmtp;
{
register pthread_t p = mac_pthread_self();
#ifdef STAND_ALONE
struct timespec now;
#else
timer_ent_t tmr;
struct timeval now;
#endif
if (rmtp)
rmtp->tv_sec = rmtp->tv_nsec = 0;
SET_KERNEL_FLAG;
if (pthread_timed_sigwait(p, rqtp, REL_TIME, NULL, NULL) == -1) {
CLEAR_KERNEL_FLAG;
return(-1);
}
/*
* clear error number before suspending
*/
set_errno(0);
p->state &= ~T_RUNNING;
p->state |= T_BLOCKED | T_SYNCTIMER | T_INTR_POINT;
if (sigismember(&p->pending, SIGCANCEL) &&
!sigismember(&p->mask, SIGCANCEL))
SIG_CLEAR_KERNEL_FLAG(TRUE);
else {
pthread_q_deq_head(&ready, PRIMARY_QUEUE);
SIM_SYSCALL(TRUE);
CLEAR_KERNEL_FLAG;
}
/*
* Check if condition was signaled or time-out was exceeded.
*/
if (get_errno() == EINTR) {
#ifdef STAND_ALONE
clock_gettime(CLOCK_REALTIME, &now);
if (pthread_timer = p)
if (GT_NTIME(p->tp, now)) {
if (rmtp)
MINUS_NTIME(*rmtp, p->tp, now);
return(-1);
}
#else !STAND_ALONE
if (gettimeofday(&now, (struct timezone *) NULL)) {
set_errno(EINVAL);
return(-1);
}
#ifdef DEF_RR
if (tmr = pthread_timer.head) {
while (tmr->thread != p)
tmr = tmr->next[TIMER_QUEUE];
#else
if (tmr = p) {
#endif
if (GT_TIME(tmr->tp, now)) {
if (rmtp) {
MINUS_TIME(now, tmr->tp, now);
U2P_TIME((*rmtp), now);
}
return(-1);
}
}
#endif STAND_ALONE
}
return(0);
}
/*------------------------------------------------------------*/
/*
* sleep - suspend thread for time interval (in seconds)
*/
unsigned int sleep(seconds)
unsigned int seconds;
{
struct timespec rqtp, rmtp;
if (rqtp.tv_sec = seconds) {
rqtp.tv_nsec = 0;
nanosleep(&rqtp, &rmtp);
if (get_errno() == EINTR)
return(rmtp.tv_sec + (rmtp.tv_nsec ? 1 : 0)); /* pessimistic round-up */
}
return(0);
}
#ifdef STAND_ALONE
/*------------------------------------------------------------*/
/*
* clock_gettime - reads the clock
*/
int clock_gettime(clock_id, tp)
int clock_id;
struct timespec *tp;
{
time_t secs;
unsigned int nsecs;
if (clock_id != CLOCK_REALTIME || !tp) {
set_errno(EINVAL);
return(-1);
}
do {
do {
tp->tv_sec = timeofday.tv_sec;
tp->tv_nsec = timeofday.tv_nsec;
secs = timeofday.tv_sec;
} while (tp->tv_sec != secs);
nsecs = (COUNTER->counter14 & CTR_USEC_MASK) >> CTR_USEC_SHIFT;
} while (tp->tv_nsec != timeofday.tv_nsec);
tp->tv_nsec += nsecs * 1000;
return(0);
}
#else !STAND_ALONE
/*------------------------------------------------------------*/
/*
* clock_gettime - reads the clock
*/
int clock_gettime(clock_id, tp)
int clock_id;
struct timespec *tp;
{
struct timeval now;
if (clock_id != CLOCK_REALTIME || !tp ||
gettimeofday(&now, (struct timezone *) NULL)) {
set_errno(EINVAL);
return(-1);
}
U2P_TIME((*tp), now);
return(0);
}
#endif STAND_ALONE
#ifdef IO
/*-------------------------------------------------------------*/
/*
* fds_update - update fields of file descriptors
*/
static int fds_update(lrfds, lwfds, lefds, grfds, gwfds, gefds, width)
fd_set *lrfds, *lwfds, *lefds, *grfds, *gwfds, *gefds;
int width;
{
int i;
int result = 0;
fd_set crfds, cwfds, cefds, gfds;
FD_ZERO(&crfds);
FD_ZERO(&cwfds);
FD_ZERO(&cefds);
for (i=0; i < howmany(width, NFDBITS); i++){
crfds.fds_bits [i] = (*lrfds).fds_bits [i] & (*grfds).fds_bits [i];
cwfds.fds_bits [i] = (*lwfds).fds_bits [i] & (*gwfds).fds_bits [i];
cefds.fds_bits [i] = (*lefds).fds_bits [i] & (*gefds).fds_bits [i];
gfds.fds_bits [i] = crfds.fds_bits [i] | cwfds.fds_bits [i] |
cefds.fds_bits [i];
}
for (i=0; i < width; i++)
if FD_ISSET (i, &gfds)
result++;
if (result > 0) {
*lrfds = crfds;
*lwfds = cwfds;
*lefds = cefds;
}
return(result);
}
/*-------------------------------------------------------------*/
/*
* fds_subtract - delete fields of file descriptors
*/
static void fds_subtract(l,r, width)
struct fd_set *l,*r;
int width;
{
int i;
for (i=0; i < howmany(width, NFDBITS); i++)
(*l).fds_bits[i] = (*l).fds_bits[i] & ~(*r).fds_bits[i];
}
/*-------------------------------------------------------------*/
/*
* aio_handle - handler for asynchronous I/O on process level,
* demultiplexes SIGIO for threads
*/
static int aio_handle()
{
pthread_t p;
int count = 0;
#ifndef SVR4
static struct timeval mintimeout = {0, 0};
fd_set creadfds, cwritefds, cexceptfds;
int cwidth = gwidth;
int flag;
if (gwidth > 0) {
pthread_fds_set(&creadfds, &greadfds);
pthread_fds_set(&cwritefds, &gwritefds);
pthread_fds_set(&cexceptfds, &gexceptfds);
if (SELECT(gwidth, &creadfds, &cwritefds, &cexceptfds, &mintimeout) > 0) {
fds_subtract(&greadfds, &creadfds, gwidth);
fds_subtract(&gwritefds, &cwritefds, gwidth);
fds_subtract(&gexceptfds, &cexceptfds, gwidth);
while (gwidth != 0 &&
!FD_ISSET(gwidth-1, &greadfds) &&
!FD_ISSET(gwidth-1, &gwritefds) &&
!FD_ISSET(gwidth-1, &gexceptfds))
gwidth--;
}
}
#endif !SVR4
for(p = all.head; p; p = p->next[ALL_QUEUE]) {
if (p->state & T_IO_OVER) {
p->state &= ~T_IO_OVER;
count++;
continue;
}
else if (sigismember(&p->sigwaitset, AIO_SIG))
#ifndef SVR4
if (cwidth > 0 && p->wait_on_select) {
p->how_many = fds_update(&p->readfds, &p->writefds, &p->exceptfds,
&creadfds, &cwritefds, &cexceptfds, p->width);
if (p->how_many > 0) {
#ifdef DEBUG
fprintf(stderr, "wake up select\n");
#endif
count++;
pthread_q_wakeup_thread(NO_QUEUE, p, NO_QUEUE_INDEX);
sigdelset(&p->sigwaitset, AIO_SIG);
if (p->state & T_SYNCTIMER)
pthread_cancel_timed_sigwait(p, FALSE, SYNC_TIME, TRUE);
p->wait_on_select = FALSE;
}
}
else
#endif !SVR4
if (p->resultp.aio_return != AIO_INPROGRESS) {
count++;
pthread_q_wakeup_thread(NO_QUEUE, p, NO_QUEUE_INDEX);
sigdelset(&p->sigwaitset, AIO_SIG);
}
}
if (count)
return(TRUE);
else
return(FALSE);
}
#endif
/*-------------------------------------------------------------*/
#ifdef STAND_ALONE
/*------------------------------------------------------------*/
/*
* alarm - Deliver SIGALARM after "seconds"
*/
unsigned int alarm(seconds)
unsigned int seconds;
{
set_errno(ENOSYS);
return(-1);
}
#endif STAND_ALONE
/*------------------------------------------------------------*/
/*
* pthread_setsigcontext_np - modify the signal context to return to a setjmp()
* call location, i.e. simulate a longjmp() but leave the signal handler
* properly.
*/
void pthread_setsigcontext_np(scp, env, val)
struct context_t *scp;
jmp_buf env;
int val;
{
scp->sc_pc = env[JB_PC];
scp->sc_npc = env[JB_PC] + 4;
#if defined(ASM_SETJMP) || !defined(C_CONTEXT_SWITCH)
scp->sc_pc += RETURN_OFFSET;
scp->sc_npc += RETURN_OFFSET;
#endif defined(ASM_SETJMP) || !defined(C_CONTEXT_SWITCH)
scp->sc_sp = env[JB_SP];
if (env[JB_SVMASK])
pthread_sigcpyset2set(&scp->sc_mask, &env[JB_MASK]); /* copy sigmasks */
scp->sc_o0 = val;
}
/*------------------------------------------------------------*/
/*
* SIGPROCMASK - change or examine signal mask of process
*/
int SIGPROCMASK(how, set, oset)
int how;
const sigset_t *set;
sigset_t *oset;
{
#ifdef SVR4
return(syscall(SYS_sigprocmask, how, set, oset));
#else !SVR4
sigset_t old;
#ifndef STAND_ALONE
proc_mask = sigblock(0);
#endif !STAND_ALONE
if (oset)
pthread_sigcpyset2set(oset, &proc_mask);
if (!set)
return(0);
switch (how) {
case SIG_BLOCK:
pthread_sigaddset2set(&proc_mask, set);
break;
case SIG_UNBLOCK:
pthread_sigdelset2set(&proc_mask, set);
break;
case SIG_SETMASK:
pthread_sigcpyset2set(&proc_mask, set);
break;
default:
set_errno(EINVAL);
return(-1);
}
pthread_sigdelset2set(&proc_mask, &cantmask);
#ifndef STAND_ALONE
sigsetmask(proc_mask);
#endif !STAND_ALONE
return(0);
#endif !SVR4
}
/*------------------------------------------------------------*/
/*
* SIGACTION - change or examine signal handlers of process
*/
int SIGACTION(sig, act, oact)
int sig;
const struct sigaction *act;
struct sigaction *oact;
{
#ifdef SVR4
#if defined(SOLARIS) && defined(IO)
if (sig == SIGIO)
sigaction(sig, act, oact);
else
#endif SOLARIS && IO
return(syscall(SYS_sigaction, sig, act, oact));
#else !SVR4
#ifndef STAND_ALONE
return(sigvec(sig, act, oact)); /* cheating: structs distinct / same layout */
#else !STAND_ALONE
set_errno(EINVAL);
return(-1);
#endif !STAND_ALONE
#endif !SVR4
}
/*------------------------------------------------------------*/
/*
* SIGSUSPEND - suspend process waiting for signals
*/
int SIGSUSPEND(set)
sigset_t *set;
{
#ifdef SVR4
return(syscall(SYS_sigsuspend, set));
#else !SVR4
#ifndef STAND_ALONE
return(sigpause(*set));
#else !STAND_ALONE
/* busy wait */;
#endif !STAND_ALONE
#endif !SVR4
}
#if defined(IO) && !defined(SVR4)
/*------------------------------------------------------------*/
/*
* ACCEPT - accept call
*/
int ACCEPT(s, addr, addrlen)
int s;
struct sockaddr *addr;
int *addrlen;
{
return(syscall(SYS_accept, s, addr, addrlen));
}
/*------------------------------------------------------------*/
/*
* SELECT - select call
*/
int SELECT(width, readfds, writefds, exceptfds, timeout)
int width;
fd_set *readfds, *writefds, *exceptfds;
struct timeval *timeout;
{
return(syscall(SYS_select, width, readfds, writefds, exceptfds, timeout));
}
#endif IO && !SVR4