summaryrefslogtreecommitdiff
path: root/OvmfPkg/Library/PlatformInitLib/X64/Paging.nasm
blob: 895a80950e200c60bb66aebc7604988d389261e6 (plain)
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
;------------------------------------------------------------------------------
; @file
;
; Switch from 5-level paging mode to 4-level paging mode.
;
; This assumes everything (code, stack, page tables) is in 32-bit
; address space.  Which is true for PEI phase even in X64 builds
; because low memory is used for early firmware setup.
;
; This also assumes the standard ResetVector GDT is active.
;
; SPDX-License-Identifier: BSD-2-Clause-Patent
;------------------------------------------------------------------------------

SECTION .text
BITS    64

global ASM_PFX(Switch4Level)
ASM_PFX(Switch4Level):

    ; save regs
    push    rax
    push    rbx
    push    rcx
    push    rdx

    ; cs:ip for long mode
    lea     rax, [rel Switch4Level64]
    mov     rbx, 0x3800000000 ; LINEAR_CODE64_SEL << 32
    or      rax, rbx
    push    rax

    ; cs:ip for 32-bit mode
    lea     rax, [rel Switch4Level32]
    mov     rbx, 0x1000000000 ; LINEAR_CODE_SEL << 32
    or      rax, rbx
    push    rax

    ; enter 32-bit mode
    retf

Switch4Level64:
    ; restore regs
    pop     rdx
    pop     rcx
    pop     rbx
    pop     rax

    ret

BITS    32

Switch4Level32:
    ; disable paging
    mov     eax, cr0
    btc     eax, 31   ; clear PG
    mov     cr0, eax

    ; disable 5-level paging
    mov     eax, cr4
    btc     eax, 12   ; clear la57
    mov     cr4, eax

    ; fixup cr3 (dereference 5th level)
    mov     eax, cr3
    mov     eax, [ eax ]
    and     eax, 0xfffff000
    mov     cr3, eax

    ; enable paging
    mov     eax, cr0
    bts     eax, 31   ; set PG
    mov     cr0, eax

    ; back to long mode
    retf