2023-06-07 13:40:36 -07:00
|
|
|
#include "interrupt/apic.h"
|
|
|
|
|
|
|
|
#include <stdint.h>
|
|
|
|
|
|
|
|
#include "boot/boot_info.h"
|
|
|
|
#include "common/msr.h"
|
|
|
|
#include "common/port.h"
|
|
|
|
#include "debug/debug.h"
|
2023-08-01 20:18:47 -07:00
|
|
|
#include "interrupt/apic_timer.h"
|
2023-06-07 13:40:36 -07:00
|
|
|
|
2023-06-12 23:32:24 -07:00
|
|
|
#define APIC_DEBUG 0
|
2023-06-07 13:40:36 -07:00
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
#define IA32_APIC_BASE_MSR 0x1B
|
|
|
|
#define IA32_APIC_BASE_MSR_BSP 0x100 // Processor is a BSP
|
|
|
|
#define IA32_APIC_BASE_MSR_ENABLE 0x800
|
|
|
|
|
|
|
|
const uint64_t kEoiOffset = 0xB0;
|
2023-08-01 20:18:47 -07:00
|
|
|
const uint64_t kLvtTimerOffset = 0x320;
|
|
|
|
const uint64_t kTimerInitOffset = 0x380;
|
|
|
|
const uint64_t kTimerCurrOffset = 0x390;
|
|
|
|
const uint64_t kTimerDivOffset = 0x3E0;
|
2023-06-07 13:40:36 -07:00
|
|
|
|
2023-06-12 19:20:51 -07:00
|
|
|
// FIXME: parse these from madt.
|
2023-06-07 13:40:36 -07:00
|
|
|
constexpr uint64_t kLApicBase = 0xFEE0'0000;
|
|
|
|
constexpr uint64_t kIoApicAddr = 0xFEC0'0000;
|
|
|
|
constexpr uint64_t kIoApicData = 0xFEC0'0010;
|
|
|
|
|
|
|
|
uint32_t volatile* GetPhys(uint64_t base, uint64_t offset = 0) {
|
|
|
|
return reinterpret_cast<uint32_t*>(boot::GetHigherHalfDirectMap() + base +
|
|
|
|
offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t GetLocalReg(uint64_t offset) {
|
|
|
|
uint32_t volatile* reg = GetPhys(kLApicBase, offset);
|
|
|
|
return *reg;
|
|
|
|
}
|
|
|
|
|
|
|
|
void WriteLocalReg(uint64_t offset, uint32_t value) {
|
|
|
|
*GetPhys(kLApicBase, offset) = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t GetIoReg(uint8_t reg) {
|
|
|
|
*GetPhys(kIoApicAddr) = reg;
|
|
|
|
return *GetPhys(kIoApicData);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint64_t GetIoEntry(uint8_t reg) {
|
|
|
|
*GetPhys(kIoApicAddr) = reg;
|
|
|
|
uint64_t entry = *GetPhys(kIoApicData);
|
|
|
|
*GetPhys(kIoApicAddr) = reg + 1;
|
|
|
|
entry |= ((uint64_t)*GetPhys(kIoApicData)) << 32;
|
|
|
|
return entry;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SetIoEntry(uint8_t reg, uint64_t value) {
|
|
|
|
*GetPhys(kIoApicAddr) = reg;
|
|
|
|
*GetPhys(kIoApicData) = value & 0xFFFFFFFF;
|
|
|
|
*GetPhys(kIoApicAddr) = reg + 1;
|
|
|
|
*GetPhys(kIoApicData) = value >> 32;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define PIC1_COMMAND 0x20
|
|
|
|
#define PIC2_COMMAND 0xA0
|
|
|
|
#define PIC1_DATA 0x21
|
|
|
|
#define PIC2_DATA 0xA1
|
|
|
|
|
|
|
|
void MaskPic() {
|
|
|
|
outb(PIC1_DATA, 0xFF);
|
|
|
|
outb(PIC2_DATA, 0xFF);
|
|
|
|
outb(PIC1_COMMAND, 0x11);
|
|
|
|
outb(PIC2_COMMAND, 0x11);
|
|
|
|
// Potential spurious interrupts at 0x87
|
|
|
|
outb(PIC1_DATA, 0x80); // PIC1 offset (high and dry).
|
|
|
|
outb(PIC2_DATA, 0x80); // PIC2 offset (high and dry).
|
|
|
|
outb(PIC1_DATA, 0x4);
|
|
|
|
outb(PIC2_DATA, 0x2);
|
|
|
|
outb(PIC1_DATA, 0x1);
|
|
|
|
outb(PIC2_DATA, 0x1);
|
|
|
|
|
|
|
|
// Mask all.
|
|
|
|
outb(PIC1_DATA, 0xFF);
|
|
|
|
outb(PIC2_DATA, 0xFF);
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
void InspectApic() {
|
|
|
|
#if APIC_DEBUG
|
|
|
|
dbgln("APIC:");
|
|
|
|
dbgln("ID: %x", GetLocalReg(0x20));
|
|
|
|
dbgln("VER: %x", GetLocalReg(0x30));
|
|
|
|
dbgln("TPR: %x", GetLocalReg(0x80));
|
|
|
|
dbgln("APR: %x", GetLocalReg(0x90));
|
|
|
|
dbgln("PPR: %x", GetLocalReg(0xA0));
|
|
|
|
dbgln("LDR: %x", GetLocalReg(0xD0));
|
|
|
|
dbgln("DFR: %x", GetLocalReg(0xE0));
|
|
|
|
dbgln("SIV: %x", GetLocalReg(0xF0));
|
|
|
|
for (uint64_t i = 0; i < 8; i++) {
|
|
|
|
dbgln("ISR(%u): %x", i, GetLocalReg(0x100 + (0x10 * i)));
|
|
|
|
}
|
|
|
|
for (uint64_t i = 0; i < 8; i++) {
|
|
|
|
dbgln("TMR(%u): %x", i, GetLocalReg(0x180 + (0x10 * i)));
|
|
|
|
}
|
|
|
|
for (uint64_t i = 0; i < 8; i++) {
|
|
|
|
dbgln("IRR(%u): %x", i, GetLocalReg(0x200 + (0x10 * i)));
|
|
|
|
}
|
|
|
|
dbgln("ESR: %x", GetLocalReg(0x280));
|
|
|
|
|
|
|
|
dbgln("IO ID: %x", GetIoReg(0x0));
|
|
|
|
dbgln("IO VER: %x", GetIoReg(0x1));
|
|
|
|
dbgln("IO ARB: %x", GetIoReg(0x2));
|
|
|
|
for (uint8_t i = 0x10; i < 0x3F; i += 2) {
|
|
|
|
dbgln("IO (%u): %x", i, GetIoEntry(i));
|
|
|
|
}
|
|
|
|
dbgln("APIC MSR: %x", GetMSR(0x1B));
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2023-08-01 11:19:48 -07:00
|
|
|
// For now set these based on the presets in the following spec.
|
|
|
|
// FIXME: However in the future we should likely use the MADT for legacy
|
|
|
|
// interrupts and AML for PCI etc.
|
|
|
|
//
|
|
|
|
// http://web.archive.org/web/20161130153145/http://download.intel.com/design/chipsets/datashts/29056601.pdf
|
2023-06-07 13:40:36 -07:00
|
|
|
void EnableApic() {
|
|
|
|
MaskPic();
|
|
|
|
// Map Timer.
|
2023-08-01 20:18:47 -07:00
|
|
|
SetIoEntry(0x14, 0x10020);
|
2023-06-12 19:20:51 -07:00
|
|
|
|
|
|
|
// PCI Line 1-4
|
|
|
|
// FIXME: These should be level triggered according to spec I believe
|
|
|
|
// but because we handle the interrupt outside of the kernel it is tricky
|
|
|
|
// to wait to send the end of interrupt message.
|
|
|
|
// Because of this leave them as edge triggered and send EOI immediately.
|
|
|
|
SetIoEntry(0x30, 0x30);
|
|
|
|
SetIoEntry(0x32, 0x31);
|
|
|
|
SetIoEntry(0x34, 0x32);
|
|
|
|
SetIoEntry(0x36, 0x33);
|
|
|
|
|
2023-06-07 13:40:36 -07:00
|
|
|
InspectApic();
|
|
|
|
}
|
|
|
|
|
2023-08-01 20:18:47 -07:00
|
|
|
void SetLocalTimer(uint32_t init_cnt, uint64_t mode) {
|
|
|
|
WriteLocalReg(kTimerInitOffset, init_cnt);
|
|
|
|
WriteLocalReg(kLvtTimerOffset, mode | 0x21);
|
|
|
|
}
|
|
|
|
uint32_t GetLocalTimer() { return GetLocalReg(kTimerCurrOffset); }
|
|
|
|
|
|
|
|
void UnmaskPit() { SetIoEntry(0x14, GetIoEntry(0x14) & ~(0x10000)); }
|
|
|
|
void MaskPit() { SetIoEntry(0x14, GetIoEntry(0x14) | 0x10000); }
|
|
|
|
|
2023-06-12 19:20:51 -07:00
|
|
|
void SignalEOI() { WriteLocalReg(kEoiOffset, 0x0); }
|