From mboxrd@z Thu Jan 1 00:00:00 1970 Authentication-Results: mx.groups.io; dkim=missing; spf=softfail (domain: citrix.com, ip: , mailfrom: anthony.perard@citrix.com) Received: from esa2.hc3370-68.iphmx.com (esa2.hc3370-68.iphmx.com []) by groups.io with SMTP; Thu, 04 Jul 2019 07:42:38 -0700 Authentication-Results: esa2.hc3370-68.iphmx.com; dkim=none (message not signed) header.i=none; spf=None smtp.pra=anthony.perard@citrix.com; spf=Pass smtp.mailfrom=anthony.perard@citrix.com; spf=None smtp.helo=postmaster@mail.citrix.com Received-SPF: None (esa2.hc3370-68.iphmx.com: no sender authenticity information available from domain of anthony.perard@citrix.com) identity=pra; client-ip=162.221.158.21; receiver=esa2.hc3370-68.iphmx.com; envelope-from="anthony.perard@citrix.com"; x-sender="anthony.perard@citrix.com"; x-conformance=sidf_compatible Received-SPF: Pass (esa2.hc3370-68.iphmx.com: domain of anthony.perard@citrix.com designates 162.221.158.21 as permitted sender) identity=mailfrom; client-ip=162.221.158.21; receiver=esa2.hc3370-68.iphmx.com; envelope-from="anthony.perard@citrix.com"; x-sender="anthony.perard@citrix.com"; x-conformance=sidf_compatible; x-record-type="v=spf1"; x-record-text="v=spf1 ip4:209.167.231.154 ip4:178.63.86.133 ip4:195.66.111.40/30 ip4:85.115.9.32/28 ip4:199.102.83.4 ip4:192.28.146.160 ip4:192.28.146.107 ip4:216.52.6.88 ip4:216.52.6.188 ip4:162.221.158.21 ip4:162.221.156.83 ~all" Received-SPF: None (esa2.hc3370-68.iphmx.com: no sender authenticity information available from domain of postmaster@mail.citrix.com) identity=helo; client-ip=162.221.158.21; receiver=esa2.hc3370-68.iphmx.com; envelope-from="anthony.perard@citrix.com"; x-sender="postmaster@mail.citrix.com"; x-conformance=sidf_compatible IronPort-SDR: RDyGZSutIL1JuRreuqnf42vksOTqydl8925ajxtB84x744hi9fsrnyHOMd8ho8G3vDgUxgmkxI Xft4NUEtDF8sXXmImyHxNn6DlcqUF9jQqE/M4zYTGtC6T4mk+rkmpCAMKTX5aYevEhF5vcEi7K OjBZqPwEughQwpnoEyskPKGPMNc4f/SJIbfJdbNeshw78i+os8IGcgpSi6QtVXXDLZbOdOhMg2 13zZKGSJ7l4IBWMfhcEOwh/BkdQ2h2Ztl5t4Dn+Fyy06dUZepepQ+z/zT2o28+rLA89lkU/y2X +08= X-SBRS: 2.7 X-MesageID: 2588784 X-Ironport-Server: esa2.hc3370-68.iphmx.com X-Remote-IP: 162.221.158.21 X-Policy: $RELAYED X-IronPort-AV: E=Sophos;i="5.63,451,1557201600"; d="scan'208";a="2588784" From: "Anthony PERARD" To: CC: , Ard Biesheuvel , Jordan Justen , Laszlo Ersek , Julien Grall , Anthony PERARD Subject: [PATCH v3 03/35] OvmfPkg: Introduce XenResetVector Date: Thu, 4 Jul 2019 15:42:01 +0100 Message-ID: <20190704144233.27968-4-anthony.perard@citrix.com> X-Mailer: git-send-email 2.22.0 In-Reply-To: <20190704144233.27968-1-anthony.perard@citrix.com> References: <20190704144233.27968-1-anthony.perard@citrix.com> MIME-Version: 1.0 Return-Path: anthony.perard@citrix.com Content-Transfer-Encoding: quoted-printable Content-Type: text/plain Introduce XenResetVector, a copy of OvmfPkg/ResetVector, with one changes: - SEC_DEFAULT_CR0: enable cache (bit 30 or CD set to 0) Xen copies the OVMF code to RAM, there is no need to disable cache. This new module will later be modified to add a new entry point, more detail in a following commit "OvmfPkg/XenResetVector: Add new entry point for Xen PVH" Value FILE_GUID of XenResetVector have not changed compare to ResetVector because it is a special value (gEfiFirmwareVolumeTopFileGuid). Ref: https://bugzilla.tianocore.org/show_bug.cgi?id=3D1689 Signed-off-by: Anthony PERARD Reviewed-by: Laszlo Ersek --- Notes: v3: - Added gEfiFirmwareVolumeTopFileGuid to the commit message. - rebased: SPDX OvmfPkg/OvmfXen.dsc | 2 +- OvmfPkg/OvmfXen.fdf | 2 +- OvmfPkg/XenResetVector/XenResetVector.inf | 38 +++++ .../XenResetVector/Ia16/Real16ToFlat32.asm | 134 ++++++++++++++++ OvmfPkg/XenResetVector/Ia32/PageTables64.asm | 149 ++++++++++++++++++ OvmfPkg/XenResetVector/XenResetVector.nasmb | 68 ++++++++ 6 files changed, 391 insertions(+), 2 deletions(-) create mode 100644 OvmfPkg/XenResetVector/XenResetVector.inf create mode 100644 OvmfPkg/XenResetVector/Ia16/Real16ToFlat32.asm create mode 100644 OvmfPkg/XenResetVector/Ia32/PageTables64.asm create mode 100644 OvmfPkg/XenResetVector/XenResetVector.nasmb diff --git a/OvmfPkg/OvmfXen.dsc b/OvmfPkg/OvmfXen.dsc index 3b56815dd7..cab54da3e8 100644 --- a/OvmfPkg/OvmfXen.dsc +++ b/OvmfPkg/OvmfXen.dsc @@ -495,7 +495,7 @@ [PcdsDynamicDefault] #=0D ##########################################################################= ######=0D [Components]=0D - OvmfPkg/ResetVector/ResetVector.inf=0D + OvmfPkg/XenResetVector/XenResetVector.inf=0D =0D #=0D # SEC Phase modules=0D diff --git a/OvmfPkg/OvmfXen.fdf b/OvmfPkg/OvmfXen.fdf index f59647fd14..6fc8479aae 100644 --- a/OvmfPkg/OvmfXen.fdf +++ b/OvmfPkg/OvmfXen.fdf @@ -118,7 +118,7 @@ [FV.SECFV] #=0D INF OvmfPkg/Sec/SecMain.inf=0D =0D -INF RuleOverride=3DRESET_VECTOR OvmfPkg/ResetVector/ResetVector.inf=0D +INF RuleOverride=3DRESET_VECTOR OvmfPkg/XenResetVector/XenResetVector.inf= =0D =0D ##########################################################################= ######=0D [FV.PEIFV]=0D diff --git a/OvmfPkg/XenResetVector/XenResetVector.inf b/OvmfPkg/XenResetVe= ctor/XenResetVector.inf new file mode 100644 index 0000000000..097fc9b5b4 --- /dev/null +++ b/OvmfPkg/XenResetVector/XenResetVector.inf @@ -0,0 +1,38 @@ +## @file=0D +# Reset Vector=0D +#=0D +# Copyright (c) 2006 - 2014, Intel Corporation. All rights reserved.
= =0D +# Copyright (c) 2019, Citrix Systems, Inc.=0D +#=0D +# SPDX-License-Identifier: BSD-2-Clause-Patent=0D +#=0D +##=0D +=0D +[Defines]=0D + INF_VERSION =3D 0x00010005=0D + BASE_NAME =3D XenResetVector=0D + FILE_GUID =3D 1BA0062E-C779-4582-8566-336AE8F78F09= =0D + MODULE_TYPE =3D SEC=0D + VERSION_STRING =3D 1.1=0D +=0D +#=0D +# The following information is for reference only and not required by the = build tools.=0D +#=0D +# VALID_ARCHITECTURES =3D IA32 X64=0D +#=0D +=0D +[Sources]=0D + XenResetVector.nasmb=0D +=0D +[Packages]=0D + OvmfPkg/OvmfPkg.dec=0D + MdePkg/MdePkg.dec=0D + UefiCpuPkg/UefiCpuPkg.dec=0D +=0D +[BuildOptions]=0D + *_*_IA32_NASMB_FLAGS =3D -I$(WORKSPACE)/UefiCpuPkg/ResetVector/Vtf0/=0D + *_*_X64_NASMB_FLAGS =3D -I$(WORKSPACE)/UefiCpuPkg/ResetVector/Vtf0/=0D +=0D +[Pcd]=0D + gUefiOvmfPkgTokenSpaceGuid.PcdOvmfSecPageTablesBase=0D + gUefiOvmfPkgTokenSpaceGuid.PcdOvmfSecPageTablesSize=0D diff --git a/OvmfPkg/XenResetVector/Ia16/Real16ToFlat32.asm b/OvmfPkg/XenRe= setVector/Ia16/Real16ToFlat32.asm new file mode 100644 index 0000000000..5c329bfaea --- /dev/null +++ b/OvmfPkg/XenResetVector/Ia16/Real16ToFlat32.asm @@ -0,0 +1,134 @@ +;-------------------------------------------------------------------------= -----=0D +; @file=0D +; Transition from 16 bit real mode into 32 bit flat protected mode=0D +;=0D +; Copyright (c) 2008 - 2010, Intel Corporation. All rights reserved.
=0D +; Copyright (c) 2019, Citrix Systems, Inc.=0D +; SPDX-License-Identifier: BSD-2-Clause-Patent=0D +;=0D +;-------------------------------------------------------------------------= -----=0D +=0D +%define SEC_DEFAULT_CR0 0x00000023=0D +%define SEC_DEFAULT_CR4 0x640=0D +=0D +BITS 16=0D +=0D +;=0D +; Modified: EAX, EBX=0D +;=0D +; @param[out] DS Selector allowing flat access to all addresses= =0D +; @param[out] ES Selector allowing flat access to all addresses= =0D +; @param[out] FS Selector allowing flat access to all addresses= =0D +; @param[out] GS Selector allowing flat access to all addresses= =0D +; @param[out] SS Selector allowing flat access to all addresses= =0D +;=0D +TransitionFromReal16To32BitFlat:=0D +=0D + debugShowPostCode POSTCODE_16BIT_MODE=0D +=0D + cli=0D +=0D + mov bx, 0xf000=0D + mov ds, bx=0D +=0D + mov bx, ADDR16_OF(gdtr)=0D +=0D +o32 lgdt [cs:bx]=0D +=0D + mov eax, SEC_DEFAULT_CR0=0D + mov cr0, eax=0D +=0D + jmp LINEAR_CODE_SEL:dword ADDR_OF(jumpTo32BitAndLandHere)=0D +BITS 32=0D +jumpTo32BitAndLandHere:=0D +=0D + mov eax, SEC_DEFAULT_CR4=0D + mov cr4, eax=0D +=0D + debugShowPostCode POSTCODE_32BIT_MODE=0D +=0D + mov ax, LINEAR_SEL=0D + mov ds, ax=0D + mov es, ax=0D + mov fs, ax=0D + mov gs, ax=0D + mov ss, ax=0D +=0D + OneTimeCallRet TransitionFromReal16To32BitFlat=0D +=0D +ALIGN 2=0D +=0D +gdtr:=0D + dw GDT_END - GDT_BASE - 1 ; GDT limit=0D + dd ADDR_OF(GDT_BASE)=0D +=0D +ALIGN 16=0D +=0D +;=0D +; Macros for GDT entries=0D +;=0D +=0D +%define PRESENT_FLAG(p) (p << 7)=0D +%define DPL(dpl) (dpl << 5)=0D +%define SYSTEM_FLAG(s) (s << 4)=0D +%define DESC_TYPE(t) (t)=0D +=0D +; Type: data, expand-up, writable, accessed=0D +%define DATA32_TYPE 3=0D +=0D +; Type: execute, readable, expand-up, accessed=0D +%define CODE32_TYPE 0xb=0D +=0D +; Type: execute, readable, expand-up, accessed=0D +%define CODE64_TYPE 0xb=0D +=0D +%define GRANULARITY_FLAG(g) (g << 7)=0D +%define DEFAULT_SIZE32(d) (d << 6)=0D +%define CODE64_FLAG(l) (l << 5)=0D +%define UPPER_LIMIT(l) (l)=0D +=0D +;=0D +; The Global Descriptor Table (GDT)=0D +;=0D +=0D +GDT_BASE:=0D +; null descriptor=0D +NULL_SEL equ $-GDT_BASE=0D + DW 0 ; limit 15:0=0D + DW 0 ; base 15:0=0D + DB 0 ; base 23:16=0D + DB 0 ; sys flag, dpl, type=0D + DB 0 ; limit 19:16, flags=0D + DB 0 ; base 31:24=0D +=0D +; linear data segment descriptor=0D +LINEAR_SEL equ $-GDT_BASE=0D + DW 0xffff ; limit 15:0=0D + DW 0 ; base 15:0=0D + DB 0 ; base 23:16=0D + DB PRESENT_FLAG(1)|DPL(0)|SYSTEM_FLAG(1)|DESC_TYPE(DATA32_TYPE)=0D + DB GRANULARITY_FLAG(1)|DEFAULT_SIZE32(1)|CODE64_FLAG(0)|UPPER_LIM= IT(0xf)=0D + DB 0 ; base 31:24=0D +=0D +; linear code segment descriptor=0D +LINEAR_CODE_SEL equ $-GDT_BASE=0D + DW 0xffff ; limit 15:0=0D + DW 0 ; base 15:0=0D + DB 0 ; base 23:16=0D + DB PRESENT_FLAG(1)|DPL(0)|SYSTEM_FLAG(1)|DESC_TYPE(CODE32_TYPE)=0D + DB GRANULARITY_FLAG(1)|DEFAULT_SIZE32(1)|CODE64_FLAG(0)|UPPER_LIM= IT(0xf)=0D + DB 0 ; base 31:24=0D +=0D +%ifdef ARCH_X64=0D +; linear code (64-bit) segment descriptor=0D +LINEAR_CODE64_SEL equ $-GDT_BASE=0D + DW 0xffff ; limit 15:0=0D + DW 0 ; base 15:0=0D + DB 0 ; base 23:16=0D + DB PRESENT_FLAG(1)|DPL(0)|SYSTEM_FLAG(1)|DESC_TYPE(CODE64_TYPE)=0D + DB GRANULARITY_FLAG(1)|DEFAULT_SIZE32(0)|CODE64_FLAG(1)|UPPER_LIM= IT(0xf)=0D + DB 0 ; base 31:24=0D +%endif=0D +=0D +GDT_END:=0D +=0D diff --git a/OvmfPkg/XenResetVector/Ia32/PageTables64.asm b/OvmfPkg/XenRese= tVector/Ia32/PageTables64.asm new file mode 100644 index 0000000000..9f1c0e2259 --- /dev/null +++ b/OvmfPkg/XenResetVector/Ia32/PageTables64.asm @@ -0,0 +1,149 @@ +;-------------------------------------------------------------------------= -----=0D +; @file=0D +; Sets the CR3 register for 64-bit paging=0D +;=0D +; Copyright (c) 2008 - 2013, Intel Corporation. All rights reserved.
=0D +; Copyright (c) 2019, Citrix Systems, Inc.=0D +; SPDX-License-Identifier: BSD-2-Clause-Patent=0D +;=0D +;-------------------------------------------------------------------------= -----=0D +=0D +BITS 32=0D +=0D +%define PAGE_PRESENT 0x01=0D +%define PAGE_READ_WRITE 0x02=0D +%define PAGE_USER_SUPERVISOR 0x04=0D +%define PAGE_WRITE_THROUGH 0x08=0D +%define PAGE_CACHE_DISABLE 0x010=0D +%define PAGE_ACCESSED 0x020=0D +%define PAGE_DIRTY 0x040=0D +%define PAGE_PAT 0x080=0D +%define PAGE_GLOBAL 0x0100=0D +%define PAGE_2M_MBO 0x080=0D +%define PAGE_2M_PAT 0x01000=0D +=0D +%define PAGE_2M_PDE_ATTR (PAGE_2M_MBO + \=0D + PAGE_ACCESSED + \=0D + PAGE_DIRTY + \=0D + PAGE_READ_WRITE + \=0D + PAGE_PRESENT)=0D +=0D +%define PAGE_PDP_ATTR (PAGE_ACCESSED + \=0D + PAGE_READ_WRITE + \=0D + PAGE_PRESENT)=0D +=0D +; Check if Secure Encrypted Virtualization (SEV) feature is enabled=0D +;=0D +; If SEV is enabled then EAX will be at least 32=0D +; If SEV is disabled then EAX will be zero.=0D +;=0D +CheckSevFeature:=0D + ; Check if we have a valid (0x8000_001F) CPUID leaf=0D + mov eax, 0x80000000=0D + cpuid=0D +=0D + ; This check should fail on Intel or Non SEV AMD CPUs. In future if=0D + ; Intel CPUs supports this CPUID leaf then we are guranteed to have ex= act=0D + ; same bit definition.=0D + cmp eax, 0x8000001f=0D + jl NoSev=0D +=0D + ; Check for memory encryption feature:=0D + ; CPUID Fn8000_001F[EAX] - Bit 1=0D + ;=0D + mov eax, 0x8000001f=0D + cpuid=0D + bt eax, 1=0D + jnc NoSev=0D +=0D + ; Check if memory encryption is enabled=0D + ; MSR_0xC0010131 - Bit 0 (SEV enabled)=0D + mov ecx, 0xc0010131=0D + rdmsr=0D + bt eax, 0=0D + jnc NoSev=0D +=0D + ; Get pte bit position to enable memory encryption=0D + ; CPUID Fn8000_001F[EBX] - Bits 5:0=0D + ;=0D + mov eax, ebx=0D + and eax, 0x3f=0D + jmp SevExit=0D +=0D +NoSev:=0D + xor eax, eax=0D +=0D +SevExit:=0D + OneTimeCallRet CheckSevFeature=0D +=0D +;=0D +; Modified: EAX, EBX, ECX, EDX=0D +;=0D +SetCr3ForPageTables64:=0D +=0D + OneTimeCall CheckSevFeature=0D + xor edx, edx=0D + test eax, eax=0D + jz SevNotActive=0D +=0D + ; If SEV is enabled, C-bit is always above 31=0D + sub eax, 32=0D + bts edx, eax=0D +=0D +SevNotActive:=0D +=0D + ;=0D + ; For OVMF, build some initial page tables at=0D + ; PcdOvmfSecPageTablesBase - (PcdOvmfSecPageTablesBase + 0x6000).=0D + ;=0D + ; This range should match with PcdOvmfSecPageTablesSize which is=0D + ; declared in the FDF files.=0D + ;=0D + ; At the end of PEI, the pages tables will be rebuilt into a=0D + ; more permanent location by DxeIpl.=0D + ;=0D +=0D + mov ecx, 6 * 0x1000 / 4=0D + xor eax, eax=0D +clearPageTablesMemoryLoop:=0D + mov dword[ecx * 4 + PT_ADDR (0) - 4], eax=0D + loop clearPageTablesMemoryLoop=0D +=0D + ;=0D + ; Top level Page Directory Pointers (1 * 512GB entry)=0D + ;=0D + mov dword[PT_ADDR (0)], PT_ADDR (0x1000) + PAGE_PDP_ATTR=0D + mov dword[PT_ADDR (4)], edx=0D +=0D + ;=0D + ; Next level Page Directory Pointers (4 * 1GB entries =3D> 4GB)=0D + ;=0D + mov dword[PT_ADDR (0x1000)], PT_ADDR (0x2000) + PAGE_PDP_ATTR=0D + mov dword[PT_ADDR (0x1004)], edx=0D + mov dword[PT_ADDR (0x1008)], PT_ADDR (0x3000) + PAGE_PDP_ATTR=0D + mov dword[PT_ADDR (0x100C)], edx=0D + mov dword[PT_ADDR (0x1010)], PT_ADDR (0x4000) + PAGE_PDP_ATTR=0D + mov dword[PT_ADDR (0x1014)], edx=0D + mov dword[PT_ADDR (0x1018)], PT_ADDR (0x5000) + PAGE_PDP_ATTR=0D + mov dword[PT_ADDR (0x101C)], edx=0D +=0D + ;=0D + ; Page Table Entries (2048 * 2MB entries =3D> 4GB)=0D + ;=0D + mov ecx, 0x800=0D +pageTableEntriesLoop:=0D + mov eax, ecx=0D + dec eax=0D + shl eax, 21=0D + add eax, PAGE_2M_PDE_ATTR=0D + mov [ecx * 8 + PT_ADDR (0x2000 - 8)], eax=0D + mov [(ecx * 8 + PT_ADDR (0x2000 - 8)) + 4], edx=0D + loop pageTableEntriesLoop=0D +=0D + ;=0D + ; Set CR3 now that the paging structures are available=0D + ;=0D + mov eax, PT_ADDR (0)=0D + mov cr3, eax=0D +=0D + OneTimeCallRet SetCr3ForPageTables64=0D diff --git a/OvmfPkg/XenResetVector/XenResetVector.nasmb b/OvmfPkg/XenReset= Vector/XenResetVector.nasmb new file mode 100644 index 0000000000..89a4b08bc3 --- /dev/null +++ b/OvmfPkg/XenResetVector/XenResetVector.nasmb @@ -0,0 +1,68 @@ +;-------------------------------------------------------------------------= -----=0D +; @file=0D +; This file includes all other code files to assemble the reset vector cod= e=0D +;=0D +; Copyright (c) 2008 - 2013, Intel Corporation. All rights reserved.
=0D +; Copyright (c) 2019, Citrix Systems, Inc.=0D +; SPDX-License-Identifier: BSD-2-Clause-Patent=0D +;=0D +;-------------------------------------------------------------------------= -----=0D +=0D +;=0D +; If neither ARCH_IA32 nor ARCH_X64 are defined, then try to include=0D +; Base.h to use the C pre-processor to determine the architecture.=0D +;=0D +%ifndef ARCH_IA32=0D + %ifndef ARCH_X64=0D + #include =0D + #if defined (MDE_CPU_IA32)=0D + %define ARCH_IA32=0D + #elif defined (MDE_CPU_X64)=0D + %define ARCH_X64=0D + #endif=0D + %endif=0D +%endif=0D +=0D +%ifdef ARCH_IA32=0D + %ifdef ARCH_X64=0D + %error "Only one of ARCH_IA32 or ARCH_X64 can be defined."=0D + %endif=0D +%elifdef ARCH_X64=0D +%else=0D + %error "Either ARCH_IA32 or ARCH_X64 must be defined."=0D +%endif=0D +=0D +%include "CommonMacros.inc"=0D +=0D +%include "PostCodes.inc"=0D +=0D +%ifdef DEBUG_PORT80=0D + %include "Port80Debug.asm"=0D +%elifdef DEBUG_SERIAL=0D + %include "SerialDebug.asm"=0D +%else=0D + %include "DebugDisabled.asm"=0D +%endif=0D +=0D +%include "Ia32/SearchForBfvBase.asm"=0D +%include "Ia32/SearchForSecEntry.asm"=0D +=0D +%ifdef ARCH_X64=0D + #include =0D +=0D + %if (FixedPcdGet32 (PcdOvmfSecPageTablesSize) !=3D 0x6000)=0D + %error "This implementation inherently depends on PcdOvmfSecPageTables= Size"=0D + %endif=0D +=0D + %define PT_ADDR(Offset) (FixedPcdGet32 (PcdOvmfSecPageTablesBase) + (Off= set))=0D +%include "Ia32/Flat32ToFlat64.asm"=0D +%include "Ia32/PageTables64.asm"=0D +%endif=0D +=0D +%include "Ia16/Real16ToFlat32.asm"=0D +%include "Ia16/Init16.asm"=0D +=0D +%include "Main.asm"=0D +=0D +%include "Ia16/ResetVectorVtf0.asm"=0D +=0D --=20 Anthony PERARD