2 * This file is part of the coreboot project.
4 * Copyright (C) 2009-2010 coresystems GmbH
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; version 2 of the License.
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License
16 * along with this program; if not, write to the Free Software
17 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA.
20 #define REALMODE_BASE 0x600
21 #define RELOCATED(x) (x - __realmode_code + REALMODE_BASE)
26 /* This is the intXX interrupt handler stub code. It gets copied
27 * to the IDT and to some fixed addresses in the F segment. Before
28 * the code can used, it gets patched up by the C function copying
29 * it: byte 3 (the $0 in movb $0, %al) is overwritten with the int#.
36 movb $0, %al /* This instruction gets modified */
37 ljmp $0, $__interrupt_handler_16bit
38 .globl __idt_handler_size
39 __idt_handler_size = ( . - __idt_handler)
42 /* In order to be independent of coreboot's position in RAM
43 * we relocate a part of the code to the low megabyte, so the
44 * CPU can use it in real-mode. This code lives at __realmode_code.
46 .globl __realmode_code
49 /* Realmode IDT pointer structure. */
51 __realmode_idt = RELOCATED(.)
52 .word 1023 /* 16 bit limit */
53 .long 0 /* 24 bit base */
56 /* Preserve old stack */
57 __stack = RELOCATED(.)
60 /* Register store for realmode_call and realmode_interrupt */
61 __registers = RELOCATED(.)
65 .long 0 /* 12 - EDX */
66 .long 0 /* 16 - ESI */
67 .long 0 /* 20 - EDI */
69 /* 256 byte buffer, used by int10 */
71 __buffer = RELOCATED(.)
75 .globl __realmode_call
76 __realmode_call = RELOCATED(.)
77 /* save all registers to the stack */
81 /* Move the protected mode stack pointer to a safe place */
85 /* This function is called with regparm=0 and we have to
86 * skip the 36 byte from pushf+pusha. Hence start at 40.
91 mov %ax, __lcall_instr + 1
92 andl $0xffff0000, %eax
94 mov %ax, __lcall_instr + 3
96 /* initial register values */
98 movl %eax, __registers + 0 /* eax */
100 movl %eax, __registers + 4 /* ebx */
102 movl %eax, __registers + 8 /* ecx */
104 movl %eax, __registers + 12 /* edx */
106 movl %eax, __registers + 16 /* esi */
108 movl %eax, __registers + 20 /* edi */
110 /* Activate the right segment descriptor real mode. */
111 ljmp $0x28, $RELOCATED(1f)
114 /* 16 bit code from here on... */
116 /* Load the segment registers w/ properly configured
117 * segment descriptors. They will retain these
118 * configurations (limits, writability, etc.) once
119 * protected mode is turned off.
128 /* Turn off protection */
133 /* Now really going into real mode */
134 ljmp $0, $RELOCATED(1f)
136 /* Setup a stack: Put the stack at the end of page zero.
137 * That way we can easily share it between real and
138 * protected, since the 16 bit ESP at segment 0 will
139 * work for any case. */
145 /* Load 16 bit IDT */
150 /* initialize registers for option rom lcall */
151 movl __registers + 0, %eax
152 movl __registers + 4, %ebx
153 movl __registers + 8, %ecx
154 movl __registers + 12, %edx
155 movl __registers + 16, %esi
156 movl __registers + 20, %edi
158 /* Set all segments to 0x0000, ds to 0x0040 */
168 /* ************************************ */
169 __lcall_instr = RELOCATED(.)
172 /* ************************************ */
174 /* If we got here, we are just about done.
175 * Need to get back to protected mode.
181 /* Now that we are in protected mode
182 * jump to a 32 bit code segment.
184 data32 ljmp $0x10, $RELOCATED(1f)
194 /* restore proper idt */
197 /* restore stack pointer, eflags and register values */
203 // TODO return AX from OPROM call
206 .globl __realmode_interrupt
207 __realmode_interrupt = RELOCATED(.)
208 /* save all registers to the stack */
212 /* save the stack pointer */
216 /* This function is called with regparm=0 and we have to
217 * skip the 36 byte from pushf+pusha. Hence start at 40.
220 /* prepare interrupt calling code */
222 movb %al, __intXX_instr + 1 /* intno */
224 /* initial register values */
226 movl %eax, __registers + 0 /* eax */
228 movl %eax, __registers + 4 /* ebx */
230 movl %eax, __registers + 8 /* ecx */
232 movl %eax, __registers + 12 /* edx */
234 movl %eax, __registers + 16 /* esi */
236 movl %eax, __registers + 20 /* edi */
238 /* This configures CS properly for real mode. */
239 ljmp $0x28, $RELOCATED(1f)
241 .code16 /* 16 bit code from here on... */
243 /* Load the segment registers w/ properly configured segment
244 * descriptors. They will retain these configurations (limits,
245 * writability, etc.) once protected mode is turned off.
254 /* Turn off protected mode */
259 /* Now really going into real mode */
260 data32 ljmp $0, $RELOCATED(1f)
263 /* put the stack at the end of page zero. That way we can easily
264 * share it between real mode and protected mode, because %esp and
265 * %ss:%sp point to the same memory.
273 /* Load 16-bit intXX IDT */
278 /* initialize registers for intXX call */
279 movl __registers + 0, %eax
280 movl __registers + 4, %ebx
281 movl __registers + 8, %ecx
282 movl __registers + 12, %edx
283 movl __registers + 16, %esi
284 movl __registers + 20, %edi
286 /* Set all segments to 0x0000 */
295 __intXX_instr = RELOCATED(.)
296 .byte 0xcd, 0x00 /* This becomes intXX */
298 /* Ok, the job is done, now go back to protected mode coreboot */
303 /* Now that we are in protected mode jump to a 32-bit code segment. */
304 data32 ljmp $0x10, $RELOCATED(1f)
314 /* restore coreboot's 32-bit IDT */
317 /* restore stack pointer, eflags and register values and exit */
323 /* This is the 16-bit interrupt entry point called by the IDT stub code.
325 * Before this code code is called, %eax is pushed to the stack, and the
326 * interrupt number is loaded into %al. On return this function cleans up
330 __interrupt_handler_16bit = RELOCATED(.)
336 /* Clear DF to not break ABI assumptions */
339 /* Clean up the interrupt number. We could have done this in the stub,
340 * but it would have cost 2 more bytes per stub entry.
343 pushl %eax /* ... and make it the first parameter */
345 /* Switch to protected mode */
350 /* ... and jump to a 32 bit code segment. */
351 data32 ljmp $0x10, $RELOCATED(1f)
363 /* Call the C interrupt handler */
364 movl $interrupt_handler, %eax
367 /* Now return to real mode ... */
368 ljmp $0x28, $RELOCATED(1f)
371 /* Load the segment registers with properly configured segment
372 * descriptors. They will retain these configurations (limits,
373 * writability, etc.) once protected mode is turned off.
382 /* Disable Protected Mode */
387 /* Now really going into real mode */
388 ljmp $0, $RELOCATED(1f)
390 /* Restore real-mode stack segment */
394 /* Restore 16 bit IDT */
399 /* Restore all registers, including those
400 * manipulated by the C handler
410 .globl __realmode_code_size
411 __realmode_code_size = (. - __realmode_code)