File:  [Research Unix] / researchv10no / cmd / cfront / libC / otask / event.c
Revision 1.1.1.1 (vendor branch): download - view: text, annotated - select for diffs
Tue Apr 24 17:21:35 2018 UTC (8 years, 1 month ago) by root
Branches: belllabs, MAIN
CVS tags: researchv10, HEAD
researchv10 Norman

#include "task.h"

const	NASS = NSIG;

static Interrupt_handler*	hnd[NASS];	// hnd[i] is used for signal i
// static SIG_TYP	old_handlers[NASS];	// for old compilers
static SIG_FUNC_TYP*	old_handlers[NASS];

Interrupt_alerter::Interrupt_alerter()
: ("Interrupt_alerter")
{
	for(;;) {
		sleep();
// fprintf(stderr, "Interrupt_alerter loop\n");
		for (register Interrupt_handler** p = &hnd[0]; p < &hnd[NASS]; p++)
			if (*p && (*p)->got_interrupt)
				(*p)->alert();
	}
}

void
sigFunc(int sigNo
#ifdef BSD
, int, sigcontext*
#endif
)
{
// fprintf(stderr, "signal #%d\n", sigNo);
	if ((unsigned)sigNo >= (unsigned)NASS)
		((object*)0)->task_error (E_BADSIG);
	Interrupt_handler*	as = hnd[sigNo];
	if (as == 0)
		((object*)0)->task_error(E_NO_HNDLR);
	as->interrupt();
	as->got_interrupt = 1;
	sched::priority_sched = &interrupt_alerter;
	signal(sigNo, sigFunc);
// fprintf(stderr, "return from interrupt #%d\n", sigNo);
	return;
}

Interrupt_handler::Interrupt_handler(int vecid)
: id(vecid), got_interrupt(0)
{
// fprintf(stderr, "Interrupt_handler::Interrupt_handler #%d\n", vecid);
	if ((unsigned)id >= (unsigned)NASS)
		task_error(E_BADSIG);
	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()
{
// fprintf(stderr, "Interrupt_handler::~Interrupt_handler #%d\n", id);
	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);
	}
	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;
}

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.