mirror of
https://git.FreeBSD.org/src.git
synced 2025-01-28 16:43:09 +00:00
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.
This commit is contained in:
parent
7f17ea13a3
commit
79d4e25bea
Notes:
svn2git
2020-12-20 02:59:44 +00:00
svn path=/head/; revision=76205
@ -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");
|
||||
}
|
||||
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");
|
||||
}
|
||||
|
||||
outb(0xf0, 0);
|
||||
mtx_lock_spin(&sched_lock);
|
||||
if (PCPU_GET(npxproc) != curproc) {
|
||||
/*
|
||||
* 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;
|
||||
}
|
||||
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)) {
|
||||
/*
|
||||
|
@ -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");
|
||||
}
|
||||
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");
|
||||
}
|
||||
|
||||
outb(0xf0, 0);
|
||||
mtx_lock_spin(&sched_lock);
|
||||
if (PCPU_GET(npxproc) != curproc) {
|
||||
/*
|
||||
* 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;
|
||||
}
|
||||
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)) {
|
||||
/*
|
||||
|
@ -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");
|
||||
}
|
||||
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");
|
||||
}
|
||||
|
||||
outb(0xf0, 0);
|
||||
mtx_lock_spin(&sched_lock);
|
||||
if (PCPU_GET(npxproc) != curproc) {
|
||||
/*
|
||||
* 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;
|
||||
}
|
||||
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)) {
|
||||
/*
|
||||
|
Loading…
Reference in New Issue
Block a user