1

为了了解操作系统的工作原理,我制作了一个简单的引导加载程序,它加载了一个用于测试保护模式等的小型测试应用程序。引导扇区在 0x7c00 加载后,引导加载程序在段 0x2000 加载测试代码并启动第一条指令。但是当我尝试进入保护模式时,系统会重新启动。谁能帮我解决这个问题?

这是我在 0x2000 部分的代码

    BITS 16

; Entering_ProtectedMode:   

    cli                                             
    mov ax, 2000h
    mov ss, ax                                      
    mov sp, 0FFFFh
    sti                                             
    cld                                             

    mov ax, 2000h                                   
    mov ds, ax                                      
    mov es, ax                                      
    mov fs, ax                                      
    mov gs, ax

    ;xor ax, ax
    ;mov ds, ax              ; update data segment

    cli                     ; clear interrupts

    lgdt [gdtr]             ; load GDT from GDTR (see gdt_32.inc)

    call OpenA20Gate        ; open the A20 gate 
    call EnablePMode        ; jumps to ProtectedMode

OpenA20Gate:
    in al, 0x93         ; switch A20 gate via fast A20 port 92
    or al, 2            ; set A20 Gate bit 1
    and al, ~1          ; clear INIT_NOW bit
    out 0x92, al
    ret

EnablePMode:
    mov eax, cr0
    or eax, 1
    mov cr0, eax
    jmp CODE_SEG : ProtectedMode

;*********************************
;* Global Descriptor Table (GDT) *
;*********************************
NULL_DESC:
    dd 0            ; null descriptor
    dd 0

CODE_DESC:
    dw 0xFFFF       ; limit low
    dw 0            ; base low
    db 0            ; base middle
    db 10011010b    ; access
    db 11001111b    ; granularity
    db 0            ; base high

DATA_DESC:
    dw 0xFFFF       ; limit low
    dw 0            ; base low
    db 0            ; base middle
    db 10010010b    ; access
    db 11001111b    ; granularity
    db 0            ; base high

gdtr:
    dw gdtr - NULL_DESC - 1 ; length of GDT
    dd NULL_DESC   ; base of GDT

CODE_SEG equ CODE_DESC - NULL_DESC
DATA_SEG equ DATA_DESC - NULL_DESC  

;******************
;* Protected Mode *
;******************
    BITS 32

ProtectedMode:

    .halt:
        jmp .halt

    ;mov     ax, 10h
    ;mov     ds, ax ; update data segment
4

1 回答 1

1

试试这个:

OpenA20Gate:
in al, 0x92; instead of 0x93; switch A20 gate via fast A20 port 92
or al, 2            ; set A20 Gate bit 1
and al, ~1          ; clear INIT_NOW bit
out 0x92, al
ret

EnablePMode:
mov eax, cr0
or eax, 1
mov cr0, eax

……

;CODE_SEG 必须等于 8,如 "8:ProtectedMode"。GDT 的第一个描述符为 8 分。16 位:15-3 - GDT 中的索引(0 表示零描述符,1 表示第一个描述符 - 代码,2 表示第二个描述符 - 数据),2 - 表指示符(0 表示 GDT,1 表示 LDT ), 1-0 - RPL(请求特权级别)。

jmp CODE_SEG:ProtectedMode
于 2015-05-11T05:16:19.263 回答