Plan 9 from Bell Labs’s /usr/web/sources/contrib/steve/root/sys/src/c++/lib/task-3.03/event.C

Copyright © 2021 Plan 9 Foundation.
Distributed under the MIT License.
Download the Plan 9 distribution.


/*ident	"@(#)cls4:lib/task/task/event.c	1.6" */
/**************************************************************************
			Copyright (c) 1984 AT&T
	  		  All Rights Reserved  	
Copyright (c) 1993  UNIX System Laboratories, Inc.
     Copyright (c) 1991, 1992 AT&T and UNIX System Laboratories, Inc.

	THIS IS UNPUBLISHED PROPRIETARY SOURCE CODE OF AT&T
	
	The copyright notice above does not evidence any   	
	actual or intended publication of such source code.

*****************************************************************************/
#include <task.h>

const int NASS = NSIG;

static Interrupt_handler*	hnd[NASS];	// hnd[i] is used for signal i
static void (*old_handlers[NASS])(int);

Interrupt_alerter::Interrupt_alerter()
: task("Interrupt_alerter")
{
	for(;;) {
		sleep();
		for (register Interrupt_handler** p = &hnd[0]; p < &hnd[NASS]; p++)
			if (*p && (*p)->got_interrupt) {
				(*p)->alert();
			}
	}
}

void
sigFunc(int sigNo)
{
	if ((unsigned)sigNo >= (unsigned)NASS)
		object::task_error(E_BADSIG, (object*)0);
	Interrupt_handler*	as = hnd[sigNo];
	if (as == 0)
		object::task_error(E_NO_HNDLR, (object*)0);
	as->interrupt();
	as->got_interrupt = 1;
	sched::set_priority_sched(&interrupt_alerter);
	signal(sigNo, sigFunc);		//reset
					//MORE:  ifdef SVR3 use sigset
}

Interrupt_handler::Interrupt_handler(int vecid)
: id(vecid), got_interrupt(0)
{
	if ((unsigned)id >= (unsigned)NASS)
		task_error(E_BADSIG, this);
	if((old = hnd[id]) == 0)
		old_handlers[id] = signal(id, sigFunc);
	else
		if (old->got_interrupt) {
			old->alert();
		}
	hnd[id] = this;
	this_task()->keep_waiting();
}

Interrupt_handler::~Interrupt_handler()
{
	this_task()->dont_wait();
	for(register Interrupt_handler** as = &hnd[id];; as = &(*as)->old) {
		if (*as == this) break;
		if (*as == 0)
			task_error(E_LOSTHNDLR, this);
	}
	if ((*as = old) == 0)
		 signal(id, old_handlers[id]);
}

void
Interrupt_handler::interrupt()
{
}

int
Interrupt_handler::pending()
{
	if (!got_interrupt)
		return 1;
	got_interrupt = 0;
	return 0;
}

#if defined(BSD)
void
sig_hold(sigset_t* mask)
{
	int i = 1;
	sigset_t omask;
	for (register int p = 0;  p < NASS; i <<= 1, p++) {
		if (hnd[p]) {
			sigprocmask(SIG_BLOCK,(sigset_t*)&p,&omask);
			if (((int)omask & i) == 0) *mask |= i;
		}
	}
}

void
sig_relse(sigset_t* mask)
{
	int i = 1;
	sigset_t omask;
	for (register int p = 0;  p < NASS; i <<= 1, p++) {
		if ((int)*mask & i)
			sigprocmask(SIG_UNBLOCK,(sigset_t*)&p,&omask);
	}
}
#endif

#if defined(SYSV)
void
sig_hold(int* mask)
{
	int i = 1;
	int omask;
	for (register int p = 0;  p < NASS; i <<= 1, p++) {
		if (hnd[p]) {
			omask = sighold(p);
			if ((omask & i) == 0) *mask |= i;
		}
	}
}

void
sig_relse(int* mask)
{
	int i = 1;
	for (register int p = 0;  p < NASS; i <<= 1, p++) {
		if (*mask & i) sigrelse(p);
	}
}
#endif


Bell Labs OSI certified Powered by Plan 9

(Return to Plan 9 Home Page)

Copyright © 2021 Plan 9 Foundation. All Rights Reserved.
Comments to webmaster@9p.io.