mirror of
https://git.FreeBSD.org/src.git
synced 2025-01-21 15:45:02 +00:00
- Move struct ithd to sys/interrupt.h.
- Add a set of MI helper functions for interrupt threads: - ithread_create() creates a new interrupt thread - ithread_destroy() destroys an interrupt thread - ithread_add_handler() attaches a new handler to an interrupt thread - ithread_remove_handler() detaches a handler from an interrupt thread - Rename sinthand_add() and sched_swi() to swi_add() and swi_sched() respectively so that they live in a consistent namespace. - struct intrhand is no longer a public type. It would be private to kern_intr.c but the current implementation of fast interrupts on the alpha requires the type to be exported. However, all handlers should be treated as void * cookies in the way that new-bus treats them. This includes references to software interrupt handlers.
This commit is contained in:
parent
3687f15b08
commit
b4151f7101
Notes:
svn2git
2020-12-20 02:59:44 +00:00
svn path=/head/; revision=72237
@ -40,37 +40,43 @@
|
||||
#include <sys/malloc.h>
|
||||
#include <sys/mutex.h>
|
||||
#include <sys/proc.h>
|
||||
#include <sys/resourcevar.h>
|
||||
#include <sys/unistd.h>
|
||||
#include <sys/vmmeter.h>
|
||||
#include <machine/atomic.h>
|
||||
#include <machine/cpu.h>
|
||||
#include <machine/md_var.h>
|
||||
#include <machine/stdarg.h>
|
||||
|
||||
#include <net/netisr.h> /* prototype for legacy_setsoftnet */
|
||||
|
||||
struct intrhand *net_ih;
|
||||
struct intrhand *vm_ih;
|
||||
struct intrhand *softclock_ih;
|
||||
void *net_ih;
|
||||
void *vm_ih;
|
||||
void *softclock_ih;
|
||||
struct ithd *clk_ithd;
|
||||
struct ithd *tty_ithd;
|
||||
|
||||
static struct mtx ithread_list_lock;
|
||||
|
||||
static MALLOC_DEFINE(M_ITHREAD, "ithread", "Interrupt Threads");
|
||||
|
||||
static void ithread_update(struct ithd *);
|
||||
static void ithread_loop(void *);
|
||||
static void ithread_init(void *);
|
||||
static void start_softintr(void *);
|
||||
static void swi_net(void *);
|
||||
|
||||
int
|
||||
ithread_priority(flags)
|
||||
int flags;
|
||||
u_char
|
||||
ithread_priority(enum intr_type flags)
|
||||
{
|
||||
int pri;
|
||||
u_char pri;
|
||||
|
||||
flags &= ~INTR_MPSAFE;
|
||||
flags &= (INTR_TYPE_TTY | INTR_TYPE_BIO | INTR_TYPE_NET |
|
||||
INTR_TYPE_CAM | INTR_TYPE_MISC | INTR_TYPE_CLK);
|
||||
switch (flags) {
|
||||
case INTR_TYPE_TTY: /* keyboard or parallel port */
|
||||
case INTR_TYPE_TTY:
|
||||
pri = PI_TTYLOW;
|
||||
break;
|
||||
case (INTR_TYPE_TTY | INTR_FAST): /* sio */
|
||||
pri = PI_TTYHIGH;
|
||||
break;
|
||||
case INTR_TYPE_BIO:
|
||||
/*
|
||||
* XXX We need to refine this. BSD/OS distinguishes
|
||||
@ -84,63 +90,238 @@ ithread_priority(flags)
|
||||
case INTR_TYPE_CAM:
|
||||
pri = PI_DISK; /* XXX or PI_CAM? */
|
||||
break;
|
||||
case INTR_TYPE_CLK:
|
||||
pri = PI_REALTIME;
|
||||
break;
|
||||
case INTR_TYPE_MISC:
|
||||
pri = PI_DULL; /* don't care */
|
||||
break;
|
||||
/* We didn't specify an interrupt level. */
|
||||
default:
|
||||
/* We didn't specify an interrupt level. */
|
||||
panic("ithread_priority: no interrupt type in flags");
|
||||
}
|
||||
|
||||
return pri;
|
||||
}
|
||||
|
||||
void sithd_loop(void *);
|
||||
/*
|
||||
* Regenerate the name (p_comm) and priority for a threaded interrupt thread.
|
||||
*/
|
||||
static void
|
||||
ithread_update(struct ithd *ithd)
|
||||
{
|
||||
struct intrhand *ih;
|
||||
struct proc *p;
|
||||
int entropy;
|
||||
|
||||
struct intrhand *
|
||||
sinthand_add(const char *name, struct ithd **ithdp, driver_intr_t handler,
|
||||
void *arg, int pri, int flags)
|
||||
p = ithd->it_proc;
|
||||
if (p == NULL)
|
||||
return;
|
||||
|
||||
strncpy(p->p_comm, ithd->it_name, sizeof(ithd->it_name));
|
||||
ih = TAILQ_FIRST(&ithd->it_handlers);
|
||||
if (ih == NULL) {
|
||||
p->p_rtprio.prio = RTP_PRIO_MAX;
|
||||
ithd->it_flags &= ~IT_ENTROPY;
|
||||
return;
|
||||
}
|
||||
|
||||
entropy = 0;
|
||||
p->p_rtprio.prio = ih->ih_pri;
|
||||
TAILQ_FOREACH(ih, &ithd->it_handlers, ih_next) {
|
||||
if (strlen(p->p_comm) + strlen(ih->ih_name) + 1 <
|
||||
sizeof(p->p_comm)) {
|
||||
strcat(p->p_comm, " ");
|
||||
strcat(p->p_comm, ih->ih_name);
|
||||
} else if (strlen(p->p_comm) + 1 == sizeof(p->p_comm)) {
|
||||
if (p->p_comm[sizeof(p->p_comm) - 2] == '+')
|
||||
p->p_comm[sizeof(p->p_comm) - 2] = '*';
|
||||
else
|
||||
p->p_comm[sizeof(p->p_comm) - 2] = '+';
|
||||
} else
|
||||
strcat(p->p_comm, "+");
|
||||
if (ih->ih_flags & IH_ENTROPY)
|
||||
entropy++;
|
||||
}
|
||||
|
||||
if (entropy) {
|
||||
printf("Warning, ithread (%d, %s) is an entropy source.\n",
|
||||
p->p_pid, p->p_comm);
|
||||
ithd->it_flags |= IT_ENTROPY;
|
||||
}
|
||||
else
|
||||
ithd->it_flags &= ~IT_ENTROPY;
|
||||
}
|
||||
|
||||
int
|
||||
ithread_create(struct ithd **ithread, int vector, int flags,
|
||||
void (*disable)(int), void (*enable)(int), const char *fmt, ...)
|
||||
{
|
||||
struct ithd *ithd;
|
||||
struct proc *p;
|
||||
int error;
|
||||
va_list ap;
|
||||
|
||||
ithd = malloc(sizeof(struct ithd), M_ITHREAD, M_WAITOK | M_ZERO);
|
||||
ithd->it_vector = vector;
|
||||
ithd->it_disable = disable;
|
||||
ithd->it_enable = enable;
|
||||
ithd->it_flags = flags;
|
||||
TAILQ_INIT(&ithd->it_handlers);
|
||||
|
||||
va_start(ap, fmt);
|
||||
vsnprintf(ithd->it_name, sizeof(ithd->it_name), fmt, ap);
|
||||
va_end(ap);
|
||||
|
||||
error = kthread_create(ithread_loop, ithd, &p, RFSTOPPED | RFHIGHPID,
|
||||
ithd->it_name);
|
||||
if (error) {
|
||||
free(ithd, M_ITHREAD);
|
||||
return (error);
|
||||
}
|
||||
p->p_rtprio.type = RTP_PRIO_ITHREAD;
|
||||
p->p_rtprio.prio = RTP_PRIO_MAX;
|
||||
p->p_stat = SWAIT;
|
||||
ithd->it_proc = p;
|
||||
p->p_ithd = ithd;
|
||||
if (ithread != NULL)
|
||||
*ithread = ithd;
|
||||
|
||||
return (0);
|
||||
}
|
||||
|
||||
int
|
||||
ithread_destroy(struct ithd *ithread)
|
||||
{
|
||||
|
||||
if (ithread == NULL || !TAILQ_EMPTY(&ithread->it_handlers))
|
||||
return (EINVAL);
|
||||
|
||||
mtx_lock_spin(&sched_lock);
|
||||
ithread->it_flags |= IT_DEAD;
|
||||
if (ithread->it_proc->p_stat == SWAIT) {
|
||||
ithread->it_proc->p_stat = SRUN;
|
||||
setrunqueue(ithread->it_proc);
|
||||
}
|
||||
mtx_unlock_spin(&sched_lock);
|
||||
return (0);
|
||||
}
|
||||
|
||||
int
|
||||
ithread_add_handler(struct ithd* ithread, const char *name,
|
||||
driver_intr_t handler, void *arg, u_char pri, enum intr_type flags,
|
||||
void **cookiep)
|
||||
{
|
||||
struct intrhand *ih, *temp_ih;
|
||||
|
||||
if (ithread == NULL || name == NULL || handler == NULL)
|
||||
return (EINVAL);
|
||||
if ((flags & INTR_FAST) !=0)
|
||||
flags |= INTR_EXCL;
|
||||
|
||||
ih = malloc(sizeof(struct intrhand), M_ITHREAD, M_WAITOK | M_ZERO);
|
||||
ih->ih_handler = handler;
|
||||
ih->ih_argument = arg;
|
||||
ih->ih_name = name;
|
||||
ih->ih_ithread = ithread;
|
||||
ih->ih_pri = pri;
|
||||
if (flags & INTR_FAST)
|
||||
ih->ih_flags = IH_FAST | IH_EXCLUSIVE;
|
||||
else if (flags & INTR_EXCL)
|
||||
ih->ih_flags = IH_EXCLUSIVE;
|
||||
if (flags & INTR_MPSAFE)
|
||||
ih->ih_flags |= IH_MPSAFE;
|
||||
if (flags & INTR_ENTROPY)
|
||||
ih->ih_flags |= IH_ENTROPY;
|
||||
|
||||
mtx_lock_spin(&ithread_list_lock);
|
||||
if ((flags & INTR_EXCL) !=0 && !TAILQ_EMPTY(&ithread->it_handlers))
|
||||
goto fail;
|
||||
if (!TAILQ_EMPTY(&ithread->it_handlers) &&
|
||||
(TAILQ_FIRST(&ithread->it_handlers)->ih_flags & IH_EXCLUSIVE) != 0)
|
||||
goto fail;
|
||||
|
||||
TAILQ_FOREACH(temp_ih, &ithread->it_handlers, ih_next)
|
||||
if (temp_ih->ih_pri > ih->ih_pri)
|
||||
break;
|
||||
if (temp_ih == NULL)
|
||||
TAILQ_INSERT_TAIL(&ithread->it_handlers, ih, ih_next);
|
||||
else
|
||||
TAILQ_INSERT_BEFORE(temp_ih, ih, ih_next);
|
||||
ithread_update(ithread);
|
||||
mtx_unlock_spin(&ithread_list_lock);
|
||||
|
||||
if (cookiep != NULL)
|
||||
*cookiep = ih;
|
||||
return (0);
|
||||
|
||||
fail:
|
||||
mtx_unlock_spin(&ithread_list_lock);
|
||||
free(ih, M_ITHREAD);
|
||||
return (EINVAL);
|
||||
}
|
||||
|
||||
int
|
||||
ithread_remove_handler(void *cookie)
|
||||
{
|
||||
struct intrhand *handler = (struct intrhand *)cookie;
|
||||
struct ithd *ithread;
|
||||
#ifdef INVARIANTS
|
||||
struct intrhand *ih;
|
||||
int found;
|
||||
#endif
|
||||
|
||||
if (handler == NULL || (ithread = handler->ih_ithread) == NULL)
|
||||
return (EINVAL);
|
||||
|
||||
mtx_lock_spin(&ithread_list_lock);
|
||||
#ifdef INVARIANTS
|
||||
found = 0;
|
||||
TAILQ_FOREACH(ih, &ithread->it_handlers, ih_next)
|
||||
if (ih == handler) {
|
||||
found++;
|
||||
break;
|
||||
}
|
||||
if (found == 0) {
|
||||
mtx_unlock_spin(&ithread_list_lock);
|
||||
return (EINVAL);
|
||||
}
|
||||
#endif
|
||||
TAILQ_REMOVE(&ithread->it_handlers, handler, ih_next);
|
||||
ithread_update(ithread);
|
||||
mtx_unlock_spin(&ithread_list_lock);
|
||||
|
||||
free(handler, M_ITHREAD);
|
||||
return (0);
|
||||
}
|
||||
|
||||
int
|
||||
swi_add(struct ithd **ithdp, const char *name, driver_intr_t handler,
|
||||
void *arg, int pri, enum intr_type flags, void **cookiep)
|
||||
{
|
||||
struct proc *p;
|
||||
struct ithd *ithd;
|
||||
struct intrhand *ih;
|
||||
struct intrhand *this_ih;
|
||||
int error;
|
||||
|
||||
ithd = (ithdp != NULL) ? *ithdp : NULL;
|
||||
|
||||
|
||||
if (ithd == NULL) {
|
||||
int error;
|
||||
ithd = malloc(sizeof (struct ithd), M_DEVBUF, M_WAITOK | M_ZERO);
|
||||
error = kthread_create(sithd_loop, NULL, &p,
|
||||
RFSTOPPED | RFHIGHPID, "swi%d: %s", pri, name);
|
||||
error = ithread_create(&ithd, pri, IT_SOFT, NULL, NULL,
|
||||
"swi%d:", pri);
|
||||
if (error)
|
||||
panic("inthand_add: Can't create interrupt thread");
|
||||
ithd->it_proc = p;
|
||||
p->p_ithd = ithd;
|
||||
p->p_rtprio.type = RTP_PRIO_ITHREAD;
|
||||
p->p_rtprio.prio = pri + PI_SOFT; /* soft interrupt */
|
||||
p->p_stat = SWAIT; /* we're idle */
|
||||
return (error);
|
||||
|
||||
/* XXX - some hacks are _really_ gross */
|
||||
p = ithd->it_proc;
|
||||
PROC_LOCK(p);
|
||||
if (pri == SWI_CLOCK)
|
||||
p->p_flag |= P_NOLOAD;
|
||||
PROC_UNLOCK(p);
|
||||
if (ithdp != NULL)
|
||||
*ithdp = ithd;
|
||||
}
|
||||
this_ih = malloc(sizeof (struct intrhand), M_DEVBUF, M_WAITOK | M_ZERO);
|
||||
this_ih->ih_handler = handler;
|
||||
this_ih->ih_argument = arg;
|
||||
this_ih->ih_flags = flags;
|
||||
this_ih->ih_ithd = ithd;
|
||||
this_ih->ih_name = malloc(strlen(name) + 1, M_DEVBUF, M_WAITOK);
|
||||
if ((ih = ithd->it_ih)) {
|
||||
while (ih->ih_next != NULL)
|
||||
ih = ih->ih_next;
|
||||
ih->ih_next = this_ih;
|
||||
} else
|
||||
ithd->it_ih = this_ih;
|
||||
strcpy(this_ih->ih_name, name);
|
||||
return (this_ih);
|
||||
return (ithread_add_handler(ithd, name, handler, arg, pri + PI_SOFT,
|
||||
flags, cookiep));
|
||||
}
|
||||
|
||||
|
||||
@ -148,14 +329,15 @@ sinthand_add(const char *name, struct ithd **ithdp, driver_intr_t handler,
|
||||
* Schedule a heavyweight software interrupt process.
|
||||
*/
|
||||
void
|
||||
sched_swi(struct intrhand *ih, int flag)
|
||||
swi_sched(void *cookie, int flags)
|
||||
{
|
||||
struct ithd *it = ih->ih_ithd; /* and the process that does it */
|
||||
struct intrhand *ih = (struct intrhand *)cookie;
|
||||
struct ithd *it = ih->ih_ithread;
|
||||
struct proc *p = it->it_proc;
|
||||
|
||||
atomic_add_int(&cnt.v_intr, 1); /* one more global interrupt */
|
||||
|
||||
CTR3(KTR_INTR, "sched_swi pid %d(%s) need=%d",
|
||||
CTR3(KTR_INTR, "swi_sched pid %d(%s) need=%d",
|
||||
p->p_pid, p->p_comm, it->it_need);
|
||||
|
||||
/*
|
||||
@ -165,67 +347,86 @@ sched_swi(struct intrhand *ih, int flag)
|
||||
* there. In any case, kick everyone so that if the new thread
|
||||
* is higher priority than their current thread, it gets run now.
|
||||
*/
|
||||
ih->ih_need = 1;
|
||||
if (!(flag & SWI_DELAY)) {
|
||||
atomic_store_rel_int(&ih->ih_need, 1);
|
||||
if (!(flags & SWI_DELAY)) {
|
||||
it->it_need = 1;
|
||||
mtx_lock_spin(&sched_lock);
|
||||
if (p->p_stat == SWAIT) { /* not on run queue */
|
||||
CTR1(KTR_INTR, "sched_swi: setrunqueue %d", p->p_pid);
|
||||
/* membar_lock(); */
|
||||
CTR1(KTR_INTR, "swi_sched: setrunqueue %d", p->p_pid);
|
||||
p->p_stat = SRUN;
|
||||
setrunqueue(p);
|
||||
aston();
|
||||
if (!cold && flags & SWI_SWITCH) {
|
||||
if (curproc != PCPU_GET(idleproc))
|
||||
setrunqueue(curproc);
|
||||
curproc->p_stats->p_ru.ru_nvcsw++;
|
||||
mi_switch();
|
||||
} else
|
||||
need_resched();
|
||||
}
|
||||
else {
|
||||
CTR3(KTR_INTR, "sched_swi %d: it_need %d, state %d",
|
||||
CTR3(KTR_INTR, "swi_sched %d: it_need %d, state %d",
|
||||
p->p_pid, it->it_need, p->p_stat );
|
||||
}
|
||||
mtx_unlock_spin(&sched_lock);
|
||||
need_resched();
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* This is the main code for soft interrupt threads.
|
||||
* This is the main code for interrupt threads.
|
||||
*/
|
||||
void
|
||||
sithd_loop(void *dummy)
|
||||
ithread_loop(void *arg)
|
||||
{
|
||||
struct ithd *it; /* our thread context */
|
||||
struct ithd *ithd; /* our thread context */
|
||||
struct intrhand *ih; /* and our interrupt handler chain */
|
||||
struct proc *p;
|
||||
|
||||
struct proc *p = curproc;
|
||||
it = p->p_ithd; /* point to myself */
|
||||
p = curproc;
|
||||
ithd = (struct ithd *)arg; /* point to myself */
|
||||
KASSERT(ithd->it_proc == p && p->p_ithd == ithd,
|
||||
(__func__ ": ithread and proc linkage out of sync"));
|
||||
|
||||
/*
|
||||
* As long as we have interrupts outstanding, go through the
|
||||
* list of handlers, giving each one a go at it.
|
||||
*/
|
||||
for (;;) {
|
||||
CTR3(KTR_INTR, "sithd_loop pid %d(%s) need=%d",
|
||||
p->p_pid, p->p_comm, it->it_need);
|
||||
while (it->it_need) {
|
||||
/*
|
||||
* If we are an orphaned thread, then just die.
|
||||
*/
|
||||
if (ithd->it_flags & IT_DEAD) {
|
||||
CTR2(KTR_INTR, __func__ ": pid %d: (%s) exiting",
|
||||
p->p_pid, p->p_comm);
|
||||
p->p_ithd = NULL;
|
||||
mtx_lock(&Giant);
|
||||
free(ithd, M_ITHREAD);
|
||||
kthread_exit(0);
|
||||
}
|
||||
|
||||
CTR3(KTR_INTR, __func__ ": pid %d: (%s) need=%d",
|
||||
p->p_pid, p->p_comm, ithd->it_need);
|
||||
while (ithd->it_need) {
|
||||
/*
|
||||
* Service interrupts. If another interrupt
|
||||
* arrives while we are running, they will set
|
||||
* it_need to denote that we should make
|
||||
* another pass.
|
||||
*/
|
||||
it->it_need = 0;
|
||||
for (ih = it->it_ih; ih != NULL; ih = ih->ih_next) {
|
||||
if (!ih->ih_need)
|
||||
atomic_store_rel_int(&ithd->it_need, 0);
|
||||
TAILQ_FOREACH(ih, &ithd->it_handlers, ih_next) {
|
||||
if (ithd->it_flags & IT_SOFT && !ih->ih_need)
|
||||
continue;
|
||||
ih->ih_need = 0;
|
||||
atomic_store_rel_int(&ih->ih_need, 0);
|
||||
CTR5(KTR_INTR,
|
||||
"sithd_loop pid %d ih=%p: %p(%p) flg=%x",
|
||||
__func__ ": pid %d ih=%p: %p(%p) flg=%x",
|
||||
p->p_pid, (void *)ih,
|
||||
(void *)ih->ih_handler, ih->ih_argument,
|
||||
ih->ih_flags);
|
||||
|
||||
if ((ih->ih_flags & INTR_MPSAFE) == 0)
|
||||
if ((ih->ih_flags & IH_MPSAFE) == 0)
|
||||
mtx_lock(&Giant);
|
||||
ih->ih_handler(ih->ih_argument);
|
||||
if ((ih->ih_flags & INTR_MPSAFE) == 0)
|
||||
if ((ih->ih_flags & IH_MPSAFE) == 0)
|
||||
mtx_unlock(&Giant);
|
||||
}
|
||||
}
|
||||
@ -237,35 +438,51 @@ sithd_loop(void *dummy)
|
||||
*/
|
||||
mtx_assert(&Giant, MA_NOTOWNED);
|
||||
mtx_lock_spin(&sched_lock);
|
||||
if (!it->it_need) {
|
||||
if (!ithd->it_need) {
|
||||
/*
|
||||
* Should we call this earlier in the loop above?
|
||||
*/
|
||||
if (ithd->it_enable != NULL)
|
||||
ithd->it_enable(ithd->it_vector);
|
||||
p->p_stat = SWAIT; /* we're idle */
|
||||
CTR1(KTR_INTR, "sithd_loop pid %d: done", p->p_pid);
|
||||
CTR1(KTR_INTR, __func__ ": pid %d: done", p->p_pid);
|
||||
mi_switch();
|
||||
CTR1(KTR_INTR, "sithd_loop pid %d: resumed", p->p_pid);
|
||||
CTR1(KTR_INTR, __func__ ": pid %d: resumed", p->p_pid);
|
||||
}
|
||||
mtx_unlock_spin(&sched_lock);
|
||||
}
|
||||
}
|
||||
|
||||
SYSINIT(start_softintr, SI_SUB_SOFTINTR, SI_ORDER_FIRST, start_softintr, NULL)
|
||||
/*
|
||||
* Initialize mutex used to protect ithread handler lists.
|
||||
*/
|
||||
static void
|
||||
ithread_init(void *dummy)
|
||||
{
|
||||
|
||||
mtx_init(&ithread_list_lock, "ithread list lock", MTX_SPIN);
|
||||
}
|
||||
SYSINIT(ithread_init, SI_SUB_INTR, SI_ORDER_FIRST, ithread_init, NULL);
|
||||
|
||||
/*
|
||||
* Start standard software interrupt threads
|
||||
*/
|
||||
static void
|
||||
start_softintr(dummy)
|
||||
void *dummy;
|
||||
start_softintr(void *dummy)
|
||||
{
|
||||
net_ih = sinthand_add("net", NULL, swi_net, NULL, SWI_NET, 0);
|
||||
softclock_ih = sinthand_add("clock", &clk_ithd, softclock, NULL,
|
||||
SWI_CLOCK, INTR_MPSAFE);
|
||||
vm_ih = sinthand_add("vm", NULL, swi_vm, NULL, SWI_VM, 0);
|
||||
|
||||
if (swi_add(NULL, "net", swi_net, NULL, SWI_NET, 0, &net_ih) ||
|
||||
swi_add(&clk_ithd, "clock", softclock, NULL, SWI_CLOCK,
|
||||
INTR_MPSAFE, &softclock_ih) ||
|
||||
swi_add(NULL, "vm", swi_vm, NULL, SWI_VM, 0, &vm_ih))
|
||||
panic("died while creating standard software ithreads");
|
||||
}
|
||||
SYSINIT(start_softintr, SI_SUB_SOFTINTR, SI_ORDER_FIRST, start_softintr, NULL)
|
||||
|
||||
void
|
||||
legacy_setsoftnet()
|
||||
legacy_setsoftnet(void)
|
||||
{
|
||||
sched_swi(net_ih, SWI_NOSWITCH);
|
||||
swi_sched(net_ih, SWI_NOSWITCH);
|
||||
}
|
||||
|
||||
/*
|
||||
|
@ -33,33 +33,72 @@
|
||||
* Describe a hardware interrupt handler.
|
||||
*
|
||||
* Multiple interrupt handlers for a specific vector can be chained
|
||||
* together via the 'next' pointer.
|
||||
* together.
|
||||
*/
|
||||
|
||||
struct intrhand {
|
||||
driver_intr_t *ih_handler; /* code address of handler */
|
||||
void *ih_argument; /* argument to pass to handler */
|
||||
enum intr_type ih_flags; /* flag bits (sys/bus.h) */
|
||||
char *ih_name; /* name of handler */
|
||||
struct ithd *ih_ithd; /* handler we're connected to */
|
||||
struct intrhand *ih_next; /* next handler for this irq */
|
||||
int ih_need; /* need interrupt */
|
||||
driver_intr_t *ih_handler; /* Handler function. */
|
||||
void *ih_argument; /* Argument to pass to handler. */
|
||||
int ih_flags;
|
||||
const char *ih_name; /* Name of handler. */
|
||||
struct ithd *ih_ithread; /* Ithread we are connected to. */
|
||||
int ih_need; /* Needs service. */
|
||||
TAILQ_ENTRY(intrhand) ih_next; /* Next handler for this vector. */
|
||||
u_char ih_pri; /* Priority of this handler. */
|
||||
};
|
||||
|
||||
int ithread_priority __P((int flags));
|
||||
void sched_swi __P((struct intrhand *, int));
|
||||
#define SWI_SWITCH 0x1
|
||||
#define SWI_NOSWITCH 0x2
|
||||
#define SWI_DELAY 0x4 /* implies NOSWITCH */
|
||||
/* Interrupt handle flags kept in ih_flags */
|
||||
#define IH_FAST 0x00000001 /* Fast interrupt. */
|
||||
#define IH_EXCLUSIVE 0x00000002 /* Exclusive interrupt. */
|
||||
#define IH_ENTROPY 0x00000004 /* Device is a good entropy source. */
|
||||
#define IH_MPSAFE 0x80000000 /* Handler does not need Giant. */
|
||||
|
||||
struct intrhand * sinthand_add __P((const char *name, struct ithd **,
|
||||
driver_intr_t, void *arg, int pri, int flags));
|
||||
/*
|
||||
* Describe an interrupt thread. There is one of these per interrupt vector.
|
||||
* Note that this actually describes an interrupt source. There may or may
|
||||
* not be an actual kernel thread attached to a given source.
|
||||
*/
|
||||
struct ithd {
|
||||
struct proc *it_proc; /* Interrupt process. */
|
||||
LIST_ENTRY(ithd) it_list; /* All interrupt threads. */
|
||||
TAILQ_HEAD(, intrhand) it_handlers; /* Interrupt handlers. */
|
||||
struct ithd *it_interrupted; /* Who we interrupted. */
|
||||
void (*it_disable)(int); /* Enable interrupt source. */
|
||||
void (*it_enable)(int); /* Disable interrupt source. */
|
||||
void *it_md; /* Hook for MD interrupt code. */
|
||||
int it_flags; /* Interrupt-specific flags. */
|
||||
int it_need; /* Needs service. */
|
||||
int it_vector;
|
||||
char it_name[MAXCOMLEN + 1];
|
||||
};
|
||||
|
||||
/* Interrupt thread flags kept in it_flags */
|
||||
#define IT_SOFT 0x000001 /* Software interrupt. */
|
||||
#define IT_ENTROPY 0x000002 /* Interrupt is an entropy source. */
|
||||
#define IT_DEAD 0x000004 /* Thread is waiting to exit. */
|
||||
|
||||
/* Flags to pass to sched_swi. */
|
||||
#define SWI_NOSWITCH 0x0
|
||||
#define SWI_SWITCH 0x1
|
||||
#define SWI_DELAY 0x2 /* implies NOSWITCH */
|
||||
|
||||
extern struct ithd *tty_ithd;
|
||||
extern struct ithd *clk_ithd;
|
||||
extern void *net_ih;
|
||||
extern void *softclock_ih;
|
||||
extern void *vm_ih;
|
||||
|
||||
extern struct intrhand *net_ih;
|
||||
extern struct intrhand *softclock_ih;
|
||||
extern struct intrhand *vm_ih;
|
||||
int ithread_create __P((struct ithd **ithread, int vector, int flags,
|
||||
void (*disable)(int), void (*enable)(int), const char *name, ...))
|
||||
__printflike(6, 7);
|
||||
int ithread_destroy __P((struct ithd *ithread));
|
||||
u_char ithread_priority __P((enum intr_type flags));
|
||||
int ithread_add_handler __P((struct ithd *ithread, const char *name,
|
||||
driver_intr_t handler, void *arg, u_char pri, enum intr_type flags,
|
||||
void **cookiep));
|
||||
int ithread_remove_handler __P((void *cookie));
|
||||
int swi_add __P((struct ithd **ithdp, const char *name,
|
||||
driver_intr_t handler, void *arg, int pri, enum intr_type flags,
|
||||
void **cookiep));
|
||||
void swi_sched __P((void *cookie, int flags));
|
||||
|
||||
#endif
|
||||
|
@ -149,6 +149,8 @@ struct pargs {
|
||||
* - Reads of this value by other processes are locked.
|
||||
* - Reads of this value by the current process need not be locked.
|
||||
*/
|
||||
struct ithd;
|
||||
|
||||
struct proc {
|
||||
TAILQ_ENTRY(proc) p_procq; /* (j) Run/mutex queue. */
|
||||
TAILQ_ENTRY(proc) p_slpq; /* (j) Sleep queue. */
|
||||
@ -352,21 +354,6 @@ struct pcred {
|
||||
struct uidinfo *p_uidinfo; /* Per uid resource consumption. */
|
||||
};
|
||||
|
||||
/*
|
||||
* Describe an interrupt thread. There is one of these per irq. BSD/OS makes
|
||||
* this a superset of struct proc, i.e. it_proc is the struct itself and not a
|
||||
* pointer. We point in both directions, because it feels good that way.
|
||||
*/
|
||||
struct ithd {
|
||||
struct proc *it_proc; /* Interrupt process. */
|
||||
LIST_ENTRY(ithd) it_list; /* All interrupt threads. */
|
||||
int it_need; /* Needs service. */
|
||||
int irq; /* Vector. */
|
||||
struct intrhand *it_ih; /* Interrupt handlers. */
|
||||
struct ithd *it_interrupted; /* Who we interrupted. */
|
||||
void *it_md; /* Hook for MD interrupt code. */
|
||||
};
|
||||
|
||||
#ifdef _KERNEL
|
||||
|
||||
#ifdef MALLOC_DECLARE
|
||||
|
Loading…
Reference in New Issue
Block a user