Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/arch/xen/xen remove the per-channel spin lock and instea...
details: https://anonhg.NetBSD.org/src/rev/5f9e03abf7c6
branches: trunk
changeset: 1009879:5f9e03abf7c6
user: bouyer <bouyer%NetBSD.org@localhost>
date: Wed May 06 13:43:48 2020 +0000
description:
remove the per-channel spin lock and instead make sure that events
add/remove will only be done on the CPU the handler is bound to, with
interrupts disabled.
Should be similar to the native x86 interrupts add/remove.
diffstat:
sys/arch/xen/xen/evtchn.c | 154 +++++++++++++++++++++++++--------------------
1 files changed, 86 insertions(+), 68 deletions(-)
diffs (truncated from 312 to 300 lines):
diff -r ef751e4007d7 -r 5f9e03abf7c6 sys/arch/xen/xen/evtchn.c
--- a/sys/arch/xen/xen/evtchn.c Wed May 06 13:13:50 2020 +0000
+++ b/sys/arch/xen/xen/evtchn.c Wed May 06 13:43:48 2020 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: evtchn.c,v 1.92 2020/05/04 15:55:56 jdolecek Exp $ */
+/* $NetBSD: evtchn.c,v 1.93 2020/05/06 13:43:48 bouyer Exp $ */
/*
* Copyright (c) 2006 Manuel Bouyer.
@@ -54,7 +54,7 @@
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: evtchn.c,v 1.92 2020/05/04 15:55:56 jdolecek Exp $");
+__KERNEL_RCSID(0, "$NetBSD: evtchn.c,v 1.93 2020/05/06 13:43:48 bouyer Exp $");
#include "opt_xen.h"
#include "isa.h"
@@ -70,6 +70,7 @@
#include <sys/reboot.h>
#include <sys/mutex.h>
#include <sys/interrupt.h>
+#include <sys/xcall.h>
#include <uvm/uvm.h>
@@ -91,9 +92,6 @@
/* event handlers */
struct evtsource *evtsource[NR_EVENT_CHANNELS];
-/* channel locks */
-static kmutex_t evtlock[NR_EVENT_CHANNELS];
-
/* Reference counts for bindings to event channels XXX: redo for SMP */
static uint8_t evtch_bindcount[NR_EVENT_CHANNELS];
@@ -362,7 +360,6 @@
KASSERT(ci->ci_ilevel >= IPL_VM);
KASSERT(cpu_intr_p());
x86_enable_intr();
- mutex_spin_enter(&evtlock[evtch]);
ih = evtsource[evtch]->ev_handlers;
while (ih != NULL) {
KASSERT(ih->ih_cpu == ci);
@@ -383,7 +380,6 @@
hypervisor_set_ipending(iplmask,
evtch >> LONG_SHIFT, evtch & LONG_MASK);
/* leave masked */
- mutex_spin_exit(&evtlock[evtch]);
goto splx;
}
iplmask &= ~(1 << XEN_IPL2SIR(ih->ih_level));
@@ -392,7 +388,6 @@
ih_fun(ih->ih_arg, regs);
ih = ih->ih_evt_next;
}
- mutex_spin_exit(&evtlock[evtch]);
x86_disable_intr();
hypervisor_unmask_event(evtch);
@@ -773,13 +768,10 @@
struct intrhand *ih;
int cpu_receive = 0;
-#ifdef MULTIPROCESSOR
- KASSERT(!mutex_owned(&evtlock[evtch]));
-#endif
- mutex_spin_enter(&evtlock[evtch]);
evts->ev_maxlevel = IPL_NONE;
evts->ev_imask = 0;
for (ih = evts->ev_handlers; ih != NULL; ih = ih->ih_evt_next) {
+ KASSERT(ih->ih_cpu == curcpu());
if (ih->ih_level > evts->ev_maxlevel)
evts->ev_maxlevel = ih->ih_level;
evts->ev_imask |= (1 << XEN_IPL2SIR(ih->ih_level));
@@ -790,7 +782,6 @@
xen_atomic_set_bit(&curcpu()->ci_evtmask[0], evtch);
else
xen_atomic_clear_bit(&curcpu()->ci_evtmask[0], evtch);
- mutex_spin_exit(&evtlock[evtch]);
}
struct intrhand *
@@ -800,7 +791,6 @@
struct cpu_info *ci = curcpu(); /* XXX: pass in ci ? */
struct evtsource *evts;
struct intrhand *ih, **ihp;
- int s;
char intrstr_buf[INTRIDBUF];
#ifdef IRQ_DEBUG
@@ -836,11 +826,7 @@
}
#endif /* MULTIPROCESSOR */
- s = splhigh();
-
- /* register per-cpu handler for spllower() */
- event_set_iplhandler(ci, ih, level);
-
+ mutex_enter(&cpu_lock);
/* register handler for event channel */
if (evtsource[evtch] == NULL) {
evtchn_op_t op;
@@ -860,7 +846,6 @@
* is more explicitly implemented.
*/
evts->ev_cpu = ci;
- mutex_init(&evtlock[evtch], MUTEX_DEFAULT, IPL_HIGH);
evtsource[evtch] = evts;
strlcpy(evts->ev_intrname, intrname, sizeof(evts->ev_intrname));
@@ -876,10 +861,14 @@
ci->ci_vcpuid);
}
}
+#ifndef XENPV
+ evts->ev_isl = intr_allocate_io_intrsource(intrname);
+ evts->ev_isl->is_pic = &xen_pic;
+ evts->ev_isl->is_handlers = evts->ev_handlers;
+#endif
} else {
evts = evtsource[evtch];
/* sort by IPL order, higher first */
- mutex_spin_enter(&evtlock[evtch]);
for (ihp = &evts->ev_handlers; ; ihp = &((*ihp)->ih_evt_next)) {
if ((*ihp)->ih_level < ih->ih_level) {
/* insert before *ihp */
@@ -892,33 +881,23 @@
break;
}
}
- mutex_spin_exit(&evtlock[evtch]);
#ifndef XENPV
- mutex_enter(&cpu_lock);
evts->ev_isl->is_handlers = evts->ev_handlers;
- mutex_exit(&cpu_lock);
#endif
}
-
+ const u_long psl = x86_read_psl();
+ x86_disable_intr();
+ /* register per-cpu handler for spllower() */
+ event_set_iplhandler(ci, ih, level);
+ intr_calculatemasks(evts, evtch, ci);
+ x86_write_psl(psl);
// append device name
if (evts->ev_xname[0] != '\0')
strlcat(evts->ev_xname, ", ", sizeof(evts->ev_xname));
strlcat(evts->ev_xname, xname, sizeof(evts->ev_xname));
- intr_calculatemasks(evts, evtch, ci);
- splx(s);
-#ifndef XENPV
- mutex_enter(&cpu_lock);
- if (evts->ev_isl == NULL) {
- evts->ev_isl = intr_allocate_io_intrsource(intrname);
- evts->ev_isl->is_pic = &xen_pic;
- }
- evts->ev_isl->is_handlers = evts->ev_handlers;
mutex_exit(&cpu_lock);
-#endif
-
-
return ih;
}
@@ -932,6 +911,7 @@
KASSERT(sir >= SIR_XENIPL_VM && sir <= SIR_XENIPL_HIGH);
KASSERT(ci == ih->ih_cpu);
+ KASSERT(ci == curcpu());
if (ci->ci_isources[sir] == NULL) {
ipls = kmem_zalloc(sizeof (struct intrsource),
KM_NOSLEEP);
@@ -950,66 +930,104 @@
x86_intr_calculatemasks(ci);
}
-int
-event_remove_handler(int evtch, int (*func)(void *), void *arg)
+/*
+ * Called on bound CPU to handle event_remove_handler()
+ * caller (on initiating CPU) holds cpu_lock on our behalf
+ * arg1: evtch
+ * arg2: struct intrhand *ih
+ */
+
+static void
+event_remove_handler_xcall(void *arg1, void *arg2)
{
struct intrsource *ipls;
struct evtsource *evts;
- struct intrhand *ih;
struct intrhand **ihp;
struct cpu_info *ci;
+ struct intrhand *ih = arg2;
+ int evtch = (intptr_t)(arg1);
evts = evtsource[evtch];
- if (evts == NULL)
- return ENOENT;
+ KASSERT(evts != NULL);
+ KASSERT(ih != NULL);
+ ci = ih->ih_cpu;
+ KASSERT(ci == curcpu());
- mutex_spin_enter(&evtlock[evtch]);
- for (ihp = &evts->ev_handlers, ih = evts->ev_handlers;
- ih != NULL;
- ihp = &ih->ih_evt_next, ih = ih->ih_evt_next) {
- if (ih->ih_realfun == func && ih->ih_realarg == arg)
+ const u_long psl = x86_read_psl();
+ x86_disable_intr();
+
+ for (ihp = &evts->ev_handlers; *ihp != NULL;
+ ihp = &(*ihp)->ih_evt_next) {
+ if ((*ihp) == ih)
break;
}
- if (ih == NULL) {
- mutex_spin_exit(&evtlock[evtch]);
- return ENOENT;
+ if (*(ihp) == NULL) {
+ panic("event_remove_handler_xcall: not in ev_handlers");
}
- ci = ih->ih_cpu;
+
*ihp = ih->ih_evt_next;
int sir = XEN_IPL2SIR(ih->ih_level);
KASSERT(sir >= SIR_XENIPL_VM && sir <= SIR_XENIPL_HIGH);
ipls = ci->ci_isources[sir];
- for (ihp = &ipls->is_handlers, ih = ipls->is_handlers;
- ih != NULL;
- ihp = &ih->ih_next, ih = ih->ih_next) {
+ for (ihp = &ipls->is_handlers; *ihp != NULL; ihp = &(*ihp)->ih_next) {
+ if (*ihp == ih)
+ break;
+ }
+ if (*ihp == NULL)
+ panic("event_remove_handler_xcall: not in is_handlers");
+ *ihp = ih->ih_next;
+ intr_calculatemasks(evts, evtch, ci);
+#ifndef XENPV
+ evts->ev_isl->is_handlers = evts->ev_handlers;
+#endif
+ if (evts->ev_handlers == NULL)
+ xen_atomic_clear_bit(&ci->ci_evtmask[0], evtch);
+
+ x86_write_psl(psl);
+}
+
+int
+event_remove_handler(int evtch, int (*func)(void *), void *arg)
+{
+ struct intrhand *ih;
+ struct cpu_info *ci;
+ struct evtsource *evts;
+
+ mutex_enter(&cpu_lock);
+ evts = evtsource[evtch];
+ if (evts == NULL)
+ return ENOENT;
+
+ for (ih = evts->ev_handlers; ih != NULL; ih = ih->ih_evt_next) {
if (ih->ih_realfun == func && ih->ih_realarg == arg)
break;
}
- if (ih == NULL)
- panic("event_remove_handler");
- *ihp = ih->ih_next;
- mutex_spin_exit(&evtlock[evtch]);
-#ifndef XENPV
- mutex_enter(&cpu_lock);
- evts->ev_isl->is_handlers = evts->ev_handlers;
- mutex_exit(&cpu_lock);
-#endif
+ if (ih == NULL) {
+ mutex_exit(&cpu_lock);
+ return ENOENT;
+ }
+ ci = ih->ih_cpu;
+
+ if (ci == curcpu() || !mp_online) {
+ event_remove_handler_xcall((void *)(intptr_t)evtch, ih);
+ } else {
+ uint64_t where = xc_unicast(0, event_remove_handler_xcall,
+ (void *)(intptr_t)evtch, ih, ci);
+ xc_wait(where);
+ }
+
kmem_free(ih, sizeof (struct intrhand));
if (evts->ev_handlers == NULL) {
#ifndef XENPV
KASSERT(evts->ev_isl->is_handlers == NULL);
- mutex_enter(&cpu_lock);
intr_free_io_intrsource(evts->ev_intrname);
- mutex_exit(&cpu_lock);
Home |
Main Index |
Thread Index |
Old Index