diff options
author | Joel Sing <jsing@cvs.openbsd.org> | 2012-10-09 11:39:58 +0000 |
---|---|---|
committer | Joel Sing <jsing@cvs.openbsd.org> | 2012-10-09 11:39:58 +0000 |
commit | 6d8d4e7980d70c2d122e261bbadd500b6ab2e8b6 (patch) | |
tree | f2dcdd575f89754a92773cc0111e10fefedcce1c /sys/arch/i386/stand/libsa/gidt.S | |
parent | 4ca1805b95143c219be93f45e8194ff7adcfea8f (diff) |
Dynamically build the Interrupt Descriptor Table (IDT), instead of using
a static table. This allows the IDT to be placed in .bss and avoids the
"relocation truncated" errors that result from the linker trying to put
a 32-bit address into a 16-bit hole.
With input from Marco Peereboom.
ok weingart@
Diffstat (limited to 'sys/arch/i386/stand/libsa/gidt.S')
-rw-r--r-- | sys/arch/i386/stand/libsa/gidt.S | 82 |
1 files changed, 50 insertions, 32 deletions
diff --git a/sys/arch/i386/stand/libsa/gidt.S b/sys/arch/i386/stand/libsa/gidt.S index 6ea7e8323d4..0dd8922618d 100644 --- a/sys/arch/i386/stand/libsa/gidt.S +++ b/sys/arch/i386/stand/libsa/gidt.S @@ -1,4 +1,4 @@ -/* $OpenBSD: gidt.S,v 1.33 2012/10/08 12:46:37 jsing Exp $ */ +/* $OpenBSD: gidt.S,v 1.34 2012/10/09 11:39:57 jsing Exp $ */ /* * Copyright (c) 1997 Michael Shalayeff @@ -148,7 +148,7 @@ ENTRY(_rtt) #endif /* Try to cause a triple fault... */ - lidt Idtr_reset + lidt Idtr_reset xorl %eax, %eax divl %eax, %eax @@ -159,26 +159,33 @@ ENTRY(_rtt) movl $0, %esp /* segment violation */ ret - .align 8, 0x90 -pmm_init: - /* load idtr for interrupts */ - lidt Idtr +#define IPROC(n) X##n +#define IEMU(n) IPROC(emu##n) + +create_idt_entry: + movw %ax, (%ebx) + movw $S32TEXT, 2(%ebx) + movw $((0x80|SDT_SYS386TGT) << 8), 4(%ebx) + shr $16, %eax + movw %ax, 6(%ebx) + addl $8, %ebx ret + .align 8, 0x90 +pmm_init: -#ifdef __STDC__ -#define IPROC(n) X##n -#define IEMU(n) IPROC(emu##n) -#else -#define IPROC(n) X/**/n -#define IEMU(n) IPROC(emu/**/n) -#endif - .align 8, 0x90 -idt: #define idte(e) \ - .short IPROC(e); .short (S32TEXT); \ - .short ((0x80|SDT_SYS386TGT) << 8); .short (LINKADDR >> 16) -/* internal (0-31) */ + movl $IPROC(e), %eax; call create_idt_entry +#define idtb(b) idte(emu##b) + + /* Build interrupt descriptor table. */ + /* Maskable interrupts (32-255) */ + movl $idt, %ebx + movl $Idtr, %eax + movw $(640 - 1), (%eax) + movl %ebx, 2(%eax) + + /* Internal (0-31) */ idte(de); idte(db); idte(nmi); idte(bp); idte(of); idte(br) idte(ud); idte(nm); idte(df); idte(fo); idte(ts); idte(np) idte(ss); idte(gp); idte(pf); idte(xx); idte(mf); idte(ac) @@ -186,14 +193,7 @@ idt: idte(xx); idte(xx); idte(xx); idte(xx); idte(xx); idte(xx) idte(xx); idte(xx); idte(xx); idte(xx); idte(xx); idte(xx) idte(xx) - /* Maskable interrupts (32-255) */ - /* BIOS entry points (32-63) */ - /* DOS entry points (64-80) */ -#ifdef __STDC__ -#define idtb(b) idte(emu##b) -#else -#define idtb(b) idte(emu/**/b) -#endif + /* BIOS entry points (32-63) */ idtb(0); idtb(1); idtb(2); idtb(3); idtb(4); idtb(5) idtb(6); idtb(7); idtb(8); idtb(9); idtb(10); idtb(11) idtb(12); idtb(13); idtb(14); idtb(15); idtb(16); idtb(17) @@ -202,20 +202,37 @@ idt: idtb(30); idtb(31); idtb(32); idtb(33); idtb(34); idtb(35) idtb(36); idtb(37); idtb(38); idtb(39); idtb(40); idtb(41) idtb(42); idtb(43); idtb(44); idtb(45); idtb(46); idtb(47) + /* DOS entry points (64-80) */ + +#undef idtb #undef idte + + /* load idtr for interrupts */ + lidt Idtr + ret + + .bss + .align 8, 0x90 +idt: + /* IDT has 80 entries at 8 bytes each. */ + .space 640 + .globl Idtr -Idtr: .word . - idt - 1 - .long idt +Idtr: .word 0 // 640 - 1 + .long 0 // idt .word 0 + .data .align 8 .globl Idtr_real -Idtr_real: .word 1023 - .long 0 - .word 0 +Idtr_real: + .word 1023 + .long 0 + .word 0 .align 8 -Idtr_reset: .long 0, 0 +Idtr_reset: + .long 0, 0 .align 8 gdt: @@ -255,6 +272,7 @@ Gdtr: .word . - gdt - 1 .long gdt .word 0 + .text #define IENTRY(name,type) \ IPROC(name): \ pushl $type ; \ |