ucore lab 实验一练习3:

为何开启A20,以及如何开启A20?

答:关于A20,早期的8086CPU所提供的地址线只有20位,所以可寻址空间为0~2^20(1MB),但是8086的数据处理位宽16位,无法直接访问1M的地址

空间,所以8086提供了段地址加偏移地址的转换机制。PC的寻址结构是segment:offset,segment和offset都是16位寄存器,最大值是0ffffh,

所以换算成物理地址的计算方法是吧segment左移4位,再加上offset,所以segment:offset所能表示的最大为10ffefh,而这个地址超过了1M,

但是超过1M会发生“回卷”的现象不会报错,但是从下一代的80286开始,地址线成为了24位,所能访问的地址空间超过了1M,

此时寻址超过1M时会报错,出现了向下不兼容,所以为了解决这个问题采用了A20机制。

开启A20是为了从实模式进入保护模式。

将A20地址线控制和键盘控制器的一个输出进行AND操作,这样来控制A20地址线的打开与关闭。

具体为首先利用0x64输入一个写入的指令,然后由0x60读进去相应的参数来将A20置1

 # Enable A20:
    #  For backwards compatibility with the earliest PCs, physical
    #  address line 20 is tied low, so that addresses higher than
    #  1MB wrap around to zero by default. This code undoes this.
seta20.1:
    inb $0x64, %al                                  # Wait for not busy(8042 input buffer empty).
    testb $0x2, %al

    jnz seta20.1

# 0xd1表示写输出端口命令,参数随后通过0x60端口写入

    movb $0xd1, %al                                 # 0xd1 -> port 0x64

    outb %al, $0x64                                 # 0xd1 means: write data to 8042's P2 port

seta20.2:
    inb $0x64, %al                                  # Wait for not busy(8042 input buffer empty).
    testb $0x2, %al

    jnz seta20.2

# 通过0x60写入数据11011111 即将A20置1,打开A20

    movb $0xdf, %al                                 # 0xdf -> port 0x60

    outb %al, $0x60                                 # 0xdf = 11011111, means set P2's A20 bit(the 1 bit) to 1

如何初始化GDT表?

lgdt gdtdesc #把gdt表的起始位置和界限装入GDTR寄存器

如何使能和进入保护模式?

movl %cr0, %eax    #将cr0内容取出
orl $CR0_PE_ON, %eax #利用或运算将cro内容最后一位设置为1.CR0_PE_ON为标志位其值为ox1
movl %eax, %cr0 #将值放回cr0


    # Jump to next instruction, but in 32-bit code segment.
    # Switches processor into 32-bit mode.

    ljmp $PROT_MODE_CSEG, $protcseg #通过长跳转指令进入保护模式

 # Set up the protected-mode data segment registers复位各种寄存器
    movw $PROT_MODE_DSEG, %ax                       # Our data segment selector
    movw %ax, %ds                                   # -> DS: Data Segment
    movw %ax, %es                                   # -> ES: Extra Segment
    movw %ax, %fs                                   # -> FS
    movw %ax, %gs                                   # -> GS
    movw %ax, %ss                                   # -> SS: Stack Segment


    # Set up the stack pointer and call into C. The stack region is from 0--start(0x7c00)进入boot主方法
    movl $0x0, %ebp
    movl $start, %esp
    call bootmain



  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值