| #include "libcflat.h" |
| #include "apic.h" |
| #include "vm.h" |
| #include "smp.h" |
| #include "desc.h" |
| #include "isr.h" |
| |
| #define EDGE_TRIGGERED 0 |
| #define LEVEL_TRIGGERED 1 |
| |
| static void set_ioapic_redir(unsigned line, unsigned vec, unsigned trig_mode) |
| { |
| ioapic_redir_entry_t e = { |
| .vector = vec, |
| .delivery_mode = 0, |
| .trig_mode = trig_mode, |
| }; |
| |
| ioapic_write_redir(line, e); |
| } |
| |
| static void set_irq_line(unsigned line, int val) |
| { |
| asm volatile("out %0, %1" : : "a"((u8)val), "d"((u16)(0x2000 + line))); |
| } |
| |
| static void toggle_irq_line(unsigned line) |
| { |
| set_irq_line(line, 1); |
| set_irq_line(line, 0); |
| } |
| |
| static void ioapic_reg_version(void) |
| { |
| u8 version_offset; |
| uint32_t data_read, data_write; |
| |
| version_offset = 0x01; |
| data_read = ioapic_read_reg(version_offset); |
| data_write = data_read ^ 0xffffffff; |
| |
| ioapic_write_reg(version_offset, data_write); |
| report("version register read only test", |
| data_read == ioapic_read_reg(version_offset)); |
| } |
| |
| static void ioapic_reg_id(void) |
| { |
| u8 id_offset; |
| uint32_t data_read, data_write, diff; |
| |
| id_offset = 0x0; |
| data_read = ioapic_read_reg(id_offset); |
| data_write = data_read ^ 0xffffffff; |
| |
| ioapic_write_reg(id_offset, data_write); |
| |
| diff = data_read ^ ioapic_read_reg(id_offset); |
| report("id register only bits [24:27] writable", |
| diff == 0x0f000000); |
| } |
| |
| static void ioapic_arbitration_id(void) |
| { |
| u8 id_offset, arb_offset; |
| uint32_t write; |
| |
| id_offset = 0x0; |
| arb_offset = 0x2; |
| write = 0x0f000000; |
| |
| ioapic_write_reg(id_offset, write); |
| report("arbitration register set by id", |
| ioapic_read_reg(arb_offset) == write); |
| |
| ioapic_write_reg(arb_offset, 0x0); |
| report("arbtration register read only", |
| ioapic_read_reg(arb_offset) == write); |
| } |
| |
| static volatile int g_isr_76; |
| |
| static void ioapic_isr_76(isr_regs_t *regs) |
| { |
| ++g_isr_76; |
| eoi(); |
| } |
| |
| static void test_ioapic_edge_intr(void) |
| { |
| handle_irq(0x76, ioapic_isr_76); |
| set_ioapic_redir(0x0e, 0x76, EDGE_TRIGGERED); |
| toggle_irq_line(0x0e); |
| asm volatile ("nop"); |
| report("edge triggered intr", g_isr_76 == 1); |
| } |
| |
| static volatile int g_isr_77; |
| |
| static void ioapic_isr_77(isr_regs_t *regs) |
| { |
| ++g_isr_77; |
| set_irq_line(0x0e, 0); |
| eoi(); |
| } |
| |
| static void test_ioapic_level_intr(void) |
| { |
| handle_irq(0x77, ioapic_isr_77); |
| set_ioapic_redir(0x0e, 0x77, LEVEL_TRIGGERED); |
| set_irq_line(0x0e, 1); |
| asm volatile ("nop"); |
| report("level triggered intr", g_isr_77 == 1); |
| } |
| |
| static int g_78, g_66, g_66_after_78; |
| static ulong g_66_rip, g_78_rip; |
| |
| static void ioapic_isr_78(isr_regs_t *regs) |
| { |
| ++g_78; |
| g_78_rip = regs->rip; |
| eoi(); |
| } |
| |
| static void ioapic_isr_66(isr_regs_t *regs) |
| { |
| ++g_66; |
| if (g_78) |
| ++g_66_after_78; |
| g_66_rip = regs->rip; |
| eoi(); |
| } |
| |
| static void test_ioapic_simultaneous(void) |
| { |
| handle_irq(0x78, ioapic_isr_78); |
| handle_irq(0x66, ioapic_isr_66); |
| set_ioapic_redir(0x0e, 0x78, EDGE_TRIGGERED); |
| set_ioapic_redir(0x0f, 0x66, EDGE_TRIGGERED); |
| irq_disable(); |
| toggle_irq_line(0x0f); |
| toggle_irq_line(0x0e); |
| irq_enable(); |
| asm volatile ("nop"); |
| report("ioapic simultaneous edge interrupts", |
| g_66 && g_78 && g_66_after_78 && g_66_rip == g_78_rip); |
| } |
| |
| static volatile int g_tmr_79 = -1; |
| |
| static void ioapic_isr_79(isr_regs_t *regs) |
| { |
| g_tmr_79 = apic_read_bit(APIC_TMR, 0x79); |
| set_irq_line(0x0e, 0); |
| eoi(); |
| } |
| |
| static void test_ioapic_edge_tmr(bool expected_tmr_before) |
| { |
| int tmr_before; |
| |
| handle_irq(0x79, ioapic_isr_79); |
| set_ioapic_redir(0x0e, 0x79, EDGE_TRIGGERED); |
| tmr_before = apic_read_bit(APIC_TMR, 0x79); |
| toggle_irq_line(0x0e); |
| asm volatile ("nop"); |
| report("TMR for ioapic edge interrupts (expected %s)", |
| tmr_before == expected_tmr_before && !g_tmr_79, |
| expected_tmr_before ? "true" : "false"); |
| } |
| |
| static void test_ioapic_level_tmr(bool expected_tmr_before) |
| { |
| int tmr_before; |
| |
| handle_irq(0x79, ioapic_isr_79); |
| set_ioapic_redir(0x0e, 0x79, LEVEL_TRIGGERED); |
| tmr_before = apic_read_bit(APIC_TMR, 0x79); |
| set_irq_line(0x0e, 1); |
| asm volatile ("nop"); |
| report("TMR for ioapic level interrupts (expected %s)", |
| tmr_before == expected_tmr_before && g_tmr_79, |
| expected_tmr_before ? "true" : "false"); |
| } |
| |
| #define IPI_DELAY 1000000 |
| |
| static void delay(int count) |
| { |
| while(count--) asm(""); |
| } |
| |
| static void toggle_irq_line_0x0e(void *data) |
| { |
| irq_disable(); |
| delay(IPI_DELAY); |
| toggle_irq_line(0x0e); |
| irq_enable(); |
| } |
| |
| static void test_ioapic_edge_tmr_smp(bool expected_tmr_before) |
| { |
| int tmr_before; |
| int i; |
| |
| g_tmr_79 = -1; |
| handle_irq(0x79, ioapic_isr_79); |
| set_ioapic_redir(0x0e, 0x79, EDGE_TRIGGERED); |
| tmr_before = apic_read_bit(APIC_TMR, 0x79); |
| on_cpu_async(1, toggle_irq_line_0x0e, 0); |
| i = 0; |
| while(g_tmr_79 == -1) i++; |
| printf("%d iterations before interrupt received\n", i); |
| report("TMR for ioapic edge interrupts (expected %s)", |
| tmr_before == expected_tmr_before && !g_tmr_79, |
| expected_tmr_before ? "true" : "false"); |
| } |
| |
| static void set_irq_line_0x0e(void *data) |
| { |
| irq_disable(); |
| delay(IPI_DELAY); |
| set_irq_line(0x0e, 1); |
| irq_enable(); |
| } |
| |
| static void test_ioapic_level_tmr_smp(bool expected_tmr_before) |
| { |
| int i, tmr_before; |
| |
| g_tmr_79 = -1; |
| handle_irq(0x79, ioapic_isr_79); |
| set_ioapic_redir(0x0e, 0x79, LEVEL_TRIGGERED); |
| tmr_before = apic_read_bit(APIC_TMR, 0x79); |
| on_cpu_async(1, set_irq_line_0x0e, 0); |
| i = 0; |
| while(g_tmr_79 == -1) i++; |
| printf("%d iterations before interrupt received\n", i); |
| report("TMR for ioapic level interrupts (expected %s)", |
| tmr_before == expected_tmr_before && g_tmr_79, |
| expected_tmr_before ? "true" : "false"); |
| } |
| |
| static int g_isr_98; |
| |
| static void ioapic_isr_98(isr_regs_t *regs) |
| { |
| ++g_isr_98; |
| if (g_isr_98 == 1) { |
| set_irq_line(0x0e, 0); |
| set_irq_line(0x0e, 1); |
| } |
| set_irq_line(0x0e, 0); |
| eoi(); |
| } |
| |
| static void test_ioapic_level_coalesce(void) |
| { |
| handle_irq(0x98, ioapic_isr_98); |
| set_ioapic_redir(0x0e, 0x98, LEVEL_TRIGGERED); |
| set_irq_line(0x0e, 1); |
| asm volatile ("nop"); |
| report("coalesce simultaneous level interrupts", g_isr_98 == 1); |
| } |
| |
| static int g_isr_99; |
| |
| static void ioapic_isr_99(isr_regs_t *regs) |
| { |
| ++g_isr_99; |
| set_irq_line(0x0e, 0); |
| eoi(); |
| } |
| |
| static void test_ioapic_level_sequential(void) |
| { |
| handle_irq(0x99, ioapic_isr_99); |
| set_ioapic_redir(0x0e, 0x99, LEVEL_TRIGGERED); |
| set_irq_line(0x0e, 1); |
| set_irq_line(0x0e, 1); |
| asm volatile ("nop"); |
| report("sequential level interrupts", g_isr_99 == 2); |
| } |
| |
| static volatile int g_isr_9a; |
| |
| static void ioapic_isr_9a(isr_regs_t *regs) |
| { |
| ++g_isr_9a; |
| if (g_isr_9a == 2) |
| set_irq_line(0x0e, 0); |
| eoi(); |
| } |
| |
| static void test_ioapic_level_retrigger(void) |
| { |
| int i; |
| |
| handle_irq(0x9a, ioapic_isr_9a); |
| set_ioapic_redir(0x0e, 0x9a, LEVEL_TRIGGERED); |
| |
| asm volatile ("cli"); |
| set_irq_line(0x0e, 1); |
| |
| for (i = 0; i < 10; i++) { |
| if (g_isr_9a == 2) |
| break; |
| |
| asm volatile ("sti; hlt; cli"); |
| } |
| |
| asm volatile ("sti"); |
| |
| report("retriggered level interrupts without masking", g_isr_9a == 2); |
| } |
| |
| static volatile int g_isr_81; |
| |
| static void ioapic_isr_81(isr_regs_t *regs) |
| { |
| ++g_isr_81; |
| set_irq_line(0x0e, 0); |
| eoi(); |
| } |
| |
| static void test_ioapic_edge_mask(void) |
| { |
| handle_irq(0x81, ioapic_isr_81); |
| set_ioapic_redir(0x0e, 0x81, EDGE_TRIGGERED); |
| |
| set_mask(0x0e, true); |
| set_irq_line(0x0e, 1); |
| set_irq_line(0x0e, 0); |
| |
| asm volatile ("nop"); |
| report("masked level interrupt", g_isr_81 == 0); |
| |
| set_mask(0x0e, false); |
| set_irq_line(0x0e, 1); |
| |
| asm volatile ("nop"); |
| report("unmasked level interrupt", g_isr_81 == 1); |
| } |
| |
| static volatile int g_isr_82; |
| |
| static void ioapic_isr_82(isr_regs_t *regs) |
| { |
| ++g_isr_82; |
| set_irq_line(0x0e, 0); |
| eoi(); |
| } |
| |
| static void test_ioapic_level_mask(void) |
| { |
| handle_irq(0x82, ioapic_isr_82); |
| set_ioapic_redir(0x0e, 0x82, LEVEL_TRIGGERED); |
| |
| set_mask(0x0e, true); |
| set_irq_line(0x0e, 1); |
| |
| asm volatile ("nop"); |
| report("masked level interrupt", g_isr_82 == 0); |
| |
| set_mask(0x0e, false); |
| |
| asm volatile ("nop"); |
| report("unmasked level interrupt", g_isr_82 == 1); |
| } |
| |
| static volatile int g_isr_83; |
| |
| static void ioapic_isr_83(isr_regs_t *regs) |
| { |
| ++g_isr_83; |
| set_mask(0x0e, true); |
| eoi(); |
| } |
| |
| static void test_ioapic_level_retrigger_mask(void) |
| { |
| handle_irq(0x83, ioapic_isr_83); |
| set_ioapic_redir(0x0e, 0x83, LEVEL_TRIGGERED); |
| |
| set_irq_line(0x0e, 1); |
| asm volatile ("nop"); |
| set_mask(0x0e, false); |
| asm volatile ("nop"); |
| report("retriggered level interrupts with mask", g_isr_83 == 2); |
| |
| set_irq_line(0x0e, 0); |
| set_mask(0x0e, false); |
| } |
| |
| |
| int main(void) |
| { |
| setup_vm(); |
| smp_init(); |
| |
| mask_pic_interrupts(); |
| |
| if (enable_x2apic()) |
| printf("x2apic enabled\n"); |
| else |
| printf("x2apic not detected\n"); |
| |
| irq_enable(); |
| |
| ioapic_reg_version(); |
| ioapic_reg_id(); |
| ioapic_arbitration_id(); |
| |
| test_ioapic_edge_intr(); |
| test_ioapic_level_intr(); |
| test_ioapic_simultaneous(); |
| |
| test_ioapic_level_coalesce(); |
| test_ioapic_level_sequential(); |
| test_ioapic_level_retrigger(); |
| |
| test_ioapic_edge_mask(); |
| test_ioapic_level_mask(); |
| test_ioapic_level_retrigger_mask(); |
| |
| test_ioapic_edge_tmr(false); |
| test_ioapic_level_tmr(false); |
| test_ioapic_level_tmr(true); |
| test_ioapic_edge_tmr(true); |
| |
| if (cpu_count() > 1) { |
| test_ioapic_edge_tmr_smp(false); |
| test_ioapic_level_tmr_smp(false); |
| test_ioapic_level_tmr_smp(true); |
| test_ioapic_edge_tmr_smp(true); |
| } |
| |
| return report_summary(); |
| } |