VirtualBox

source: vbox/trunk/src/VBox/Devices/PC/Etherboot-src/filo/i386/switch.S@ 1372

Last change on this file since 1372 was 1, checked in by vboxsync, 55 years ago

import

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 2.5 KB
Line 
1 .globl entry, __switch_context, __exit_context, halt
2
3 .text
4 .align 4
5
6/*
7 * Entry point
8 * We start execution from here.
9 * It is assumed that CPU is in 32-bit protected mode and
10 * all segments are 4GB and base zero (flat model).
11 */
12entry:
13 /* Save boot context and switch to our main context.
14 * Main context is statically defined in C.
15 */
16 pushl %cs
17 call __switch_context
18
19 /* We get here when the main context switches back to
20 * the boot context.
21 * Return to previous bootloader.
22 */
23 ret
24
25/*
26 * Switch execution context
27 * This saves registers, segments, and GDT in the stack, then
28 * switches the stack, and restores everything from the new stack.
29 * This function takes no argument. New stack pointer is
30 * taken from global variable __context, and old stack pointer
31 * is also saved to __context. This way we can just jump to
32 * this routine to get back to the original context.
33 *
34 * Call this routine with lcall or pushl %cs; call.
35 */
36__switch_context:
37 /* Save everything in current stack */
38 pushfl /* 56 */
39 pushl %ds /* 52 */
40 pushl %es /* 48 */
41 pushl %fs /* 44 */
42 pushl %gs /* 40 */
43 pushal /* 8 */
44 subl $8, %esp
45 movw %ss, (%esp) /* 0 */
46 sgdt 2(%esp) /* 2 */
47
48#if 0
49 /* Swap %cs and %eip on the stack, so lret will work */
50 movl 60(%esp), %eax
51 xchgl %eax, 64(%esp)
52 movl %eax, 60(%esp)
53#endif
54
55 /* At this point we don't know if we are on flat segment
56 * or relocated. So compute the address offset from %eip.
57 * Assuming CS.base==DS.base==SS.base.
58 */
59 call 1f
601: popl %ebx
61 subl $1b, %ebx
62
63 /* Interrupts are not allowed... */
64 cli
65
66 /* Current context pointer is our stack pointer */
67 movl %esp, %esi
68
69 /* Normalize the ctx pointer */
70 subl %ebx, %esi
71
72 /* Swap it with new value */
73 xchgl %esi, __context(%ebx)
74
75 /* Adjust new ctx pointer for current address offset */
76 addl %ebx, %esi
77
78 /* Load new %ss and %esp to temporary */
79 movzwl (%esi), %edx
80 movl 20(%esi), %eax
81
82 /* Load new GDT */
83 lgdt 2(%esi)
84
85 /* Load new stack segment with new GDT */
86 movl %edx, %ss
87
88 /* Set new stack pointer, but we have to adjust it because
89 * pushal saves %esp value before pushal, and we want the value
90 * after pushal.
91 */
92 leal -32(%eax), %esp
93
94 /* Load the rest from new stack */
95 popal
96 popl %gs
97 popl %fs
98 popl %es
99 popl %ds
100 popfl
101
102 /* Finally, load new %cs and %eip */
103 lret
104
105__exit_context:
106 /* Get back to the original context */
107 pushl %cs
108 call __switch_context
109
110 /* We get here if the other context attempt to switch to this
111 * dead context. This should not happen. */
112
113halt:
114 cli
115 hlt
116 jmp halt
Note: See TracBrowser for help on using the repository browser.

© 2025 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette