Skip to content
Snippets Groups Projects
lapic.c 4.4 KiB
Newer Older
rsc's avatar
rsc committed
// The local APIC manages internal (non-I/O) interrupts.
// See Chapter 8 & Appendix C of Intel processor manual volume 3.

kaashoek's avatar
kaashoek committed
#include "types.h"
rsc's avatar
 
rsc committed
#include "defs.h"
kaashoek's avatar
kaashoek committed
#include "traps.h"
rsc's avatar
 
rsc committed
#include "mmu.h"
#include "x86.h"
rsc's avatar
 
rsc committed

// Local APIC registers, divided by 4 for use as uint[] indices.
#define ID      (0x0020/4)   // ID
#define VER     (0x0030/4)   // Version
#define TPR     (0x0080/4)   // Task Priority
#define EOI     (0x00B0/4)   // EOI
#define SVR     (0x00F0/4)   // Spurious Interrupt Vector
rsc's avatar
rsc committed
  #define ENABLE     0x00000100   // Unit Enable
rsc's avatar
 
rsc committed
#define ESR     (0x0280/4)   // Error Status
#define ICRLO   (0x0300/4)   // Interrupt Command
rsc's avatar
rsc committed
  #define INIT       0x00000500   // INIT/RESET
  #define STARTUP    0x00000600   // Startup IPI
  #define DELIVS     0x00001000   // Delivery status
  #define ASSERT     0x00004000   // Assert interrupt (vs deassert)
  #define LEVEL      0x00008000   // Level triggered
  #define BCAST      0x00080000   // Send to all APICs, including self.
rsc's avatar
 
rsc committed
#define ICRHI   (0x0310/4)   // Interrupt Command [63:32]
#define TIMER   (0x0320/4)   // Local Vector Table 0 (TIMER)
rsc's avatar
rsc committed
  #define X1         0x0000000B   // divide counts by 1
  #define PERIODIC   0x00020000   // Periodic
rsc's avatar
 
rsc committed
#define PCINT   (0x0340/4)   // Performance Counter LVT
#define LINT0   (0x0350/4)   // Local Vector Table 1 (LINT0)
#define LINT1   (0x0360/4)   // Local Vector Table 2 (LINT1)
#define ERROR   (0x0370/4)   // Local Vector Table 3 (ERROR)
rsc's avatar
rsc committed
  #define MASKED     0x00010000   // Interrupt masked
rsc's avatar
 
rsc committed
#define TICR    (0x0380/4)   // Timer Initial Count
#define TCCR    (0x0390/4)   // Timer Current Count
#define TDCR    (0x03E0/4)   // Timer Divide Configuration

volatile uint *lapic;  // Initialized in mp.c
kaashoek's avatar
kaashoek committed

rsc's avatar
rsc committed
//PAGEBREAK!
kaashoek's avatar
kaashoek committed
void
lapic_init(int c)
{
rsc's avatar
 
rsc committed
  if(!lapic) 
rsc's avatar
rsc committed
  // Enable local APIC; set spurious interrupt vector.
rsc's avatar
 
rsc committed
  lapic[SVR] = ENABLE | (IRQ_OFFSET+IRQ_SPURIOUS);
kaashoek's avatar
kaashoek committed

rsc's avatar
rsc committed
  // The timer repeatedly counts down at bus frequency
  // from lapic[TICR] and then issues an interrupt.  
rsc's avatar
 
rsc committed
  // If xv6 cared more about precise timekeeping,
  // TICR would be calibrated using an external time source.
rsc's avatar
rsc committed
  lapic[TDCR] = X1;
  lapic[TIMER] = PERIODIC | (IRQ_OFFSET + IRQ_TIMER);
rsc's avatar
 
rsc committed
  lapic[TICR] = 10000000; 
kaashoek's avatar
kaashoek committed

rsc's avatar
rsc committed
  // Disable logical interrupt lines.
  lapic[LINT0] = MASKED;
  lapic[LINT1] = MASKED;
kaashoek's avatar
kaashoek committed

rsc's avatar
rsc committed
  // Disable performance counter overflow interrupts
  // on machines that provide that interrupt entry.
  if(((lapic[VER]>>16) & 0xFF) >= 4)
    lapic[PCINT] = MASKED;

  // Map error interrupt to IRQ_ERROR.
rsc's avatar
 
rsc committed
  lapic[ERROR] = IRQ_OFFSET+IRQ_ERROR;
rsc's avatar
rsc committed

  // Clear error status register (requires back-to-back writes).
rsc's avatar
 
rsc committed
  lapic[ESR] = 0;
rsc's avatar
rsc committed
  lapic[ESR] = 0;

  // Ack any outstanding interrupts.
  lapic[EOI] = 0;
kaashoek's avatar
kaashoek committed

rsc's avatar
rsc committed
  // Send an Init Level De-Assert to synchronise arbitration ID's.
rsc's avatar
 
rsc committed
  lapic[ICRHI] = 0;
rsc's avatar
rsc committed
  lapic[ICRLO] = BCAST | INIT | LEVEL;
  while(lapic[ICRLO] & DELIVS)
kaashoek's avatar
kaashoek committed
    ;

rsc's avatar
rsc committed
  // Enable interrupts on the APIC (but not on the processor).
rsc's avatar
 
rsc committed
  lapic[TPR] = 0;
kaashoek's avatar
kaashoek committed
}

rsc's avatar
 
rsc committed
int
cpu(void)
kaashoek's avatar
kaashoek committed
{
rsc's avatar
 
rsc committed
  // Cannot call cpu when interrupts are enabled:
  // result not guaranteed to last long enough to be used!
  // Would prefer to panic but even printing is chancy here:
  // everything, including cprintf, calls cpu, at least indirectly
  // through acquire and release.
  if(read_eflags()&FL_IF){
    static int n;
    int i;
    uint pcs[10];

    if(n++%999 == 0){
      getcallerpcs((uint*)read_ebp() + 2, pcs);
      cprintf("cpu called from %x with interrupts enabled: stk");
      for(i=0; i<10 && pcs[i] && pcs[i] != -1; i++)
        cprintf(" %x", pcs[i]);
      cprintf("\n");
    }
  }

rsc's avatar
 
rsc committed
  if(lapic)
    return lapic[ID]>>24;
  return 0;
kaashoek's avatar
kaashoek committed
}

rsc's avatar
 
rsc committed
// Acknowledge interrupt.
rsc's avatar
 
rsc committed
  if(lapic)
    lapic[EOI] = 0;
kaashoek's avatar
kaashoek committed
}

rsc's avatar
rsc committed
// Spin for a given number of microseconds.
// On real hardware would want to tune this dynamically.
static void
microdelay(int us)
{
  volatile int j = 0;
  
  while(us-- > 0)
    for(j=0; j<10000; j++);
}

rsc's avatar
 
rsc committed
// Start additional processor running bootstrap code at addr.
rsc's avatar
rsc committed
// See Appendix B of MultiProcessor Specification.
kaashoek's avatar
kaashoek committed
void
rsc's avatar
 
rsc committed
lapic_startap(uchar apicid, uint addr)
kaashoek's avatar
kaashoek committed
{
rsc's avatar
 
rsc committed
  int i;
kaashoek's avatar
kaashoek committed
  volatile int j = 0;

rsc's avatar
rsc committed
  // Send INIT interrupt to reset other CPU.
rsc's avatar
 
rsc committed
  lapic[ICRHI] = apicid<<24;
rsc's avatar
rsc committed
  lapic[ICRLO] = INIT | LEVEL;
  microdelay(10);
  
  // Send startup IPI (twice!) to enter bootstrap code.
rsc's avatar
rsc committed
  // Regular hardware wants it twice, but Bochs complains.
  // Too bad for Bochs.
rsc's avatar
 
rsc committed
  for(i = 0; i < 2; i++){
    lapic[ICRHI] = apicid<<24;
rsc's avatar
rsc committed
    lapic[ICRLO] = STARTUP | (addr>>12);
    for(j=0; j<10000; j++);  // 200us
kaashoek's avatar
kaashoek committed
  }
}