summaryrefslogtreecommitdiffhomepage
path: root/src/loader/rem.asm
blob: 7a330484e971adf72e6274a9981edc8a1f651a06 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
; Real mode interrupt from protected mode
; Written by mintsuki, licensed under BSD 2-Clause "Simplified" License

section .realmode

global rem_int
rem_int:
	; Self-modifying code: int $int_no
	mov al, byte [esp+4]
	mov byte [.int_no], al

	; Save out_regs
	mov eax, dword [esp+8]
	mov dword [.out_regs], eax

	; Save in_regs
	mov eax, dword [esp+12]
	mov dword [.in_regs], eax

	; Save GDT in case BIOS overwrites it
	sgdt [.gdt]

	; Save IDT
	sidt [.idt]

	; Load BIOS IVT
	lidt [.rem_idt]

	; Save non-scratch GPRs
	push ebx
	push esi
	push edi
	push ebp

	; Jump to real mode
	jmp 0x08:.bits16
 .bits16:
	bits 16
	mov ax, 0x10
	mov ds, ax
	mov es, ax
	mov fs, ax
	mov gs, ax
	mov ss, ax
	mov eax, cr0
	and al, 0xfe
	mov cr0, eax
	jmp 0x00:.cszero
 .cszero:
	xor ax, ax
	mov ss, ax

	; Load in_regs
	mov dword [ss:.esp], esp
	mov esp, dword [ss:.in_regs]
	pop gs
	pop fs
	pop es
	pop ds
	popfd
	pop ebp
	pop edi
	pop esi
	pop edx
	pop ecx
	pop ebx
	pop eax
	mov esp, dword [ss:.esp]

	sti

	; Indirect interrupt call
	db 0xcd
 .int_no:
	db 0

	cli

	; Load out_regs
	mov dword [ss:.esp], esp
	mov esp, dword [ss:.out_regs]
	lea esp, [esp + 10*4]
	push eax
	push ebx
	push ecx
	push edx
	push esi
	push edi
	push ebp
	pushfd
	push ds
	push es
	push fs
	push gs
	mov esp, dword [ss:.esp]

	; Restore GDT
	o32 lgdt [ss:.gdt]

	; Restore IDT
	o32 lidt [ss:.idt]

	; Jump back to pmode
	mov eax, cr0
	or al, 1
	mov cr0, eax
	jmp 0x18:.bits32
 .bits32:
	bits 32
	mov ax, 0x20
	mov ds, ax
	mov es, ax
	mov fs, ax
	mov gs, ax
	mov ss, ax

	; Restore non-scratch GPRs
	pop ebp
	pop edi
	pop esi
	pop ebx

	; Exit
	ret

align 16
.esp:
	dd 0
.out_regs:
	dd 0
.in_regs:
	dd 0
.gdt:
	dq 0
.idt:
	dq 0
.rem_idt:
	dw 0x3ff
	dd 0