summaryrefslogtreecommitdiffstats
path: root/sys/amd64/isa
diff options
context:
space:
mode:
authorbde <bde@FreeBSD.org>2001-05-02 13:06:58 +0000
committerbde <bde@FreeBSD.org>2001-05-02 13:06:58 +0000
commita3c8b91e8eb2355576e69f7c407f40bc82279aad (patch)
tree477ccfdfead3184dfc589d2ecd13d3872b743e47 /sys/amd64/isa
parentb9667fbcd9bdabaa90301341e6e4ab2808510e2d (diff)
downloadFreeBSD-src-a3c8b91e8eb2355576e69f7c407f40bc82279aad.zip
FreeBSD-src-a3c8b91e8eb2355576e69f7c407f40bc82279aad.tar.gz
Fixed panics in npx exception handling. When using IRQ13 exception
handling, SMPng always switches the npx context away from curproc before calling the handler, so the handler always paniced. When using exception 16 exception handling, SMPng sometimes switches the npx context away from curproc before calling the handler, so the handler sometimes paniced. Also, we didn't lock the context while using it, so we sometimes didn't detect the switch and then paniced in a less controlled way. Just lock the context while using it, and return without doing anything except clearing the busy latch if the context is not for curproc. This fixes the exception 16 case and makes the IRQ13 case harmless. In both cases, the instruction that caused the exception is restarted and the exception repeats. In the exception 16 case, we soon get an exception that can be handled without doing anything special. In the IRQ13 case, we get an easy to kill hung process.
Diffstat (limited to 'sys/amd64/isa')
-rw-r--r--sys/amd64/isa/npx.c25
1 files changed, 18 insertions, 7 deletions
diff --git a/sys/amd64/isa/npx.c b/sys/amd64/isa/npx.c
index 91dbcad..4b04b46 100644
--- a/sys/amd64/isa/npx.c
+++ b/sys/amd64/isa/npx.c
@@ -725,26 +725,37 @@ npx_intr(dummy)
u_short control;
struct intrframe *frame;
- mtx_lock(&Giant);
- if (PCPU_GET(npxproc) == NULL || !npx_exists) {
+ if (!npx_exists) {
printf("npxintr: npxproc = %p, curproc = %p, npx_exists = %d\n",
PCPU_GET(npxproc), curproc, npx_exists);
panic("npxintr from nowhere");
}
+ outb(0xf0, 0);
+ mtx_lock_spin(&sched_lock);
if (PCPU_GET(npxproc) != curproc) {
- printf("npxintr: npxproc = %p, curproc = %p, npx_exists = %d\n",
- PCPU_GET(npxproc), curproc, npx_exists);
- panic("npxintr from non-current process");
+ /*
+ * Interrupt handling (for this or another interrupt) has
+ * switched npxproc from underneath us before we managed
+ * to handle this interrupt. Just ignore this interrupt.
+ * Control will eventually return to the instruction that
+ * caused it and it will repeat. In the npx_ex16 case,
+ * then we will eventually (usually soon) win the race.
+ * In the npx_irq13 case, we will always lose the race
+ * because we have switched to the IRQ13 thread. This will
+ * be fixed later.
+ */
+ mtx_unlock_spin(&sched_lock);
+ return;
}
-
- outb(0xf0, 0);
fnstsw(&PCPU_GET(curpcb)->pcb_savefpu.sv_ex_sw);
fnstcw(&control);
fnclex();
+ mtx_unlock_spin(&sched_lock);
/*
* Pass exception to process.
*/
+ mtx_lock(&Giant);
frame = (struct intrframe *)&dummy; /* XXX */
if ((ISPL(frame->if_cs) == SEL_UPL) || (frame->if_eflags & PSL_VM)) {
/*
OpenPOWER on IntegriCloud