summaryrefslogtreecommitdiff
path: root/sys/arch/i386/stand/libsa/gidt.S
diff options
context:
space:
mode:
authorJoel Sing <jsing@cvs.openbsd.org>2012-10-09 11:39:58 +0000
committerJoel Sing <jsing@cvs.openbsd.org>2012-10-09 11:39:58 +0000
commit6d8d4e7980d70c2d122e261bbadd500b6ab2e8b6 (patch)
treef2dcdd575f89754a92773cc0111e10fefedcce1c /sys/arch/i386/stand/libsa/gidt.S
parent4ca1805b95143c219be93f45e8194ff7adcfea8f (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.S82
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 ; \