1997-05-26 17:58:27 +00:00
|
|
|
/*
|
|
|
|
* from: vector.s, 386BSD 0.1 unknown origin
|
1998-01-15 07:34:01 +00:00
|
|
|
* $Id: apic_vector.s,v 1.24 1997/12/08 22:59:34 fsmp Exp $
|
1997-05-26 17:58:27 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
|
1997-08-10 20:59:07 +00:00
|
|
|
#include <machine/apic.h>
|
1997-07-28 03:59:54 +00:00
|
|
|
#include <machine/smp.h>
|
1997-08-10 20:59:07 +00:00
|
|
|
|
1997-07-18 21:27:53 +00:00
|
|
|
#include "i386/isa/intr_machdep.h"
|
1997-06-27 23:48:05 +00:00
|
|
|
|
1997-08-10 20:59:07 +00:00
|
|
|
|
1997-08-20 05:25:48 +00:00
|
|
|
#ifdef FAST_SIMPLELOCK
|
|
|
|
|
|
|
|
#define GET_FAST_INTR_LOCK \
|
|
|
|
pushl $_fast_intr_lock ; /* address of lock */ \
|
|
|
|
call _s_lock ; /* MP-safe */ \
|
|
|
|
addl $4,%esp
|
|
|
|
|
|
|
|
#define REL_FAST_INTR_LOCK \
|
|
|
|
pushl $_fast_intr_lock ; /* address of lock */ \
|
|
|
|
call _s_unlock ; /* MP-safe */ \
|
|
|
|
addl $4,%esp
|
|
|
|
|
|
|
|
#else /* FAST_SIMPLELOCK */
|
|
|
|
|
|
|
|
#define GET_FAST_INTR_LOCK \
|
|
|
|
call _get_isrlock
|
1997-08-21 05:08:25 +00:00
|
|
|
|
1997-08-20 05:25:48 +00:00
|
|
|
#define REL_FAST_INTR_LOCK \
|
|
|
|
pushl $_mp_lock ; /* GIANT_LOCK */ \
|
|
|
|
call _MPrellock ; \
|
|
|
|
add $4, %esp
|
|
|
|
|
|
|
|
#endif /* FAST_SIMPLELOCK */
|
|
|
|
|
1997-05-26 17:58:27 +00:00
|
|
|
/* convert an absolute IRQ# into a bitmask */
|
|
|
|
#define IRQ_BIT(irq_num) (1 << (irq_num))
|
|
|
|
|
|
|
|
/* make an index into the IO APIC from the IRQ# */
|
|
|
|
#define REDTBL_IDX(irq_num) (0x10 + ((irq_num) * 2))
|
|
|
|
|
1997-08-20 05:25:48 +00:00
|
|
|
|
1997-05-26 17:58:27 +00:00
|
|
|
/*
|
|
|
|
* Macros for interrupt interrupt entry, call to handler, and exit.
|
|
|
|
*/
|
|
|
|
|
1997-08-20 05:25:48 +00:00
|
|
|
#ifdef FAST_WITHOUTCPL
|
|
|
|
|
1997-08-21 05:08:25 +00:00
|
|
|
/*
|
|
|
|
*/
|
1997-05-26 17:58:27 +00:00
|
|
|
#define FAST_INTR(irq_num, vec_name) \
|
|
|
|
.text ; \
|
|
|
|
SUPERALIGN_TEXT ; \
|
|
|
|
IDTVEC(vec_name) ; \
|
|
|
|
pushl %eax ; /* save only call-used registers */ \
|
|
|
|
pushl %ecx ; \
|
|
|
|
pushl %edx ; \
|
|
|
|
pushl %ds ; \
|
|
|
|
MAYBE_PUSHL_ES ; \
|
|
|
|
movl $KDSEL,%eax ; \
|
|
|
|
movl %ax,%ds ; \
|
|
|
|
MAYBE_MOVW_AX_ES ; \
|
|
|
|
FAKE_MCOUNT((4+ACTUALLY_PUSHED)*4(%esp)) ; \
|
|
|
|
pushl _intr_unit + (irq_num) * 4 ; \
|
1997-08-30 08:08:10 +00:00
|
|
|
GET_FAST_INTR_LOCK ; \
|
1997-05-26 17:58:27 +00:00
|
|
|
call *_intr_handler + (irq_num) * 4 ; /* do the work ASAP */ \
|
1997-08-30 08:08:10 +00:00
|
|
|
REL_FAST_INTR_LOCK ; \
|
1997-08-20 05:25:48 +00:00
|
|
|
addl $4, %esp ; \
|
1997-07-15 02:49:21 +00:00
|
|
|
movl $0, lapic_eoi ; \
|
1997-08-20 05:25:48 +00:00
|
|
|
lock ; \
|
|
|
|
incl _cnt+V_INTR ; /* book-keeping can wait */ \
|
|
|
|
movl _intr_countp + (irq_num) * 4, %eax ; \
|
|
|
|
lock ; \
|
|
|
|
incl (%eax) ; \
|
|
|
|
MEXITCOUNT ; \
|
|
|
|
MAYBE_POPL_ES ; \
|
|
|
|
popl %ds ; \
|
|
|
|
popl %edx ; \
|
|
|
|
popl %ecx ; \
|
|
|
|
popl %eax ; \
|
|
|
|
iret
|
|
|
|
|
1997-08-24 00:05:37 +00:00
|
|
|
#else /* FAST_WITHOUTCPL */
|
1997-08-20 05:25:48 +00:00
|
|
|
|
|
|
|
#define FAST_INTR(irq_num, vec_name) \
|
|
|
|
.text ; \
|
|
|
|
SUPERALIGN_TEXT ; \
|
|
|
|
IDTVEC(vec_name) ; \
|
|
|
|
pushl %eax ; /* save only call-used registers */ \
|
|
|
|
pushl %ecx ; \
|
|
|
|
pushl %edx ; \
|
|
|
|
pushl %ds ; \
|
|
|
|
MAYBE_PUSHL_ES ; \
|
1997-08-24 00:05:37 +00:00
|
|
|
movl $KDSEL, %eax ; \
|
|
|
|
movl %ax, %ds ; \
|
1997-08-20 05:25:48 +00:00
|
|
|
MAYBE_MOVW_AX_ES ; \
|
|
|
|
FAKE_MCOUNT((4+ACTUALLY_PUSHED)*4(%esp)) ; \
|
|
|
|
GET_FAST_INTR_LOCK ; \
|
|
|
|
pushl _intr_unit + (irq_num) * 4 ; \
|
|
|
|
call *_intr_handler + (irq_num) * 4 ; /* do the work ASAP */ \
|
1997-08-24 00:05:37 +00:00
|
|
|
addl $4, %esp ; \
|
1997-08-20 05:25:48 +00:00
|
|
|
movl $0, lapic_eoi ; \
|
1997-08-24 00:05:37 +00:00
|
|
|
lock ; \
|
1997-05-26 17:58:27 +00:00
|
|
|
incl _cnt+V_INTR ; /* book-keeping can wait */ \
|
|
|
|
movl _intr_countp + (irq_num) * 4,%eax ; \
|
1997-08-24 00:05:37 +00:00
|
|
|
lock ; \
|
1997-05-26 17:58:27 +00:00
|
|
|
incl (%eax) ; \
|
1997-08-24 00:05:37 +00:00
|
|
|
movl _cpl, %eax ; /* unmasking pending HWIs or SWIs? */ \
|
1997-05-26 17:58:27 +00:00
|
|
|
notl %eax ; \
|
1997-08-24 00:05:37 +00:00
|
|
|
andl _ipending, %eax ; \
|
1997-05-26 17:58:27 +00:00
|
|
|
jne 2f ; /* yes, maybe handle them */ \
|
|
|
|
1: ; \
|
|
|
|
MEXITCOUNT ; \
|
1997-08-20 05:25:48 +00:00
|
|
|
REL_FAST_INTR_LOCK ; \
|
1997-05-26 17:58:27 +00:00
|
|
|
MAYBE_POPL_ES ; \
|
|
|
|
popl %ds ; \
|
|
|
|
popl %edx ; \
|
|
|
|
popl %ecx ; \
|
|
|
|
popl %eax ; \
|
|
|
|
iret ; \
|
|
|
|
; \
|
|
|
|
ALIGN_TEXT ; \
|
|
|
|
2: ; \
|
1997-08-24 00:05:37 +00:00
|
|
|
cmpb $3, _intr_nesting_level ; /* enough stack? */ \
|
1997-05-26 17:58:27 +00:00
|
|
|
jae 1b ; /* no, return */ \
|
1997-08-24 00:05:37 +00:00
|
|
|
movl _cpl, %eax ; \
|
1997-05-26 17:58:27 +00:00
|
|
|
/* XXX next line is probably unnecessary now. */ \
|
1997-08-24 00:05:37 +00:00
|
|
|
movl $HWI_MASK|SWI_MASK, _cpl ; /* limit nesting ... */ \
|
|
|
|
lock ; \
|
1997-05-26 17:58:27 +00:00
|
|
|
incb _intr_nesting_level ; /* ... really limit it ... */ \
|
|
|
|
sti ; /* to do this as early as possible */ \
|
|
|
|
MAYBE_POPL_ES ; /* discard most of thin frame ... */ \
|
|
|
|
popl %ecx ; /* ... original %ds ... */ \
|
|
|
|
popl %edx ; \
|
1997-08-24 00:05:37 +00:00
|
|
|
xchgl %eax, 4(%esp) ; /* orig %eax; save cpl */ \
|
1997-05-26 17:58:27 +00:00
|
|
|
pushal ; /* build fat frame (grrr) ... */ \
|
|
|
|
pushl %ecx ; /* ... actually %ds ... */ \
|
|
|
|
pushl %es ; \
|
1997-08-24 00:05:37 +00:00
|
|
|
movl $KDSEL, %eax ; \
|
|
|
|
movl %ax, %es ; \
|
|
|
|
movl (2+8+0)*4(%esp), %ecx ; /* %ecx from thin frame ... */ \
|
|
|
|
movl %ecx, (2+6)*4(%esp) ; /* ... to fat frame ... */ \
|
|
|
|
movl (2+8+1)*4(%esp), %eax ; /* ... cpl from thin frame */ \
|
1997-05-26 17:58:27 +00:00
|
|
|
pushl %eax ; \
|
1997-08-24 00:05:37 +00:00
|
|
|
subl $4, %esp ; /* junk for unit number */ \
|
1997-05-26 17:58:27 +00:00
|
|
|
MEXITCOUNT ; \
|
|
|
|
jmp _doreti
|
|
|
|
|
1997-08-20 05:25:48 +00:00
|
|
|
#endif /** FAST_WITHOUTCPL */
|
|
|
|
|
|
|
|
|
1997-08-21 05:08:25 +00:00
|
|
|
/*
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
#define PUSH_FRAME \
|
1997-05-26 17:58:27 +00:00
|
|
|
pushl $0 ; /* dummy error code */ \
|
|
|
|
pushl $0 ; /* dummy trap type */ \
|
|
|
|
pushal ; \
|
|
|
|
pushl %ds ; /* save data and extra segments ... */ \
|
1997-08-21 05:08:25 +00:00
|
|
|
pushl %es
|
|
|
|
|
|
|
|
#define POP_FRAME \
|
|
|
|
popl %es ; \
|
|
|
|
popl %ds ; \
|
|
|
|
popal ; \
|
|
|
|
addl $4+4,%esp
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Test to see whether we are handling an edge or level triggered INT.
|
|
|
|
* Level-triggered INTs must still be masked as we don't clear the source,
|
|
|
|
* and the EOI cycle would cause redundant INTs to occur.
|
|
|
|
*/
|
|
|
|
#define MASK_LEVEL_IRQ(irq_num) \
|
|
|
|
testl $IRQ_BIT(irq_num), _apic_pin_trigger ; \
|
|
|
|
jz 8f ; /* edge, don't mask */ \
|
1997-09-07 22:04:09 +00:00
|
|
|
IMASK_LOCK ; /* into critical reg */ \
|
1997-08-21 05:08:25 +00:00
|
|
|
orl $IRQ_BIT(irq_num), _apic_imen ; /* set the mask bit */ \
|
|
|
|
movl _ioapic, %ecx ; /* ioapic[0] addr */ \
|
|
|
|
movl $REDTBL_IDX(irq_num), (%ecx) ; /* write the index */ \
|
|
|
|
movl IOAPIC_WINDOW(%ecx), %eax ; /* current value */ \
|
|
|
|
orl $IOART_INTMASK, %eax ; /* set the mask */ \
|
|
|
|
movl %eax, IOAPIC_WINDOW(%ecx) ; /* new value */ \
|
1997-09-07 22:04:09 +00:00
|
|
|
IMASK_UNLOCK ; \
|
|
|
|
8:
|
1997-08-21 05:08:25 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Test to see if the source is currntly masked, clear if so.
|
|
|
|
*/
|
|
|
|
#define UNMASK_IRQ(irq_num) \
|
|
|
|
IMASK_LOCK ; /* into critical reg */ \
|
|
|
|
testl $IRQ_BIT(irq_num), _apic_imen ; \
|
1997-12-08 22:59:39 +00:00
|
|
|
jne 7f ; /* bit set, masked */ \
|
|
|
|
testl $IRQ_BIT(irq_num), _apic_pin_trigger ; \
|
|
|
|
jz 9f ; /* edge, don't EOI */ \
|
|
|
|
movl $0, lapic_eoi ; /* should be safe */ \
|
|
|
|
jmp 9f ; /* skip unmasking */ \
|
|
|
|
7: \
|
1997-08-21 05:08:25 +00:00
|
|
|
andl $~IRQ_BIT(irq_num), _apic_imen ;/* clear mask bit */ \
|
|
|
|
movl _ioapic,%ecx ; /* ioapic[0]addr */ \
|
|
|
|
movl $REDTBL_IDX(irq_num),(%ecx) ; /* write the index */ \
|
|
|
|
movl IOAPIC_WINDOW(%ecx),%eax ; /* current value */ \
|
|
|
|
andl $~IOART_INTMASK,%eax ; /* clear the mask */ \
|
|
|
|
movl %eax,IOAPIC_WINDOW(%ecx) ; /* new value */ \
|
|
|
|
9: ; \
|
|
|
|
IMASK_UNLOCK
|
|
|
|
|
1997-08-29 18:45:23 +00:00
|
|
|
#ifdef INTR_SIMPLELOCK
|
1997-09-07 22:04:09 +00:00
|
|
|
#define ENLOCK
|
|
|
|
#define DELOCK
|
|
|
|
#define LATELOCK call _get_isrlock
|
|
|
|
#else
|
|
|
|
#define ENLOCK \
|
|
|
|
ISR_TRYLOCK ; /* XXX this is going away... */ \
|
|
|
|
testl %eax, %eax ; /* did we get it? */ \
|
|
|
|
jz 1f
|
|
|
|
#define DELOCK ISR_RELLOCK
|
|
|
|
#define LATELOCK
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef CPL_AND_CML
|
1997-08-29 18:45:23 +00:00
|
|
|
|
1997-08-21 05:08:25 +00:00
|
|
|
#define INTR(irq_num, vec_name) \
|
|
|
|
.text ; \
|
|
|
|
SUPERALIGN_TEXT ; \
|
1997-09-07 22:04:09 +00:00
|
|
|
/* _XintrNN: entry point used by IDT/HWIs & splz_unpend via _vec[]. */ \
|
1997-08-21 05:08:25 +00:00
|
|
|
IDTVEC(vec_name) ; \
|
|
|
|
PUSH_FRAME ; \
|
|
|
|
movl $KDSEL, %eax ; /* reload with kernel's data segment */ \
|
|
|
|
movl %ax, %ds ; \
|
|
|
|
movl %ax, %es ; \
|
|
|
|
; \
|
|
|
|
lock ; /* MP-safe */ \
|
|
|
|
btsl $(irq_num), iactive ; /* lazy masking */ \
|
|
|
|
jc 1f ; /* already active */ \
|
|
|
|
; \
|
1997-09-07 22:04:09 +00:00
|
|
|
ENLOCK ; \
|
1997-08-21 05:08:25 +00:00
|
|
|
; \
|
1997-08-24 00:05:37 +00:00
|
|
|
AVCPL_LOCK ; /* MP-safe */ \
|
1997-08-21 05:08:25 +00:00
|
|
|
testl $IRQ_BIT(irq_num), _cpl ; \
|
1997-08-24 00:05:37 +00:00
|
|
|
jne 2f ; /* this INT masked */ \
|
1997-08-29 18:45:23 +00:00
|
|
|
testl $IRQ_BIT(irq_num), _cml ; \
|
|
|
|
jne 2f ; /* this INT masked */ \
|
1997-08-21 05:08:25 +00:00
|
|
|
orl $IRQ_BIT(irq_num), _cil ; \
|
1997-08-24 00:05:37 +00:00
|
|
|
AVCPL_UNLOCK ; \
|
1997-08-21 05:08:25 +00:00
|
|
|
; \
|
1997-12-08 22:59:39 +00:00
|
|
|
;;; movl $0, lapic_eoi ; /* XXX too soon? */ \
|
1997-05-26 17:58:27 +00:00
|
|
|
incb _intr_nesting_level ; \
|
1997-09-07 22:04:09 +00:00
|
|
|
; \
|
|
|
|
/* entry point used by doreti_unpend for HWIs. */ \
|
1997-05-26 17:58:27 +00:00
|
|
|
__CONCAT(Xresume,irq_num): ; \
|
1997-08-21 05:08:25 +00:00
|
|
|
FAKE_MCOUNT(12*4(%esp)) ; /* XXX avoid dbl cnt */ \
|
|
|
|
lock ; incl _cnt+V_INTR ; /* tally interrupts */ \
|
|
|
|
movl _intr_countp + (irq_num) * 4, %eax ; \
|
|
|
|
lock ; incl (%eax) ; \
|
1997-08-29 18:45:23 +00:00
|
|
|
; \
|
|
|
|
AVCPL_LOCK ; /* MP-safe */ \
|
|
|
|
movl _cml, %eax ; \
|
|
|
|
pushl %eax ; \
|
|
|
|
orl _intr_mask + (irq_num) * 4, %eax ; \
|
|
|
|
movl %eax, _cml ; \
|
|
|
|
AVCPL_UNLOCK ; \
|
|
|
|
; \
|
|
|
|
pushl _intr_unit + (irq_num) * 4 ; \
|
1997-09-07 22:04:09 +00:00
|
|
|
incl _inside_intr ; \
|
1997-08-29 18:45:23 +00:00
|
|
|
sti ; \
|
|
|
|
call *_intr_handler + (irq_num) * 4 ; \
|
|
|
|
cli ; \
|
1997-09-07 22:04:09 +00:00
|
|
|
decl _inside_intr ; \
|
1997-08-29 18:45:23 +00:00
|
|
|
; \
|
|
|
|
lock ; andl $~IRQ_BIT(irq_num), iactive ; \
|
|
|
|
lock ; andl $~IRQ_BIT(irq_num), _cil ; \
|
|
|
|
UNMASK_IRQ(irq_num) ; \
|
|
|
|
sti ; /* doreti repeats cli/sti */ \
|
|
|
|
MEXITCOUNT ; \
|
1997-09-07 22:04:09 +00:00
|
|
|
LATELOCK ; \
|
1997-08-29 18:45:23 +00:00
|
|
|
jmp _doreti ; \
|
|
|
|
; \
|
|
|
|
ALIGN_TEXT ; \
|
|
|
|
1: ; /* active or locked */ \
|
|
|
|
MASK_LEVEL_IRQ(irq_num) ; \
|
|
|
|
movl $0, lapic_eoi ; /* do the EOI */ \
|
|
|
|
; \
|
|
|
|
AVCPL_LOCK ; /* MP-safe */ \
|
|
|
|
orl $IRQ_BIT(irq_num), _ipending ; \
|
|
|
|
AVCPL_UNLOCK ; \
|
|
|
|
; \
|
|
|
|
POP_FRAME ; \
|
|
|
|
iret ; \
|
|
|
|
; \
|
|
|
|
ALIGN_TEXT ; \
|
|
|
|
2: ; /* masked by cpl|cml */ \
|
|
|
|
AVCPL_UNLOCK ; \
|
1997-09-07 22:04:09 +00:00
|
|
|
DELOCK ; /* XXX this is going away... */ \
|
1997-08-29 18:45:23 +00:00
|
|
|
jmp 1b
|
|
|
|
|
1997-09-07 22:04:09 +00:00
|
|
|
#else /* CPL_AND_CML */
|
1997-08-29 18:45:23 +00:00
|
|
|
|
|
|
|
#define INTR(irq_num, vec_name) \
|
|
|
|
.text ; \
|
|
|
|
SUPERALIGN_TEXT ; \
|
1997-09-07 22:04:09 +00:00
|
|
|
/* _XintrNN: entry point used by IDT/HWIs & splz_unpend via _vec[]. */ \
|
1997-08-29 18:45:23 +00:00
|
|
|
IDTVEC(vec_name) ; \
|
|
|
|
PUSH_FRAME ; \
|
|
|
|
movl $KDSEL, %eax ; /* reload with kernel's data segment */ \
|
|
|
|
movl %ax, %ds ; \
|
|
|
|
movl %ax, %es ; \
|
|
|
|
; \
|
|
|
|
lock ; /* MP-safe */ \
|
|
|
|
btsl $(irq_num), iactive ; /* lazy masking */ \
|
|
|
|
jc 1f ; /* already active */ \
|
|
|
|
; \
|
|
|
|
ISR_TRYLOCK ; /* XXX this is going away... */ \
|
|
|
|
testl %eax, %eax ; /* did we get it? */ \
|
|
|
|
jz 1f ; /* no */ \
|
|
|
|
; \
|
|
|
|
AVCPL_LOCK ; /* MP-safe */ \
|
|
|
|
testl $IRQ_BIT(irq_num), _cpl ; \
|
|
|
|
jne 2f ; /* this INT masked */ \
|
|
|
|
AVCPL_UNLOCK ; \
|
|
|
|
; \
|
1997-12-08 22:59:39 +00:00
|
|
|
;;; movl $0, lapic_eoi ; /* XXX too soon? */ \
|
1997-08-29 18:45:23 +00:00
|
|
|
incb _intr_nesting_level ; \
|
1997-09-07 22:04:09 +00:00
|
|
|
; \
|
|
|
|
/* entry point used by doreti_unpend for HWIs. */ \
|
1997-08-29 18:45:23 +00:00
|
|
|
__CONCAT(Xresume,irq_num): ; \
|
|
|
|
FAKE_MCOUNT(12*4(%esp)) ; /* XXX avoid dbl cnt */ \
|
|
|
|
lock ; incl _cnt+V_INTR ; /* tally interrupts */ \
|
|
|
|
movl _intr_countp + (irq_num) * 4, %eax ; \
|
|
|
|
lock ; incl (%eax) ; \
|
1997-08-21 05:08:25 +00:00
|
|
|
; \
|
1997-08-24 00:05:37 +00:00
|
|
|
AVCPL_LOCK ; /* MP-safe */ \
|
1997-08-21 05:08:25 +00:00
|
|
|
movl _cpl, %eax ; \
|
1997-05-26 17:58:27 +00:00
|
|
|
pushl %eax ; \
|
1997-08-21 05:08:25 +00:00
|
|
|
orl _intr_mask + (irq_num) * 4, %eax ; \
|
|
|
|
movl %eax, _cpl ; \
|
1997-08-24 00:05:37 +00:00
|
|
|
AVCPL_UNLOCK ; \
|
1997-08-21 05:08:25 +00:00
|
|
|
; \
|
1997-05-26 17:58:27 +00:00
|
|
|
pushl _intr_unit + (irq_num) * 4 ; \
|
|
|
|
sti ; \
|
|
|
|
call *_intr_handler + (irq_num) * 4 ; \
|
1997-08-21 05:08:25 +00:00
|
|
|
cli ; \
|
|
|
|
; \
|
|
|
|
lock ; andl $~IRQ_BIT(irq_num), iactive ; \
|
|
|
|
UNMASK_IRQ(irq_num) ; \
|
|
|
|
sti ; /* doreti repeats cli/sti */ \
|
1997-05-26 17:58:27 +00:00
|
|
|
MEXITCOUNT ; \
|
|
|
|
jmp _doreti ; \
|
|
|
|
; \
|
|
|
|
ALIGN_TEXT ; \
|
1997-08-21 05:08:25 +00:00
|
|
|
1: ; /* active or locked */ \
|
|
|
|
MASK_LEVEL_IRQ(irq_num) ; \
|
|
|
|
movl $0, lapic_eoi ; /* do the EOI */ \
|
|
|
|
; \
|
1997-08-24 00:05:37 +00:00
|
|
|
AVCPL_LOCK ; /* MP-safe */ \
|
1997-05-26 17:58:27 +00:00
|
|
|
orl $IRQ_BIT(irq_num), _ipending ; \
|
1997-08-24 00:05:37 +00:00
|
|
|
AVCPL_UNLOCK ; \
|
1997-08-21 05:08:25 +00:00
|
|
|
; \
|
|
|
|
POP_FRAME ; \
|
|
|
|
iret ; \
|
|
|
|
; \
|
|
|
|
ALIGN_TEXT ; \
|
|
|
|
2: ; /* masked by cpl */ \
|
1997-08-24 00:05:37 +00:00
|
|
|
AVCPL_UNLOCK ; \
|
1997-08-21 05:08:25 +00:00
|
|
|
ISR_RELLOCK ; /* XXX this is going away... */ \
|
|
|
|
jmp 1b
|
1997-05-26 17:58:27 +00:00
|
|
|
|
1997-09-07 22:04:09 +00:00
|
|
|
#endif /* CPL_AND_CML */
|
1997-08-29 18:45:23 +00:00
|
|
|
|
1997-07-06 23:32:38 +00:00
|
|
|
|
1997-07-13 01:18:51 +00:00
|
|
|
/*
|
|
|
|
* Handle "spurious INTerrupts".
|
|
|
|
* Notes:
|
|
|
|
* This is different than the "spurious INTerrupt" generated by an
|
|
|
|
* 8259 PIC for missing INTs. See the APIC documentation for details.
|
|
|
|
* This routine should NOT do an 'EOI' cycle.
|
|
|
|
*/
|
|
|
|
.text
|
|
|
|
SUPERALIGN_TEXT
|
|
|
|
.globl _Xspuriousint
|
|
|
|
_Xspuriousint:
|
|
|
|
|
|
|
|
/* No EOI cycle used here */
|
|
|
|
|
|
|
|
iret
|
|
|
|
|
|
|
|
|
1997-07-06 23:32:38 +00:00
|
|
|
/*
|
|
|
|
* Handle TLB shootdowns.
|
|
|
|
*/
|
1997-05-26 17:58:27 +00:00
|
|
|
.text
|
|
|
|
SUPERALIGN_TEXT
|
|
|
|
.globl _Xinvltlb
|
|
|
|
_Xinvltlb:
|
|
|
|
pushl %eax
|
1997-07-06 23:32:38 +00:00
|
|
|
|
|
|
|
#ifdef COUNT_XINVLTLB_HITS
|
|
|
|
ss
|
|
|
|
movl _cpuid, %eax
|
|
|
|
ss
|
|
|
|
incl _xhits(,%eax,4)
|
|
|
|
#endif /* COUNT_XINVLTLB_HITS */
|
|
|
|
|
1997-05-26 17:58:27 +00:00
|
|
|
movl %cr3, %eax /* invalidate the TLB */
|
|
|
|
movl %eax, %cr3
|
1997-07-06 23:32:38 +00:00
|
|
|
|
1997-05-26 17:58:27 +00:00
|
|
|
ss /* stack segment, avoid %ds load */
|
1997-07-15 02:49:21 +00:00
|
|
|
movl $0, lapic_eoi /* End Of Interrupt to APIC */
|
1997-07-06 23:32:38 +00:00
|
|
|
|
|
|
|
popl %eax
|
|
|
|
iret
|
|
|
|
|
|
|
|
|
1997-12-08 22:59:39 +00:00
|
|
|
#ifdef BETTER_CLOCK
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Executed by a CPU when it receives an Xcpucheckstate IPI from another CPU,
|
|
|
|
*
|
|
|
|
* - Stores current cpu state in checkstate_cpustate[cpuid]
|
|
|
|
* 0 == user, 1 == sys, 2 == intr
|
|
|
|
* - Stores current process in checkstate_curproc[cpuid]
|
|
|
|
*
|
|
|
|
* - Signals its receipt by setting bit cpuid in checkstate_probed_cpus.
|
|
|
|
*
|
|
|
|
* stack: 0 -> ds, 4 -> ebx, 8 -> eax, 12 -> eip, 16 -> cs, 20 -> eflags
|
|
|
|
*/
|
|
|
|
|
|
|
|
.text
|
|
|
|
SUPERALIGN_TEXT
|
|
|
|
.globl _Xcpucheckstate
|
|
|
|
.globl _checkstate_cpustate
|
|
|
|
.globl _checkstate_curproc
|
|
|
|
.globl _checkstate_pc
|
|
|
|
_Xcpucheckstate:
|
|
|
|
pushl %eax
|
|
|
|
pushl %ebx
|
|
|
|
pushl %ds /* save current data segment */
|
|
|
|
|
|
|
|
movl $KDSEL, %eax
|
|
|
|
movl %ax, %ds /* use KERNEL data segment */
|
|
|
|
|
|
|
|
movl $0, lapic_eoi /* End Of Interrupt to APIC */
|
|
|
|
|
|
|
|
movl $0, %ebx
|
|
|
|
movl 16(%esp), %eax
|
|
|
|
andl $3, %eax
|
|
|
|
cmpl $3, %eax
|
|
|
|
je 1f
|
|
|
|
#ifdef VM86
|
|
|
|
testl $PSL_VM, 20(%esp)
|
|
|
|
jne 1f
|
|
|
|
#endif
|
|
|
|
incl %ebx /* system or interrupt */
|
|
|
|
#ifdef CPL_AND_CML
|
|
|
|
cmpl $0, _inside_intr
|
|
|
|
je 1f
|
|
|
|
incl %ebx /* interrupt */
|
|
|
|
#endif
|
|
|
|
1:
|
|
|
|
movl _cpuid, %eax
|
|
|
|
movl %ebx, _checkstate_cpustate(,%eax,4)
|
|
|
|
movl _curproc, %ebx
|
|
|
|
movl %ebx, _checkstate_curproc(,%eax,4)
|
|
|
|
movl 12(%esp), %ebx
|
|
|
|
movl %ebx, _checkstate_pc(,%eax,4)
|
|
|
|
|
|
|
|
lock /* checkstate_probed_cpus |= (1<<id) */
|
|
|
|
btsl %eax, _checkstate_probed_cpus
|
|
|
|
|
|
|
|
popl %ds /* restore previous data segment */
|
|
|
|
popl %ebx
|
|
|
|
popl %eax
|
|
|
|
iret
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Executed by a CPU when it receives an Xcpuast IPI from another CPU,
|
|
|
|
*
|
|
|
|
* - Signals its receipt by clearing bit cpuid in checkstate_need_ast.
|
|
|
|
*
|
|
|
|
* - We need a better method of triggering asts on other cpus.
|
|
|
|
*/
|
|
|
|
|
|
|
|
.text
|
|
|
|
SUPERALIGN_TEXT
|
|
|
|
.globl _Xcpuast
|
|
|
|
_Xcpuast:
|
|
|
|
PUSH_FRAME
|
|
|
|
movl $KDSEL, %eax
|
|
|
|
movl %ax, %ds /* use KERNEL data segment */
|
|
|
|
movl %ax, %es
|
|
|
|
|
|
|
|
movl _cpuid, %eax
|
|
|
|
lock /* checkstate_need_ast &= ~(1<<id) */
|
|
|
|
btrl %eax, _checkstate_need_ast
|
|
|
|
movl $0, lapic_eoi /* End Of Interrupt to APIC */
|
|
|
|
|
|
|
|
lock
|
|
|
|
btsl %eax, _checkstate_pending_ast
|
|
|
|
jc 1f
|
|
|
|
|
|
|
|
FAKE_MCOUNT(12*4(%esp))
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Giant locks do not come cheap.
|
|
|
|
* A lot of cycles are going to be wasted here.
|
|
|
|
*/
|
|
|
|
call _get_isrlock
|
|
|
|
|
|
|
|
AVCPL_LOCK
|
|
|
|
#ifdef CPL_AND_CML
|
|
|
|
movl _cml, %eax
|
|
|
|
#else
|
|
|
|
movl _cpl, %eax
|
|
|
|
#endif
|
|
|
|
pushl %eax
|
|
|
|
AVCPL_UNLOCK
|
|
|
|
lock
|
|
|
|
incb _intr_nesting_level
|
|
|
|
sti
|
|
|
|
|
|
|
|
pushl $0
|
|
|
|
|
|
|
|
lock
|
|
|
|
orl $SWI_AST_PENDING, _ipending
|
|
|
|
|
|
|
|
movl _cpuid, %eax
|
|
|
|
lock
|
|
|
|
btrl %eax, _checkstate_pending_ast
|
|
|
|
|
|
|
|
MEXITCOUNT
|
|
|
|
jmp _doreti
|
|
|
|
1:
|
|
|
|
/* We are already in the process of delivering an ast for this CPU */
|
|
|
|
POP_FRAME
|
|
|
|
iret
|
|
|
|
|
|
|
|
#endif /* BETTER_CLOCK */
|
|
|
|
|
|
|
|
|
1997-06-27 23:48:05 +00:00
|
|
|
/*
|
|
|
|
* Executed by a CPU when it receives an Xcpustop IPI from another CPU,
|
|
|
|
*
|
|
|
|
* - Signals its receipt.
|
|
|
|
* - Waits for permission to restart.
|
|
|
|
* - Signals its restart.
|
|
|
|
*/
|
|
|
|
|
|
|
|
.text
|
|
|
|
SUPERALIGN_TEXT
|
|
|
|
.globl _Xcpustop
|
|
|
|
_Xcpustop:
|
|
|
|
pushl %eax
|
|
|
|
pushl %ds /* save current data segment */
|
|
|
|
|
|
|
|
movl $KDSEL, %eax
|
|
|
|
movl %ax, %ds /* use KERNEL data segment */
|
|
|
|
|
1997-07-06 23:32:38 +00:00
|
|
|
movl _cpuid, %eax
|
1997-07-13 01:18:51 +00:00
|
|
|
|
1997-06-27 23:48:05 +00:00
|
|
|
lock
|
|
|
|
btsl %eax, _stopped_cpus /* stopped_cpus |= (1<<id) */
|
|
|
|
1:
|
|
|
|
btl %eax, _started_cpus /* while (!(started_cpus & (1<<id))) */
|
|
|
|
jnc 1b
|
|
|
|
|
|
|
|
lock
|
|
|
|
btrl %eax, _started_cpus /* started_cpus &= ~(1<<id) */
|
|
|
|
|
1997-07-15 02:49:21 +00:00
|
|
|
movl $0, lapic_eoi /* End Of Interrupt to APIC */
|
1997-06-27 23:48:05 +00:00
|
|
|
|
|
|
|
popl %ds /* restore previous data segment */
|
|
|
|
popl %eax
|
|
|
|
iret
|
1997-07-06 23:32:38 +00:00
|
|
|
|
1997-06-27 23:48:05 +00:00
|
|
|
|
1997-05-26 17:58:27 +00:00
|
|
|
MCOUNT_LABEL(bintr)
|
|
|
|
FAST_INTR(0,fastintr0)
|
|
|
|
FAST_INTR(1,fastintr1)
|
|
|
|
FAST_INTR(2,fastintr2)
|
|
|
|
FAST_INTR(3,fastintr3)
|
|
|
|
FAST_INTR(4,fastintr4)
|
|
|
|
FAST_INTR(5,fastintr5)
|
|
|
|
FAST_INTR(6,fastintr6)
|
|
|
|
FAST_INTR(7,fastintr7)
|
|
|
|
FAST_INTR(8,fastintr8)
|
|
|
|
FAST_INTR(9,fastintr9)
|
|
|
|
FAST_INTR(10,fastintr10)
|
|
|
|
FAST_INTR(11,fastintr11)
|
|
|
|
FAST_INTR(12,fastintr12)
|
|
|
|
FAST_INTR(13,fastintr13)
|
|
|
|
FAST_INTR(14,fastintr14)
|
|
|
|
FAST_INTR(15,fastintr15)
|
|
|
|
FAST_INTR(16,fastintr16)
|
|
|
|
FAST_INTR(17,fastintr17)
|
|
|
|
FAST_INTR(18,fastintr18)
|
|
|
|
FAST_INTR(19,fastintr19)
|
|
|
|
FAST_INTR(20,fastintr20)
|
|
|
|
FAST_INTR(21,fastintr21)
|
|
|
|
FAST_INTR(22,fastintr22)
|
|
|
|
FAST_INTR(23,fastintr23)
|
|
|
|
INTR(0,intr0)
|
|
|
|
INTR(1,intr1)
|
|
|
|
INTR(2,intr2)
|
|
|
|
INTR(3,intr3)
|
|
|
|
INTR(4,intr4)
|
|
|
|
INTR(5,intr5)
|
|
|
|
INTR(6,intr6)
|
|
|
|
INTR(7,intr7)
|
|
|
|
INTR(8,intr8)
|
|
|
|
INTR(9,intr9)
|
|
|
|
INTR(10,intr10)
|
|
|
|
INTR(11,intr11)
|
|
|
|
INTR(12,intr12)
|
|
|
|
INTR(13,intr13)
|
|
|
|
INTR(14,intr14)
|
|
|
|
INTR(15,intr15)
|
|
|
|
INTR(16,intr16)
|
|
|
|
INTR(17,intr17)
|
|
|
|
INTR(18,intr18)
|
|
|
|
INTR(19,intr19)
|
|
|
|
INTR(20,intr20)
|
|
|
|
INTR(21,intr21)
|
|
|
|
INTR(22,intr22)
|
|
|
|
INTR(23,intr23)
|
|
|
|
MCOUNT_LABEL(eintr)
|
|
|
|
|
|
|
|
.data
|
1997-09-07 22:04:09 +00:00
|
|
|
/*
|
|
|
|
* Addresses of interrupt handlers.
|
|
|
|
* XresumeNN: Resumption addresses for HWIs.
|
|
|
|
*/
|
1997-09-21 21:41:49 +00:00
|
|
|
.globl _ihandlers
|
|
|
|
_ihandlers:
|
1997-09-07 22:04:09 +00:00
|
|
|
ihandlers:
|
|
|
|
/*
|
|
|
|
* used by:
|
|
|
|
* ipl.s: doreti_unpend
|
|
|
|
*/
|
1997-05-26 17:58:27 +00:00
|
|
|
.long Xresume0, Xresume1, Xresume2, Xresume3
|
|
|
|
.long Xresume4, Xresume5, Xresume6, Xresume7
|
|
|
|
.long Xresume8, Xresume9, Xresume10, Xresume11
|
|
|
|
.long Xresume12, Xresume13, Xresume14, Xresume15
|
|
|
|
.long Xresume16, Xresume17, Xresume18, Xresume19
|
|
|
|
.long Xresume20, Xresume21, Xresume22, Xresume23
|
1997-09-07 22:04:09 +00:00
|
|
|
/*
|
|
|
|
* used by:
|
|
|
|
* ipl.s: doreti_unpend
|
|
|
|
* apic_ipl.s: splz_unpend
|
|
|
|
*/
|
1997-09-28 19:30:01 +00:00
|
|
|
.long swi_tty, swi_net
|
|
|
|
.long dummycamisr, dummycamisr
|
1998-01-15 07:34:01 +00:00
|
|
|
.long _swi_vm, 0
|
1997-05-31 08:59:51 +00:00
|
|
|
.long _softclock, swi_ast
|
1997-05-26 17:58:27 +00:00
|
|
|
|
|
|
|
imasks: /* masks for interrupt handlers */
|
|
|
|
.space NHWI*4 /* padding; HWI masks are elsewhere */
|
|
|
|
|
1997-05-31 08:59:51 +00:00
|
|
|
.long SWI_TTY_MASK, SWI_NET_MASK
|
1997-09-21 21:41:49 +00:00
|
|
|
.long SWI_CAMNET_MASK, SWI_CAMBIO_MASK
|
1998-01-15 07:34:01 +00:00
|
|
|
.long SWI_VM_MASK, 0
|
1997-05-31 08:59:51 +00:00
|
|
|
.long SWI_CLOCK_MASK, SWI_AST_MASK
|
1997-05-26 17:58:27 +00:00
|
|
|
|
1997-09-07 22:04:09 +00:00
|
|
|
/*
|
|
|
|
* IDT vector entry points for the HWIs.
|
|
|
|
*
|
|
|
|
* used by:
|
|
|
|
* i386/isa/clock.c: setup Xintr8254
|
|
|
|
*/
|
1997-05-26 17:58:27 +00:00
|
|
|
.globl _ivectors
|
|
|
|
_ivectors:
|
|
|
|
.long _Xintr0, _Xintr1, _Xintr2, _Xintr3
|
|
|
|
.long _Xintr4, _Xintr5, _Xintr6, _Xintr7
|
|
|
|
.long _Xintr8, _Xintr9, _Xintr10, _Xintr11
|
|
|
|
.long _Xintr12, _Xintr13, _Xintr14, _Xintr15
|
|
|
|
.long _Xintr16, _Xintr17, _Xintr18, _Xintr19
|
|
|
|
.long _Xintr20, _Xintr21, _Xintr22, _Xintr23
|
|
|
|
|
|
|
|
/* active flag for lazy masking */
|
|
|
|
iactive:
|
|
|
|
.long 0
|
|
|
|
|
1997-07-06 23:32:38 +00:00
|
|
|
#ifdef COUNT_XINVLTLB_HITS
|
|
|
|
.globl _xhits
|
|
|
|
_xhits:
|
1997-07-18 21:27:53 +00:00
|
|
|
.space (NCPU * 4), 0
|
1997-07-06 23:32:38 +00:00
|
|
|
#endif /* COUNT_XINVLTLB_HITS */
|
|
|
|
|
1997-07-18 21:27:53 +00:00
|
|
|
/* variables used by stop_cpus()/restart_cpus()/Xcpustop */
|
|
|
|
.globl _stopped_cpus, _started_cpus
|
1997-06-27 23:48:05 +00:00
|
|
|
_stopped_cpus:
|
|
|
|
.long 0
|
|
|
|
_started_cpus:
|
|
|
|
.long 0
|
1997-07-06 23:32:38 +00:00
|
|
|
|
1997-12-08 22:59:39 +00:00
|
|
|
#ifdef BETTER_CLOCK
|
|
|
|
.globl _checkstate_probed_cpus
|
|
|
|
.globl _checkstate_need_ast
|
|
|
|
_checkstate_probed_cpus:
|
|
|
|
.long 0
|
|
|
|
_checkstate_need_ast:
|
|
|
|
.long 0
|
|
|
|
_checkstate_pending_ast:
|
|
|
|
.long 0
|
|
|
|
#endif
|
|
|
|
|
1997-07-28 03:59:54 +00:00
|
|
|
.globl _apic_pin_trigger
|
|
|
|
_apic_pin_trigger:
|
|
|
|
.space (NAPIC * 4), 0
|
|
|
|
|
1997-05-26 17:58:27 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Interrupt counters and names. The format of these and the label names
|
|
|
|
* must agree with what vmstat expects. The tables are indexed by device
|
|
|
|
* ids so that we don't have to move the names around as devices are
|
|
|
|
* attached.
|
|
|
|
*/
|
|
|
|
#include "vector.h"
|
|
|
|
.globl _intrcnt, _eintrcnt
|
|
|
|
_intrcnt:
|
|
|
|
.space (NR_DEVICES + ICU_LEN) * 4
|
|
|
|
_eintrcnt:
|
|
|
|
|
|
|
|
.globl _intrnames, _eintrnames
|
|
|
|
_intrnames:
|
|
|
|
.ascii DEVICE_NAMES
|
|
|
|
.asciz "stray irq0"
|
|
|
|
.asciz "stray irq1"
|
|
|
|
.asciz "stray irq2"
|
|
|
|
.asciz "stray irq3"
|
|
|
|
.asciz "stray irq4"
|
|
|
|
.asciz "stray irq5"
|
|
|
|
.asciz "stray irq6"
|
|
|
|
.asciz "stray irq7"
|
|
|
|
.asciz "stray irq8"
|
|
|
|
.asciz "stray irq9"
|
|
|
|
.asciz "stray irq10"
|
|
|
|
.asciz "stray irq11"
|
|
|
|
.asciz "stray irq12"
|
|
|
|
.asciz "stray irq13"
|
|
|
|
.asciz "stray irq14"
|
|
|
|
.asciz "stray irq15"
|
|
|
|
.asciz "stray irq16"
|
|
|
|
.asciz "stray irq17"
|
|
|
|
.asciz "stray irq18"
|
|
|
|
.asciz "stray irq19"
|
|
|
|
.asciz "stray irq20"
|
|
|
|
.asciz "stray irq21"
|
|
|
|
.asciz "stray irq22"
|
|
|
|
.asciz "stray irq23"
|
|
|
|
_eintrnames:
|
|
|
|
|
|
|
|
.text
|