better lapic writes, suggested by cliff
This commit is contained in:
parent
4721271961
commit
b5dcebdbeb
1 changed files with 26 additions and 19 deletions
45
lapic.c
45
lapic.c
|
@ -37,6 +37,13 @@
|
||||||
|
|
||||||
volatile uint *lapic; // Initialized in mp.c
|
volatile uint *lapic; // Initialized in mp.c
|
||||||
|
|
||||||
|
static void
|
||||||
|
lapicw(int index, int value)
|
||||||
|
{
|
||||||
|
lapic[index] = value;
|
||||||
|
lapic[ID]; // wait for write to finish, by reading
|
||||||
|
}
|
||||||
|
|
||||||
//PAGEBREAK!
|
//PAGEBREAK!
|
||||||
void
|
void
|
||||||
lapic_init(int c)
|
lapic_init(int c)
|
||||||
|
@ -45,43 +52,43 @@ lapic_init(int c)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
// Enable local APIC; set spurious interrupt vector.
|
// Enable local APIC; set spurious interrupt vector.
|
||||||
lapic[SVR] = ENABLE | (IRQ_OFFSET+IRQ_SPURIOUS);
|
lapicw(SVR, ENABLE | (IRQ_OFFSET+IRQ_SPURIOUS));
|
||||||
|
|
||||||
// The timer repeatedly counts down at bus frequency
|
// The timer repeatedly counts down at bus frequency
|
||||||
// from lapic[TICR] and then issues an interrupt.
|
// from lapic[TICR] and then issues an interrupt.
|
||||||
// If xv6 cared more about precise timekeeping,
|
// If xv6 cared more about precise timekeeping,
|
||||||
// TICR would be calibrated using an external time source.
|
// TICR would be calibrated using an external time source.
|
||||||
lapic[TDCR] = X1;
|
lapicw(TDCR, X1);
|
||||||
lapic[TIMER] = PERIODIC | (IRQ_OFFSET + IRQ_TIMER);
|
lapicw(TIMER, PERIODIC | (IRQ_OFFSET + IRQ_TIMER));
|
||||||
lapic[TICR] = 10000000;
|
lapicw(TICR, 10000000);
|
||||||
|
|
||||||
// Disable logical interrupt lines.
|
// Disable logical interrupt lines.
|
||||||
lapic[LINT0] = MASKED;
|
lapicw(LINT0, MASKED);
|
||||||
lapic[LINT1] = MASKED;
|
lapicw(LINT1, MASKED);
|
||||||
|
|
||||||
// Disable performance counter overflow interrupts
|
// Disable performance counter overflow interrupts
|
||||||
// on machines that provide that interrupt entry.
|
// on machines that provide that interrupt entry.
|
||||||
if(((lapic[VER]>>16) & 0xFF) >= 4)
|
if(((lapic[VER]>>16) & 0xFF) >= 4)
|
||||||
lapic[PCINT] = MASKED;
|
lapicw(PCINT, MASKED);
|
||||||
|
|
||||||
// Map error interrupt to IRQ_ERROR.
|
// Map error interrupt to IRQ_ERROR.
|
||||||
lapic[ERROR] = IRQ_OFFSET+IRQ_ERROR;
|
lapicw(ERROR, IRQ_OFFSET+IRQ_ERROR);
|
||||||
|
|
||||||
// Clear error status register (requires back-to-back writes).
|
// Clear error status register (requires back-to-back writes).
|
||||||
lapic[ESR] = 0;
|
lapicw(ESR, 0);
|
||||||
lapic[ESR] = 0;
|
lapicw(ESR, 0);
|
||||||
|
|
||||||
// Ack any outstanding interrupts.
|
// Ack any outstanding interrupts.
|
||||||
lapic[EOI] = 0;
|
lapicw(EOI, 0);
|
||||||
|
|
||||||
// Send an Init Level De-Assert to synchronise arbitration ID's.
|
// Send an Init Level De-Assert to synchronise arbitration ID's.
|
||||||
lapic[ICRHI] = 0;
|
lapicw(ICRHI, 0);
|
||||||
lapic[ICRLO] = BCAST | INIT | LEVEL;
|
lapicw(ICRLO, BCAST | INIT | LEVEL);
|
||||||
while(lapic[ICRLO] & DELIVS)
|
while(lapic[ICRLO] & DELIVS)
|
||||||
;
|
;
|
||||||
|
|
||||||
// Enable interrupts on the APIC (but not on the processor).
|
// Enable interrupts on the APIC (but not on the processor).
|
||||||
lapic[TPR] = 0;
|
lapicw(TPR, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
|
@ -116,7 +123,7 @@ void
|
||||||
lapic_eoi(void)
|
lapic_eoi(void)
|
||||||
{
|
{
|
||||||
if(lapic)
|
if(lapic)
|
||||||
lapic[EOI] = 0;
|
lapicw(EOI, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Spin for a given number of microseconds.
|
// Spin for a given number of microseconds.
|
||||||
|
@ -139,16 +146,16 @@ lapic_startap(uchar apicid, uint addr)
|
||||||
volatile int j = 0;
|
volatile int j = 0;
|
||||||
|
|
||||||
// Send INIT interrupt to reset other CPU.
|
// Send INIT interrupt to reset other CPU.
|
||||||
lapic[ICRHI] = apicid<<24;
|
lapicw(ICRHI, apicid<<24);
|
||||||
lapic[ICRLO] = INIT | LEVEL;
|
lapicw(ICRLO, INIT | LEVEL);
|
||||||
microdelay(10);
|
microdelay(10);
|
||||||
|
|
||||||
// Send startup IPI (twice!) to enter bootstrap code.
|
// Send startup IPI (twice!) to enter bootstrap code.
|
||||||
// Regular hardware wants it twice, but Bochs complains.
|
// Regular hardware wants it twice, but Bochs complains.
|
||||||
// Too bad for Bochs.
|
// Too bad for Bochs.
|
||||||
for(i = 0; i < 2; i++){
|
for(i = 0; i < 2; i++){
|
||||||
lapic[ICRHI] = apicid<<24;
|
lapicw(ICRHI, apicid<<24);
|
||||||
lapic[ICRLO] = STARTUP | (addr>>12);
|
lapicw(ICRLO, STARTUP | (addr>>12));
|
||||||
for(j=0; j<10000; j++); // 200us
|
for(j=0; j<10000; j++); // 200us
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue