最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

assembly - Returning from 32-bit protected mode to real mode without a far jump - Stack Overflow

programmeradmin1浏览0评论

I'm following the advice on to return from 32-bit protected mode to real mode. That code includes a far jump. My code looks like this now:

    ; At this point we are still in protected mode with CS pointing
    ; to a 16-bit descriptor.
    bits 16
    cli  ; Just to make sure.
    mov eax, cr0
    and eax, 0x7FFFFFFe ; Disable paging bit & disable 16-bit pmode.
    mov cr0, eax
f1: jmp 0:GoRMode  ; ?? Is this far jump needed ??
GoRMode:
    mov sp, 0x8000
    mov ax, 0       ; Reset segment registers to 0.
    mov ds, ax
    mov es, ax
    mov fs, ax
    mov gs, ax
    mov ss, ax
    pushf
    pop ax
    or ax, strict word 0x200  ; IF := 1 (sti) by the iret below.
    push ax
    push word my_segment
    push word my_offset
    iret  ; Pops IP first, then CS, then FLAGS.

My question is: Can I omit the far jump without breaking anything?

My idea is that the final iret will take care of setting CS.

The code works in QEMU (with or without KVM), VirtualBox, v86 and 86Box (Intel 430VX chipset, Pentium-S P54C 90 MHz CPU) with and without the far jump. I've also verified that iret doesn't pop SS and SP in this situation.

By using iret I'm avoiding the problems with an interrupt happening too early. Only the iret will enable interrupts, and by that time the system is already fully in real mode (i.e. all segment registers use real-mode descriptors).

FYI I know that I can replace the last few instructions with sti + jmp my_segment:my_offset, but that may break if an interrupt happens between the two.

发布评论

评论列表(0)

  1. 暂无评论