forked from VictorYXL/MyOS
-
Notifications
You must be signed in to change notification settings - Fork 0
/
entry.nsm
186 lines (152 loc) · 2.93 KB
/
entry.nsm
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
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
; haribote-os boot asm
; TAB=4
VBEMODE EQU 0x105 ;画面模式
BOTPAK EQU 0x00280000 ; main要放置的位置
DSKCAC EQU 0x00100000 ; 启动区要放置的位置
DSKCAC0 EQU 0x00008000 ; 磁盘剩余部分要放置的位置
;获取引导信息
;图像信息的位置 0x0ff0 ~0x0ff8
CYLS EQU 0x0ff0 ;定位区(在IPL.nsm中将装在区域的结束位置写入了0x0ff8中)
LEDS EQU 0x0ff1 ;LED灯
VMODE EQU 0x0ff2 ;色位数
SCRNX EQU 0x0ff4 ;分辨率(x,y)
SCRNY EQU 0x0ff6
VRAM EQU 0x0ff8 ;图像缓冲区(保存屏幕内容的地方起始始的位置
ORG 0xc200
;确认VBE存在
MOV AX,0x9000
MOV ES,AX
MOV DI,0
MOV AX,0x4f00
INT 0x10
CMP AX,0x004f
JNE scrn320
;查看VBE版本
MOV AX,[ES:DI+4]
CMP AX,0x0200
JB scrn320
;获取画面信息
MOV CX,VBEMODE
MOV AX,0x4f01
INT 0x10
CMP AX,0x004f
JNE scrn320
;确认画面信息
CMP BYTE [ES:DI+0x19],8
JNE scrn320
CMP BYTE [ES: DI+0x1b],4
JNE scrn320
MOV AX,[ES:DI+0x00]
AND AX,0x0080
JZ scrn320
;切换画面模式
MOV BX,VBEMODE+0x4000
MOV AX,0x4f02
INT 0x10
MOV BYTE [VMODE],8
MOV AX,[ES:DI+0x12]
MOV [SCRNX],AX
MOV AX,[ES:DI+0x14]
MOV [SCRNY],AX
MOV EAX,[ES:DI+0x28]
MOV [VRAM],EAX
JMP keystatus
;切换失败,回到320*200模式
scrn320:
MOV AL,0x13
MOV AH,0x00
INT 0x10
MOV BYTE [VMODE],8
MOV WORD [SCRNX],320
MOV WORD [SCRNY],200
MOV DWORD [VRAM],0x000a0000
keystatus:
;获取LED灯的信息
MOV AH,0x02
INT 0x16
MOV [LEDS],AL
MOV AL,0xff ;禁止主PIC的全部中断
OUT 0x21,AL
NOP ;连续OUT可能导致无法运行
OUT 0xa1,AL ;禁止从PIC的全部中断
CLI ;禁止CPU级别的中断
; 设定A20GATE,为了使CPU能访问1MB以上内存
CALL waitkbdout
MOV AL,0xd1
OUT 0x64,AL
CALL waitkbdout
MOV AL,0xdf ;向附属端口输出0xdf
OUT 0x60,AL
CALL waitkbdout
; 切换到保护模式
;[INSTRSET "i486p"]
LGDT [GDTR0] ; 设定临时GDT
MOV EAX,CR0 ;CR0寄存器,特殊的寄存器
AND EAX,0x7fffffff ; 最高位置0
OR EAX,0x00000001 ; 最低位置1
MOV CR0,EAX
JMP pipelineflush
pipelineflush:
MOV AX,1*8 ; 所有寄存器都变成0x0008
MOV DS,AX
MOV ES,AX
MOV FS,AX
MOV GS,AX
MOV SS,AX
; 传送main.hrb
MOV ESI,main ;源
MOV EDI,BOTPAK ;目的
MOV ECX,512*1024/4 ;传送数据大小(单位双字)
CALL memcpy ;传送函数(注意是CALL而非JMP)
; 传送启动区
MOV ESI,0x7c00 ;源
MOV EDI,DSKCAC ;目的
MOV ECX,512/4
CALL memcpy
;传送剩余磁盘内容(10个柱面*两个磁头*18个扇区)
MOV ESI,DSKCAC0+512 ;源
MOV EDI,DSKCAC+512 ;目的
MOV ECX,0
MOV CL,BYTE [CYLS]
IMUL ECX,512*18*2/4
SUB ECX,512/4 ;cyls*512*18*2/4-512/4(除去引导区)
CALL memcpy
; main的启动(这段没看懂)
MOV EBX,BOTPAK
MOV ECX,[EBX+16]
ADD ECX,3 ; ECX += 3;
SHR ECX,2 ; ECX /= 4;
JZ skip ;Jmup if zero
MOV ESI,[EBX+20] ; 源
ADD ESI,EBX
MOV EDI,[EBX+12] ; 目的
CALL memcpy
skip:
MOV ESP,[EBX+12]
JMP DWORD 2*8:0x0000001b
waitkbdout:
IN AL,0x64
AND AL,0x02
JNZ waitkbdout
RET
;将内存进行赋值,源地址为ESI,目的地址为EDI,数据长度为ECX
memcpy:
MOV EAX,[ESI]
ADD ESI,4
MOV [EDI],EAX
ADD EDI,4
SUB ECX,1
JNZ memcpy
RET
;填入地址使GTD能按16位对齐,看不太懂
ALIGNB 16
GDT0:
RESB 8
DW 0xffff,0x0000,0x9200,0x00cf
DW 0xffff,0x0000,0x9a28,0x0047
DW 0
GDTR0:
DW 8*3-1
DD GDT0
ALIGNB 16
main: