mirror of https://github.com/acidanthera/audk.git
OvmfPkg/VirtHstiDxe: add varstore flash check
Detects qemu config issue: vars pflash is not in secure mode (write access restricted to smm). Applies to Q35 with SMM only. Cc: Ard Biesheuvel <ardb+tianocore@kernel.org> Cc: Jiewen Yao <jiewen.yao@intel.com> Cc: Konstantin Kostiuk <kkostiuk@redhat.com> Signed-off-by: Gerd Hoffmann <kraxel@redhat.com> Reviewed-by: Jiewen Yao <Jiewen.yao@intel.com>
This commit is contained in:
parent
538b8944c1
commit
ddc43e7a41
|
@ -0,0 +1,90 @@
|
||||||
|
/** @file
|
||||||
|
|
||||||
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
#include <Library/BaseLib.h>
|
||||||
|
#include <Library/DebugLib.h>
|
||||||
|
|
||||||
|
#include "VirtHstiDxe.h"
|
||||||
|
|
||||||
|
#define WRITE_BYTE_CMD 0x10
|
||||||
|
#define BLOCK_ERASE_CMD 0x20
|
||||||
|
#define CLEAR_STATUS_CMD 0x50
|
||||||
|
#define READ_STATUS_CMD 0x70
|
||||||
|
#define READ_DEVID_CMD 0x90
|
||||||
|
#define BLOCK_ERASE_CONFIRM_CMD 0xd0
|
||||||
|
#define READ_ARRAY_CMD 0xff
|
||||||
|
#define CLEARED_ARRAY_STATUS 0x00
|
||||||
|
|
||||||
|
/* based on QemuFlashDetected (QemuFlashFvbServicesRuntimeDxe) */
|
||||||
|
UINT32
|
||||||
|
VirtHstiQemuFirmwareFlashCheck (
|
||||||
|
UINT32 Address
|
||||||
|
)
|
||||||
|
{
|
||||||
|
volatile UINT8 *Ptr;
|
||||||
|
|
||||||
|
UINTN Offset;
|
||||||
|
UINT8 OriginalUint8;
|
||||||
|
UINT8 ProbeUint8;
|
||||||
|
|
||||||
|
for (Offset = 0; Offset < EFI_PAGE_SIZE; Offset++) {
|
||||||
|
Ptr = (UINT8 *)(UINTN)(Address + Offset);
|
||||||
|
ProbeUint8 = *Ptr;
|
||||||
|
if ((ProbeUint8 != CLEAR_STATUS_CMD) &&
|
||||||
|
(ProbeUint8 != READ_STATUS_CMD) &&
|
||||||
|
(ProbeUint8 != CLEARED_ARRAY_STATUS))
|
||||||
|
{
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Offset >= EFI_PAGE_SIZE) {
|
||||||
|
DEBUG ((DEBUG_INFO, "%a: check failed\n", __func__));
|
||||||
|
return QEMU_FIRMWARE_FLASH_UNKNOWN;
|
||||||
|
}
|
||||||
|
|
||||||
|
OriginalUint8 = *Ptr;
|
||||||
|
*Ptr = CLEAR_STATUS_CMD;
|
||||||
|
ProbeUint8 = *Ptr;
|
||||||
|
if ((OriginalUint8 != CLEAR_STATUS_CMD) &&
|
||||||
|
(ProbeUint8 == CLEAR_STATUS_CMD))
|
||||||
|
{
|
||||||
|
*Ptr = OriginalUint8;
|
||||||
|
DEBUG ((DEBUG_INFO, "%a: %p behaves as RAM\n", __func__, Ptr));
|
||||||
|
return QEMU_FIRMWARE_FLASH_IS_RAM;
|
||||||
|
}
|
||||||
|
|
||||||
|
*Ptr = READ_STATUS_CMD;
|
||||||
|
ProbeUint8 = *Ptr;
|
||||||
|
if (ProbeUint8 == OriginalUint8) {
|
||||||
|
DEBUG ((DEBUG_INFO, "%a: %p behaves as ROM\n", __func__, Ptr));
|
||||||
|
return QEMU_FIRMWARE_FLASH_IS_ROM;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ProbeUint8 == READ_STATUS_CMD) {
|
||||||
|
*Ptr = OriginalUint8;
|
||||||
|
DEBUG ((DEBUG_INFO, "%a: %p behaves as RAM\n", __func__, Ptr));
|
||||||
|
return QEMU_FIRMWARE_FLASH_IS_RAM;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ProbeUint8 == CLEARED_ARRAY_STATUS) {
|
||||||
|
*Ptr = WRITE_BYTE_CMD;
|
||||||
|
*Ptr = OriginalUint8;
|
||||||
|
*Ptr = READ_STATUS_CMD;
|
||||||
|
ProbeUint8 = *Ptr;
|
||||||
|
*Ptr = READ_ARRAY_CMD;
|
||||||
|
if (ProbeUint8 & 0x10 /* programming error */) {
|
||||||
|
DEBUG ((DEBUG_INFO, "%a: %p behaves as FLASH, write-protected\n", __func__, Ptr));
|
||||||
|
return QEMU_FIRMWARE_FLASH_READ_ONLY;
|
||||||
|
} else {
|
||||||
|
DEBUG ((DEBUG_INFO, "%a: %p behaves as FLASH, writable\n", __func__, Ptr));
|
||||||
|
return QEMU_FIRMWARE_FLASH_WRITABLE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
DEBUG ((DEBUG_INFO, "%a: check failed\n", __func__));
|
||||||
|
return QEMU_FIRMWARE_FLASH_UNKNOWN;
|
||||||
|
}
|
|
@ -29,6 +29,7 @@ VirtHstiQemuQ35Init (
|
||||||
{
|
{
|
||||||
if (FeaturePcdGet (PcdSmmSmramRequire)) {
|
if (FeaturePcdGet (PcdSmmSmramRequire)) {
|
||||||
VirtHstiSetSupported (&mHstiQ35, 0, VIRT_HSTI_BYTE0_SMM_SMRAM_LOCK);
|
VirtHstiSetSupported (&mHstiQ35, 0, VIRT_HSTI_BYTE0_SMM_SMRAM_LOCK);
|
||||||
|
VirtHstiSetSupported (&mHstiQ35, 0, VIRT_HSTI_BYTE0_SMM_SECURE_VARS_FLASH);
|
||||||
}
|
}
|
||||||
|
|
||||||
return &mHstiQ35;
|
return &mHstiQ35;
|
||||||
|
@ -55,4 +56,16 @@ VirtHstiQemuQ35Verify (
|
||||||
|
|
||||||
VirtHstiTestResult (ErrorMsg, 0, VIRT_HSTI_BYTE0_SMM_SMRAM_LOCK);
|
VirtHstiTestResult (ErrorMsg, 0, VIRT_HSTI_BYTE0_SMM_SMRAM_LOCK);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (VirtHstiIsSupported (&mHstiQ35, 0, VIRT_HSTI_BYTE0_SMM_SECURE_VARS_FLASH)) {
|
||||||
|
CHAR16 *ErrorMsg = NULL;
|
||||||
|
|
||||||
|
switch (VirtHstiQemuFirmwareFlashCheck (PcdGet32 (PcdOvmfFlashNvStorageVariableBase))) {
|
||||||
|
case QEMU_FIRMWARE_FLASH_WRITABLE:
|
||||||
|
ErrorMsg = L"qemu vars pflash is not secure";
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
VirtHstiTestResult (ErrorMsg, 0, VIRT_HSTI_BYTE0_SMM_SECURE_VARS_FLASH);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,6 +7,7 @@ SPDX-License-Identifier: BSD-2-Clause-Patent
|
||||||
#define VIRT_HSTI_SECURITY_FEATURE_SIZE 2
|
#define VIRT_HSTI_SECURITY_FEATURE_SIZE 2
|
||||||
|
|
||||||
#define VIRT_HSTI_BYTE0_SMM_SMRAM_LOCK BIT0
|
#define VIRT_HSTI_BYTE0_SMM_SMRAM_LOCK BIT0
|
||||||
|
#define VIRT_HSTI_BYTE0_SMM_SECURE_VARS_FLASH BIT1
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
// ADAPTER_INFO_PLATFORM_SECURITY
|
// ADAPTER_INFO_PLATFORM_SECURITY
|
||||||
|
@ -65,3 +66,16 @@ VOID
|
||||||
VirtHstiQemuPCVerify (
|
VirtHstiQemuPCVerify (
|
||||||
VOID
|
VOID
|
||||||
);
|
);
|
||||||
|
|
||||||
|
/* Flash.c */
|
||||||
|
|
||||||
|
#define QEMU_FIRMWARE_FLASH_UNKNOWN 0
|
||||||
|
#define QEMU_FIRMWARE_FLASH_IS_ROM 1
|
||||||
|
#define QEMU_FIRMWARE_FLASH_IS_RAM 2
|
||||||
|
#define QEMU_FIRMWARE_FLASH_READ_ONLY 3
|
||||||
|
#define QEMU_FIRMWARE_FLASH_WRITABLE 4
|
||||||
|
|
||||||
|
UINT32
|
||||||
|
VirtHstiQemuFirmwareFlashCheck (
|
||||||
|
UINT32 Address
|
||||||
|
);
|
||||||
|
|
|
@ -22,6 +22,7 @@
|
||||||
VirtHstiDxe.c
|
VirtHstiDxe.c
|
||||||
QemuPC.c
|
QemuPC.c
|
||||||
QemuQ35.c
|
QemuQ35.c
|
||||||
|
Flash.c
|
||||||
|
|
||||||
[Packages]
|
[Packages]
|
||||||
MdePkg/MdePkg.dec
|
MdePkg/MdePkg.dec
|
||||||
|
@ -46,5 +47,8 @@
|
||||||
[FeaturePcd]
|
[FeaturePcd]
|
||||||
gUefiOvmfPkgTokenSpaceGuid.PcdSmmSmramRequire
|
gUefiOvmfPkgTokenSpaceGuid.PcdSmmSmramRequire
|
||||||
|
|
||||||
|
[Pcd]
|
||||||
|
gUefiOvmfPkgTokenSpaceGuid.PcdOvmfFlashNvStorageVariableBase
|
||||||
|
|
||||||
[Depex]
|
[Depex]
|
||||||
TRUE
|
TRUE
|
||||||
|
|
Loading…
Reference in New Issue