/* $Id: DevOVMF.cpp 43712 2012-10-23 14:02:24Z vboxsync $ */ /** @file * DevEFI - EFI <-> VirtualBox Integration Framework. */ /* * Copyright (C) 2012 Oracle Corporation * * This file is part of VirtualBox Open Source Edition (OSE), as * available from http://www.virtualbox.org. This file is free software; * you can redistribute it and/or modify it under the terms of the GNU * General Public License (GPL) as published by the Free Software * Foundation, in version 2 as it comes in the "COPYING" file of the * VirtualBox OSE distribution. VirtualBox OSE is distributed in the * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind. */ /******************************************************************************* * Header Files * *******************************************************************************/ #define LOG_GROUP LOG_GROUP_DEV_EFI #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #ifdef DEBUG # include # define DEVEFI_WITH_VBOXDBG_SCRIPT #endif #define VBOX_WITH_OVMF #include "Firmware2/VBoxPkg/Include/DevEFI.h" #include "VBoxDD.h" #include "VBoxDD2.h" #include "../PC/DevFwCommon.h" /* EFI includes */ #include #include #include #include #define EFI_SSM_VERSION 1 /******************************************************************************* * Structures and Typedefs * *******************************************************************************/ typedef struct { RTLISTNODE List; int idxVariable; RTUUID uuid; char szVariableName[EFI_VARIABLE_NAME_MAX]; uint32_t cbVariableName; uint8_t au8Value[EFI_VARIABLE_VALUE_MAX]; uint32_t cbValue; uint32_t u32Attribute; } EFIVAR, *PEFIVAR; typedef PEFIVAR *PPEFIVAR; typedef struct { EFIVAROP enmOp; uint32_t u32Status; uint32_t idxOpBuffer; EFIVAR OperationVarOp; int cNvramVariables; int iNvramLastIndex; EFIVAR NvramVariableList; int idxCurrentVar; PEFIVAR pCurrentVarOp; } NVRAMDESC; typedef struct DEVEFI { /** Pointer back to the device instance. */ PPDMDEVINS pDevIns; /** EFI message buffer. */ char szMsg[VBOX_EFI_DEBUG_BUFFER]; /** EFI message buffer index. */ uint32_t iMsg; /** EFI panic message buffer. */ char szPanicMsg[2048]; /** EFI panic message buffer index. */ uint32_t iPanicMsg; /** The system EFI ROM data. */ uint8_t *pu8EfiRom; /** The size of the system EFI ROM. */ uint64_t cbEfiRom; /** The name of the EFI ROM file. */ char *pszEfiRomFile; /** Thunk page pointer. */ uint8_t *pu8EfiThunk; /** First entry point of the EFI firmware */ RTGCPHYS GCEntryPoint0; /* Second Entry Point (PeiCore)*/ RTGCPHYS GCEntryPoint1; /** EFI firmware physical load address */ RTGCPHYS GCLoadAddress; /** Current info selector */ uint32_t iInfoSelector; /** Current info position */ int32_t iInfoPosition; /** Number of virtual CPUs. (Config) */ uint32_t cCpus; /** RAM below 4GB (in bytes). (Config) */ uint32_t cbBelow4GB; /** RAM above 4GB (in bytes). (Config) */ uint64_t cbAbove4GB; uint64_t cbRam; uint64_t cbRamHole; /** The size of the DMI tables */ uint16_t cbDmiTables; /** The DMI tables. */ uint8_t au8DMIPage[0x1000]; /** I/O-APIC enabled? */ uint8_t u8IOAPIC; /* Boot parameters passed to the firmware */ char szBootArgs[256]; /* Host UUID (for DMI) */ RTUUID aUuid; /* Device properties buffer */ uint8_t* pu8DeviceProps; /* Device properties buffer size */ uint32_t u32DevicePropsLen; /* Virtual machine front side bus frequency */ uint64_t u64FsbFrequency; /* Virtual machine time stamp counter frequency */ uint64_t u64TscFrequency; /* Virtual machine CPU frequency */ uint64_t u64CpuFrequency; /* GOP mode */ uint32_t u32GopMode; /* Uga mode resolutions */ uint32_t u32UgaHorisontal; uint32_t u32UgaVertical; NVRAMDESC NVRAM; struct { PPDMIBASE pDrvBase; PDMIBASE IBase; PPDMINVRAM pNvramDown; } Lun0; } DEVEFI; typedef DEVEFI *PDEVEFI; static SSMFIELD const g_aEfiNvramDescField[] = { SSMFIELD_ENTRY (NVRAMDESC, enmOp), SSMFIELD_ENTRY (NVRAMDESC, u32Status), SSMFIELD_ENTRY (NVRAMDESC, idxOpBuffer), SSMFIELD_ENTRY_IGNORE (NVRAMDESC, OperationVarOp), SSMFIELD_ENTRY (NVRAMDESC, cNvramVariables), SSMFIELD_ENTRY (NVRAMDESC, iNvramLastIndex), SSMFIELD_ENTRY_IGNORE (NVRAMDESC, NvramVariableList), SSMFIELD_ENTRY (NVRAMDESC, idxCurrentVar), SSMFIELD_ENTRY_IGNORE (NVRAMDESC, pCurrentVarOp), SSMFIELD_ENTRY_TERM() }; static SSMFIELD const g_aEfiVariableDescFields[] = { SSMFIELD_ENTRY_IGNORE (EFIVAR, List), SSMFIELD_ENTRY (EFIVAR, idxVariable), SSMFIELD_ENTRY (EFIVAR, uuid), SSMFIELD_ENTRY (EFIVAR, szVariableName), SSMFIELD_ENTRY (EFIVAR, cbVariableName), SSMFIELD_ENTRY (EFIVAR, au8Value), SSMFIELD_ENTRY (EFIVAR, cbValue), SSMFIELD_ENTRY (EFIVAR, u32Attribute), SSMFIELD_ENTRY_TERM() }; /** * Write to CMOS memory. * This is used by the init complete code. */ static void cmosWrite(PPDMDEVINS pDevIns, int off, uint32_t u32Val) { Assert(off < 128); Assert(u32Val < 256); int rc = PDMDevHlpCMOSWrite(pDevIns, off, u32Val); AssertRC(rc); } DECLINLINE(void) nvramFlushDeviceVariableList(PDEVEFI pThis) { PEFIVAR pEfiVar = NULL; while (!RTListIsEmpty(&pThis->NVRAM.NvramVariableList.List)) { pEfiVar = RTListNodeGetNext(&pThis->NVRAM.NvramVariableList.List, EFIVAR, List); RTListNodeRemove(&pEfiVar->List); RTMemFree(pEfiVar); } } /** * This function looks up variable in NVRAM list */ static int nvramLookupVariableByUuidAndName(PDEVEFI pThis, char *pszVariableName, PCRTUUID pUuid, PPEFIVAR ppEfiVar) { int rc = VERR_NOT_FOUND; PEFIVAR pEfiVar = NULL; LogFlowFunc(("pszVariableName:%s, pUuid:%RTuuid\n", pszVariableName, pUuid)); int idxVar = 0; RTListForEach((PRTLISTNODE)&pThis->NVRAM.NvramVariableList.List, pEfiVar, EFIVAR, List) { LogFlowFunc(("pEfiVar:%p\n", pEfiVar)); idxVar++; if ( pEfiVar && RTUuidCompare(pUuid, &pEfiVar->uuid) == 0 && RTStrCmp(pszVariableName, pEfiVar->szVariableName) == 0) { *ppEfiVar = pEfiVar; rc = VINF_SUCCESS; break; } } Assert(pThis->NVRAM.cNvramVariables >= idxVar); LogFlowFuncLeaveRC(rc); return rc; } static int nvramLoad(PDEVEFI pThis) { int rc = VINF_SUCCESS; PEFIVAR pEfiVar = NULL; int idxValue = 0; while(idxValue < 100) { pEfiVar = (PEFIVAR)RTMemAllocZ(sizeof(EFIVAR)); if (!pEfiVar) { LogRel(("EFI: Can't allocate space for stored EFI variable\n")); return VERR_NO_MEMORY; } pEfiVar->cbVariableName = EFI_VARIABLE_NAME_MAX; pEfiVar->cbValue = EFI_VARIABLE_VALUE_MAX; rc = pThis->Lun0.pNvramDown->pfnLoadNvramValue(pThis->Lun0.pNvramDown, idxValue, &pEfiVar->uuid, pEfiVar->szVariableName, (size_t *)&pEfiVar->cbVariableName, pEfiVar->au8Value, (size_t *)&pEfiVar->cbValue); idxValue++; if (RT_FAILURE(rc)) { RTMemFree(pEfiVar); break; } pThis->NVRAM.cNvramVariables++; RTListAppend((PRTLISTNODE)&pThis->NVRAM.NvramVariableList.List, &pEfiVar->List); } if ( RT_FAILURE(rc) && rc == VERR_NOT_FOUND) rc = VINF_SUCCESS; AssertRCReturn(rc, rc); return rc; } static int nvramStore(PDEVEFI pThis) { int rc = VINF_SUCCESS; PEFIVAR pEfiVar = NULL; int idxVar = 0; pThis->Lun0.pNvramDown->pfnFlushNvramStorage(pThis->Lun0.pNvramDown); RTListForEach((PRTLISTNODE)&pThis->NVRAM.NvramVariableList.List, pEfiVar, EFIVAR, List) { pThis->Lun0.pNvramDown->pfnStoreNvramValue(pThis->Lun0.pNvramDown, idxVar, &pEfiVar->uuid, pEfiVar->szVariableName, pEfiVar->cbVariableName, pEfiVar->au8Value, pEfiVar->cbValue); idxVar++; } Assert((pThis->NVRAM.cNvramVariables == idxVar)); return VINF_SUCCESS; } static uint32_t efiInfoSize(PDEVEFI pThis) { switch (pThis->iInfoSelector) { case EFI_INFO_INDEX_VOLUME_BASE: case EFI_INFO_INDEX_VOLUME_SIZE: case EFI_INFO_INDEX_TEMPMEM_BASE: case EFI_INFO_INDEX_TEMPMEM_SIZE: case EFI_INFO_INDEX_STACK_BASE: case EFI_INFO_INDEX_STACK_SIZE: case EFI_INFO_INDEX_GOP_MODE: case EFI_INFO_INDEX_UGA_VERTICAL_RESOLUTION: case EFI_INFO_INDEX_UGA_HORISONTAL_RESOLUTION: return 4; case EFI_INFO_INDEX_BOOT_ARGS: return (uint32_t)RTStrNLen(pThis->szBootArgs, sizeof pThis->szBootArgs) + 1; case EFI_INFO_INDEX_DEVICE_PROPS: return pThis->u32DevicePropsLen; case EFI_INFO_INDEX_FSB_FREQUENCY: case EFI_INFO_INDEX_CPU_FREQUENCY: case EFI_INFO_INDEX_TSC_FREQUENCY: return 8; } Assert(false); return 0; } static uint8_t efiInfoNextByte(PDEVEFI pThis) { union { uint32_t u32; uint64_t u64; } value; switch (pThis->iInfoSelector) { case EFI_INFO_INDEX_VOLUME_BASE: value.u32 = pThis->GCLoadAddress; break; case EFI_INFO_INDEX_VOLUME_SIZE: value.u32 = pThis->cbEfiRom; break; case EFI_INFO_INDEX_TEMPMEM_BASE: value.u32 = VBOX_EFI_TOP_OF_STACK; /* just after stack */ break; case EFI_INFO_INDEX_TEMPMEM_SIZE: value.u32 = 512 * 1024; /* 512 K */ break; case EFI_INFO_INDEX_STACK_BASE: /* Keep in sync with value in EfiThunk.asm */ value.u32 = VBOX_EFI_TOP_OF_STACK - 128*1024; /* 2M - 128 K */ break; case EFI_INFO_INDEX_STACK_SIZE: value.u32 = 128*1024; /* 128 K */ break; case EFI_INFO_INDEX_FSB_FREQUENCY: value.u64 = pThis->u64FsbFrequency; break; case EFI_INFO_INDEX_TSC_FREQUENCY: value.u64 = pThis->u64TscFrequency; break; case EFI_INFO_INDEX_CPU_FREQUENCY: value.u64 = pThis->u64CpuFrequency; break; case EFI_INFO_INDEX_BOOT_ARGS: return pThis->szBootArgs[pThis->iInfoPosition]; case EFI_INFO_INDEX_DEVICE_PROPS: return pThis->pu8DeviceProps[pThis->iInfoPosition]; case EFI_INFO_INDEX_GOP_MODE: value.u32 = pThis->u32GopMode; break; case EFI_INFO_INDEX_UGA_HORISONTAL_RESOLUTION: value.u32 = pThis->u32UgaHorisontal; break; case EFI_INFO_INDEX_UGA_VERTICAL_RESOLUTION: value.u32 = pThis->u32UgaVertical; break; default: Assert(false); value.u64 = 0; break; } return *((uint8_t*)&value+pThis->iInfoPosition); } static DECLCALLBACK(int) efiSaveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM) { int rc = VINF_SUCCESS; PEFIVAR pEfiVar = NULL; LogFlowFuncEnter(); PDEVEFI pThis = PDMINS_2_DATA(pDevIns, PDEVEFI); rc = SSMR3PutStructEx(pSSM, &pThis->NVRAM, sizeof(NVRAMDESC), 0, g_aEfiNvramDescField, NULL); AssertRCReturn(rc, rc); rc = SSMR3PutStructEx(pSSM, &pThis->NVRAM.OperationVarOp, sizeof(EFIVAR), 0, g_aEfiVariableDescFields, NULL); AssertRCReturn(rc, rc); int idxV = 0; RTListForEach(&pThis->NVRAM.NvramVariableList.List, pEfiVar, EFIVAR, List) { rc = SSMR3PutStructEx(pSSM, pEfiVar, sizeof(EFIVAR), 0, g_aEfiVariableDescFields, NULL); AssertRCReturn(rc, rc); idxV++; } Assert((pThis->NVRAM.cNvramVariables == idxV)); Log2(("idxV: %d\n", idxV)); LogFlowFuncLeaveRC(rc); return rc; } static DECLCALLBACK(int) efiLoadExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uVersion, uint32_t uPass) { int rc = VINF_SUCCESS; NOREF(uPass); LogFlowFunc(("ENTER: uVersion:%d, uPass:%d\n", uVersion, uPass)); PDEVEFI pThis = PDMINS_2_DATA(pDevIns, PDEVEFI); if (uPass != SSM_PASS_FINAL) return rc; /* we should clean up the loaded values */ nvramFlushDeviceVariableList(pThis); if (uVersion == 1) { rc = SSMR3GetStructEx(pSSM, &pThis->NVRAM, sizeof(NVRAMDESC), 0, g_aEfiNvramDescField, NULL); AssertRCReturn(rc, rc); rc = SSMR3GetStructEx(pSSM, &pThis->NVRAM.OperationVarOp, sizeof(EFIVAR), 0, g_aEfiVariableDescFields, NULL); AssertRCReturn(rc, rc); int idxVariable = 0; Assert(RTListIsEmpty(&pThis->NVRAM.NvramVariableList.List)); RTListInit(&pThis->NVRAM.NvramVariableList.List); for (idxVariable = 0; idxVariable < pThis->NVRAM.cNvramVariables; ++idxVariable) { PEFIVAR pEfiVar = (PEFIVAR)RTMemAllocZ(sizeof(EFIVAR)); AssertPtrReturn(pEfiVar, VERR_NO_MEMORY); rc = SSMR3GetStructEx(pSSM, pEfiVar, sizeof(EFIVAR), 0, g_aEfiVariableDescFields, NULL); AssertRCReturn(rc, rc); RTListInit(&pEfiVar->List); RTListAppend(&pThis->NVRAM.NvramVariableList.List, &pEfiVar->List); if (pThis->NVRAM.idxCurrentVar == pEfiVar->idxVariable) pThis->NVRAM.pCurrentVarOp = pEfiVar; } } LogFlowFuncLeaveRC(rc); return rc; } #if 0 static DECLCALLBACK(int) efiLoadDone(PPDMDEVINS pDevIns, PSSMHANDLE pSSM) { int rc = VINF_SUCCESS; PDEVEFI pThis = PDMINS_2_DATA(pDevIns, PDEVEFI); LogFlowFuncEnter(); if (RTListIsEmpty(&pThis->NVRAM.NvramVariableList.List)) nvramLoad(pThis); LogFlowFuncLeaveRC(rc); return rc; } #endif /** * Port I/O Handler for IN operations. * * @returns VBox status code. * * @param pDevIns The device instance. * @param pvUser User argument - ignored. * @param Port Port number used for the IN operation. * @param pu32 Where to store the result. * @param cb Number of bytes read. */ static DECLCALLBACK(int) efiIOPortRead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb) { PDEVEFI pThis = PDMINS_2_DATA(pDevIns, PDEVEFI); Log4(("EFI in: %x %x\n", Port, cb)); switch (Port) { case EFI_INFO_PORT: if (pThis->iInfoPosition == -1 && cb == 4) { *pu32 = efiInfoSize(pThis); pThis->iInfoPosition = 0; } else { /* So far */ if (cb != 1) return VERR_IOM_IOPORT_UNUSED; *pu32 = efiInfoNextByte(pThis); pThis->iInfoPosition++; } return VINF_SUCCESS; case EFI_PANIC_PORT: #ifdef IN_RING3 LogRel(("Panic port read!\n")); /* Insert special code here on panic reads */ return PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS, "EFI Panic: panic port read!\n"); #else /* Reschedule to R3 */ return VINF_IOM_R3_IOPORT_READ; #endif case EFI_VARIABLE_OP: switch (pThis->NVRAM.enmOp) { case EFI_VM_VARIABLE_OP_START: /* @todo: nop ? */ *pu32 = pThis->NVRAM.u32Status; break; case EFI_VM_VARIABLE_OP_END: break; case EFI_VM_VARIABLE_OP_INDEX: break; case EFI_VM_VARIABLE_OP_GUID: *pu32 = pThis->NVRAM.OperationVarOp.uuid.au8[pThis->NVRAM.idxOpBuffer]; pThis->NVRAM.idxOpBuffer++; break; case EFI_VM_VARIABLE_OP_ATTRIBUTE: *pu32 = pThis->NVRAM.OperationVarOp.u32Attribute; break; case EFI_VM_VARIABLE_OP_NAME: *pu32 = pThis->NVRAM.OperationVarOp.szVariableName[pThis->NVRAM.idxOpBuffer]; pThis->NVRAM.idxOpBuffer++; break; case EFI_VM_VARIABLE_OP_NAME_LENGTH: *pu32 = pThis->NVRAM.OperationVarOp.cbVariableName; break; case EFI_VM_VARIABLE_OP_VALUE: *pu32 = pThis->NVRAM.OperationVarOp.au8Value[pThis->NVRAM.idxOpBuffer]; pThis->NVRAM.idxOpBuffer++; break; case EFI_VM_VARIABLE_OP_VALUE_LENGTH: *pu32 = pThis->NVRAM.OperationVarOp.cbValue; break; default: break; } return VINF_SUCCESS; case EFI_VARIABLE_PARAM: { break; } return VINF_SUCCESS; } return VERR_IOM_IOPORT_UNUSED; } /** * Port I/O Handler for OUT operations. * * @returns VBox status code. * * @param pDevIns The device instance. * @param pvUser User argument - ignored. * @param Port Port number used for the IN operation. * @param u32 The value to output. * @param cb The value size in bytes. */ static DECLCALLBACK(int) efiIOPortWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb) { PDEVEFI pThis = PDMINS_2_DATA(pDevIns, PDEVEFI); Log4(("efi: out %x %x %d\n", Port, u32, cb)); switch (Port) { case EFI_INFO_PORT: pThis->iInfoSelector = u32; pThis->iInfoPosition = -1; break; case EFI_DEBUG_PORT: { /* The raw version. */ switch (u32) { case '\r': Log3(("efi: \n")); break; case '\n': Log3(("efi: \n")); break; case '\t': Log3(("efi: \n")); break; default: Log3(("efi: %c (%02x)\n", u32, u32)); break; } /* The readable, buffered version. */ if (u32 == '\n' || u32 == '\r') { pThis->szMsg[pThis->iMsg] = '\0'; if (pThis->iMsg) { Log(("efi: %s\n", pThis->szMsg)); #ifdef DEVEFI_WITH_VBOXDBG_SCRIPT const char *pszVBoxDbg = strstr(pThis->szMsg, "VBoxDbg> "); if (pszVBoxDbg) { pszVBoxDbg += sizeof("VBoxDbg> ") - 1; PRTSTREAM pStrm; int rc = RTStrmOpen("./DevEFI.VBoxDbg", "a", &pStrm); if (RT_SUCCESS(rc)) { RTStrmPutStr(pStrm, pszVBoxDbg); RTStrmPutCh(pStrm, '\n'); RTStrmClose(pStrm); } } #endif } pThis->iMsg = 0; } else { if (pThis->iMsg >= sizeof(pThis->szMsg)-1) { pThis->szMsg[pThis->iMsg] = '\0'; Log(("efi: %s\n", pThis->szMsg)); pThis->iMsg = 0; } pThis->szMsg[pThis->iMsg] = (char )u32; pThis->szMsg[++pThis->iMsg] = '\0'; } break; } case EFI_PANIC_PORT: { switch (u32) { case EFI_PANIC_CMD_BAD_ORG: case EFI_PANIC_CMD_THUNK_TRAP: LogRel(("EFI Panic: Unexpected trap!!\n")); #ifdef VBOX_STRICT return PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS, "EFI Panic: Unexpected trap during early bootstrap!\n"); #else AssertReleaseMsgFailed(("Unexpected trap during early EFI bootstrap!!\n")); #endif break; case EFI_PANIC_CMD_START_MSG: pThis->iPanicMsg = 0; pThis->szPanicMsg[0] = '\0'; break; case EFI_PANIC_CMD_END_MSG: LogRel(("EFI Panic: %s\n", pThis->szPanicMsg)); #ifdef VBOX_STRICT return PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS, "EFI Panic: %s\n", pThis->szPanicMsg); #else return VERR_INTERNAL_ERROR; #endif default: if ( u32 >= EFI_PANIC_CMD_MSG_FIRST && u32 <= EFI_PANIC_CMD_MSG_LAST) { /* Add the message char to the buffer. */ uint32_t i = pThis->iPanicMsg; if (i + 1 < sizeof(pThis->szPanicMsg)) { char ch = EFI_PANIC_CMD_MSG_GET_CHAR(u32); if ( ch == '\n' && i > 0 && pThis->szPanicMsg[i - 1] == '\r') i--; pThis->szPanicMsg[i] = ch; pThis->szPanicMsg[i + 1] = '\0'; pThis->iPanicMsg = i + 1; } } else Log(("EFI: Unknown panic command: %#x (cb=%d)\n", u32, cb)); break; } break; } case EFI_VARIABLE_OP: { /* clear buffer index */ Assert(u32 < EFI_VM_VARIABLE_OP_MAX); if (u32 >= EFI_VM_VARIABLE_OP_MAX) { u32 = EFI_VARIABLE_OP_STATUS_ERROR; break; } pThis->NVRAM.idxOpBuffer = 0; pThis->NVRAM.enmOp = (EFIVAROP)u32; } break; case EFI_VARIABLE_PARAM: { switch (pThis->NVRAM.enmOp) { case EFI_VM_VARIABLE_OP_START: { pThis->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_BSY; switch (u32) { case EFI_VARIABLE_OP_QUERY: { LogRel(("EFI: variable lookup %RTuuid, %s\n", &pThis->NVRAM.OperationVarOp.uuid, pThis->NVRAM.OperationVarOp.szVariableName)); pThis->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_BSY; PEFIVAR pEfiVar = NULL; memset(pThis->NVRAM.OperationVarOp.au8Value, 0, EFI_VARIABLE_NAME_MAX); int nvramRc = nvramLookupVariableByUuidAndName( pThis, pThis->NVRAM.OperationVarOp.szVariableName, &pThis->NVRAM.OperationVarOp.uuid, &pEfiVar); if (RT_SUCCESS(nvramRc)) { pThis->NVRAM.OperationVarOp.u32Attribute = pEfiVar->u32Attribute; pThis->NVRAM.OperationVarOp.cbVariableName = pEfiVar->cbVariableName; pThis->NVRAM.OperationVarOp.cbValue = pEfiVar->cbValue; memcpy(pThis->NVRAM.OperationVarOp.au8Value, pEfiVar->au8Value, pEfiVar->cbValue); pThis->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_OK; pThis->NVRAM.pCurrentVarOp = pEfiVar; pThis->NVRAM.idxCurrentVar = pEfiVar->idxVariable; LogFlowFunc(("OperationVar: au8Value:%.*Rhxs\n", pThis->NVRAM.OperationVarOp.cbValue, pThis->NVRAM.OperationVarOp.au8Value)); } else pThis->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_NOT_FOUND; } break; case EFI_VARIABLE_OP_ADD: { LogRel(("EFI: variable add %RTuuid, %s\n", &pThis->NVRAM.OperationVarOp.uuid, pThis->NVRAM.OperationVarOp.szVariableName)); PEFIVAR pEfiVar = NULL; LogFlowFunc(("OperationVar: au8Value:%.*Rhxs\n", pThis->NVRAM.OperationVarOp.cbValue, pThis->NVRAM.OperationVarOp.au8Value)); int nvramRc = nvramLookupVariableByUuidAndName( pThis, pThis->NVRAM.OperationVarOp.szVariableName, &pThis->NVRAM.OperationVarOp.uuid, &pEfiVar); if (RT_SUCCESS(nvramRc)) { /* delete or update ? */ /* @todo: check whether pEfiVar is WP */ LogFlowFunc(("pEfiVar: au8Value:%.*Rhxs\n", pEfiVar->cbValue, pEfiVar->au8Value)); if (pThis->NVRAM.OperationVarOp.cbValue == 0) { /* delete */ RTListNodeRemove(&pEfiVar->List); RTMemFree(pEfiVar); pThis->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_OK; pThis->NVRAM.cNvramVariables--; } else { /* update */ pEfiVar->cbValue = pThis->NVRAM.OperationVarOp.cbValue; memcpy(pEfiVar->au8Value, pThis->NVRAM.OperationVarOp.au8Value, pEfiVar->cbValue); pThis->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_OK; } } else { if (pThis->NVRAM.OperationVarOp.cbValue != 0) { pEfiVar = (PEFIVAR)RTMemAllocZ(sizeof(EFIVAR)); if (!pEfiVar) { pThis->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_ERROR; break; } } else { pThis->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_OK; break; } memcpy(pEfiVar, &pThis->NVRAM.OperationVarOp, sizeof(EFIVAR)); RTListInit(&pEfiVar->List); RTListAppend(&pThis->NVRAM.NvramVariableList.List, &pEfiVar->List); pThis->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_OK; pThis->NVRAM.cNvramVariables++; pEfiVar->idxVariable = pThis->NVRAM.iNvramLastIndex; pThis->NVRAM.iNvramLastIndex++; } } LogFunc(("cNvramVariables:%d, iNvramLastIndex:%d\n", pThis->NVRAM.cNvramVariables, pThis->NVRAM.iNvramLastIndex)); break; case EFI_VARIABLE_OP_QUERY_NEXT: { PEFIVAR pEfiVar = RTListNodeGetNext(&pThis->NVRAM.pCurrentVarOp->List, EFIVAR, List); if (pEfiVar) { memcpy(&pThis->NVRAM.OperationVarOp, pEfiVar, sizeof(EFIVAR)); pThis->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_OK; } else pThis->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_NOT_FOUND; } break; default: /* @todo: return error */ break; } } case EFI_VM_VARIABLE_OP_END: break; case EFI_VM_VARIABLE_OP_INDEX: break; case EFI_VM_VARIABLE_OP_GUID: pThis->NVRAM.OperationVarOp.uuid.au8[pThis->NVRAM.idxOpBuffer] = (uint8_t)u32; pThis->NVRAM.idxOpBuffer++; break; case EFI_VM_VARIABLE_OP_ATTRIBUTE: pThis->NVRAM.OperationVarOp.u32Attribute = u32; break; case EFI_VM_VARIABLE_OP_NAME: pThis->NVRAM.OperationVarOp.szVariableName[pThis->NVRAM.idxOpBuffer] = (uint8_t)u32; pThis->NVRAM.idxOpBuffer++; break; case EFI_VM_VARIABLE_OP_NAME_LENGTH: pThis->NVRAM.OperationVarOp.cbVariableName = u32; memset(pThis->NVRAM.OperationVarOp.szVariableName, 0, EFI_VARIABLE_NAME_MAX); break; case EFI_VM_VARIABLE_OP_VALUE: pThis->NVRAM.OperationVarOp.au8Value[pThis->NVRAM.idxOpBuffer] = (uint8_t)u32; pThis->NVRAM.idxOpBuffer++; break; case EFI_VM_VARIABLE_OP_VALUE_LENGTH: pThis->NVRAM.OperationVarOp.cbValue = u32; memset(pThis->NVRAM.OperationVarOp.au8Value, 0, EFI_VARIABLE_VALUE_MAX); break; default: break; } } break; default: Log(("EFI: Write to reserved port %RTiop: %#x (cb=%d)\n", Port, u32, cb)); break; } return VINF_SUCCESS; } /** * Init complete notification. * * @returns VBOX status code. * @param pDevIns The device instance. */ static DECLCALLBACK(int) efiInitComplete(PPDMDEVINS pDevIns) { /* PC Bios */ PDEVEFI pThis = PDMINS_2_DATA(pDevIns, PDEVEFI); uint32_t u32; /* * Memory sizes. */ uint64_t const offRamHole = _4G - pThis->cbRamHole; if (pThis->cbRam > 16 * _1M) u32 = (uint32_t)( (RT_MIN(RT_MIN(pThis->cbRam, offRamHole), UINT32_C(0xffe00000)) - 16U * _1M) / _64K ); else u32 = 0; cmosWrite(pDevIns, 0x34, u32 & 0xff); cmosWrite(pDevIns, 0x35, u32 >> 8); /* * Number of CPUs. */ cmosWrite(pDevIns, 0x60, pThis->cCpus & 0xff); return VINF_SUCCESS; } /** * Reset notification. * * @returns VBox status. * @param pDevIns The device instance data. */ static DECLCALLBACK(void) efiReset(PPDMDEVINS pDevIns) { PDEVEFI pThis = PDMINS_2_DATA(pDevIns, PDEVEFI); int rc; LogFlow(("efiReset\n")); pThis->iInfoSelector = 0; pThis->iInfoPosition = -1; pThis->iMsg = 0; pThis->szMsg[0] = '\0'; pThis->iPanicMsg = 0; pThis->szPanicMsg[0] = '\0'; /* * Plan some structures in RAM. */ FwCommonPlantSmbiosAndDmiHdrs(pDevIns, pThis->cbDmiTables); if (pThis->u8IOAPIC) FwCommonPlantMpsFloatPtr(pDevIns); /* * Re-shadow the Firmware Volume and make it RAM/RAM. */ uint32_t cPages = RT_ALIGN_64(pThis->cbEfiRom, PAGE_SIZE) >> PAGE_SHIFT; RTGCPHYS GCPhys = pThis->GCLoadAddress; while (cPages > 0) { uint8_t abPage[PAGE_SIZE]; /* Read the (original) ROM page and write it back to the RAM page. */ rc = PDMDevHlpROMProtectShadow(pDevIns, GCPhys, PAGE_SIZE, PGMROMPROT_READ_ROM_WRITE_RAM); AssertLogRelRC(rc); rc = PDMDevHlpPhysRead(pDevIns, GCPhys, abPage, PAGE_SIZE); AssertLogRelRC(rc); if (RT_FAILURE(rc)) memset(abPage, 0xcc, sizeof(abPage)); rc = PDMDevHlpPhysWrite(pDevIns, GCPhys, abPage, PAGE_SIZE); AssertLogRelRC(rc); /* Switch to the RAM/RAM mode. */ rc = PDMDevHlpROMProtectShadow(pDevIns, GCPhys, PAGE_SIZE, PGMROMPROT_READ_RAM_WRITE_RAM); AssertLogRelRC(rc); /* Advance */ GCPhys += PAGE_SIZE; cPages--; } } /** * Destruct a device instance. * * Most VM resources are freed by the VM. This callback is provided so that any non-VM * resources can be freed correctly. * * @param pDevIns The device instance data. */ static DECLCALLBACK(int) efiDestruct(PPDMDEVINS pDevIns) { PDEVEFI pThis = PDMINS_2_DATA(pDevIns, PDEVEFI); nvramStore(pThis); nvramFlushDeviceVariableList(pThis); /* * Free MM heap pointers. */ if (pThis->pu8EfiRom) { RTFileReadAllFree(pThis->pu8EfiRom, (size_t)pThis->cbEfiRom); pThis->pu8EfiRom = NULL; } if (pThis->pszEfiRomFile) { MMR3HeapFree(pThis->pszEfiRomFile); pThis->pszEfiRomFile = NULL; } if (pThis->pu8EfiThunk) { MMR3HeapFree(pThis->pu8EfiThunk); pThis->pu8EfiThunk = NULL; } if (pThis->pu8DeviceProps) { MMR3HeapFree(pThis->pu8DeviceProps); pThis->pu8DeviceProps = NULL; pThis->u32DevicePropsLen = 0; } return VINF_SUCCESS; } /** * Helper that searches for a FFS file of a given type. * * @returns Pointer to the FFS file header if found, NULL if not. * * @param pFfsFile Pointer to the FFS file header to start searching at. * @param pbEnd The end of the firmware volume. * @param FileType The file type to look for. * @param pcbFfsFile Where to store the FFS file size (includes header). */ DECLINLINE(EFI_FFS_FILE_HEADER const *) efiFwVolFindFileByType(EFI_FFS_FILE_HEADER const *pFfsFile, uint8_t const *pbEnd, EFI_FV_FILETYPE FileType, uint32_t *pcbFile) { #define FFS_SIZE(hdr) RT_MAKE_U32_FROM_U8((hdr)->Size[0], (hdr)->Size[1], (hdr)->Size[2], 0) while ((uintptr_t)pFfsFile < (uintptr_t)pbEnd) { if (pFfsFile->Type == FileType) { *pcbFile = FFS_SIZE(pFfsFile); LogFunc(("Found %RTuuid of type:%d\n", &pFfsFile->Name, FileType)); return pFfsFile; } pFfsFile = (EFI_FFS_FILE_HEADER *)((uintptr_t)pFfsFile + RT_ALIGN(FFS_SIZE(pFfsFile), 8)); } #undef FFS_SIZE return NULL; } /** * Parse EFI ROM headers and find entry points. * * @returns VBox status. * @param pThis The device instance data. */ static int efiParseFirmware(PDEVEFI pThis) { EFI_FIRMWARE_VOLUME_HEADER const *pFwVolHdr = (EFI_FIRMWARE_VOLUME_HEADER const *)pThis->pu8EfiRom; /* * Validate firmware volume header. */ AssertLogRelMsgReturn(pFwVolHdr->Signature == RT_MAKE_U32_FROM_U8('_', 'F', 'V', 'H'), ("%#x, expected %#x\n", pFwVolHdr->Signature, RT_MAKE_U32_FROM_U8('_', 'F', 'V', 'H')), VERR_INVALID_MAGIC); AssertLogRelMsgReturn(pFwVolHdr->Revision == EFI_FVH_REVISION, ("%#x, expected %#x\n", pFwVolHdr->Signature, EFI_FVH_REVISION), VERR_VERSION_MISMATCH); /** @todo check checksum, see PE spec vol. 3 */ AssertLogRelMsgReturn(pFwVolHdr->FvLength <= pThis->cbEfiRom, ("%#llx, expected %#llx\n", pFwVolHdr->FvLength, pThis->cbEfiRom), VERR_INVALID_PARAMETER); AssertLogRelMsgReturn( pFwVolHdr->BlockMap[0].Length > 0 && pFwVolHdr->BlockMap[0].NumBlocks > 0, ("%#x, %x\n", pFwVolHdr->BlockMap[0].Length, pFwVolHdr->BlockMap[0].NumBlocks), VERR_INVALID_PARAMETER); AssertLogRelMsgReturn(!(pThis->cbEfiRom & PAGE_OFFSET_MASK), ("%RX64\n", pThis->cbEfiRom), VERR_INVALID_PARAMETER); uint8_t const * const pbFwVolEnd = pThis->pu8EfiRom + pFwVolHdr->FvLength; pThis->GCLoadAddress = UINT32_C(0xfffff000) - pThis->cbEfiRom + PAGE_SIZE; return VINF_SUCCESS; } /** * Load EFI ROM file into the memory. * * @returns VBox status. * @param pThis The device instance data. * @param pCfg Configuration node handle for the device. */ static int efiLoadRom(PDEVEFI pThis, PCFGMNODE pCfg) { /* * Read the entire firmware volume into memory. */ void *pvFile; size_t cbFile; int rc = RTFileReadAllEx(pThis->pszEfiRomFile, 0 /*off*/, RTFOFF_MAX /*cbMax*/, RTFILE_RDALL_O_DENY_WRITE, &pvFile, &cbFile); if (RT_FAILURE(rc)) return PDMDevHlpVMSetError(pThis->pDevIns, rc, RT_SRC_POS, N_("Loading the EFI firmware volume '%s' failed with rc=%Rrc"), pThis->pszEfiRomFile, rc); pThis->pu8EfiRom = (uint8_t *)pvFile; pThis->cbEfiRom = cbFile; /* * Validate firmware volume and figure out the load address as well as the SEC entry point. */ rc = efiParseFirmware(pThis); if (RT_FAILURE(rc)) return PDMDevHlpVMSetError(pThis->pDevIns, rc, RT_SRC_POS, N_("Parsing the EFI firmware volume '%s' failed with rc=%Rrc"), pThis->pszEfiRomFile, rc); /* * Map the firmware volume into memory as shadowed ROM. */ /** @todo fix PGMR3PhysRomRegister so it doesn't mess up in SUPLib when mapping a big ROM image. */ RTGCPHYS cbQuart = RT_ALIGN_64(pThis->cbEfiRom / 4, PAGE_SIZE); rc = PDMDevHlpROMRegister(pThis->pDevIns, pThis->GCLoadAddress, cbQuart, pThis->pu8EfiRom, cbQuart, PGMPHYS_ROM_FLAGS_SHADOWED | PGMPHYS_ROM_FLAGS_PERMANENT_BINARY, "EFI Firmware Volume"); AssertRCReturn(rc, rc); rc = PDMDevHlpROMProtectShadow(pThis->pDevIns, pThis->GCLoadAddress, (uint32_t)cbQuart, PGMROMPROT_READ_RAM_WRITE_IGNORE); AssertRCReturn(rc, rc); rc = PDMDevHlpROMRegister(pThis->pDevIns, pThis->GCLoadAddress + cbQuart, cbQuart, pThis->pu8EfiRom + cbQuart, cbQuart, PGMPHYS_ROM_FLAGS_SHADOWED | PGMPHYS_ROM_FLAGS_PERMANENT_BINARY, "EFI Firmware Volume (Part 2)"); if (RT_FAILURE(rc)) return rc; rc = PDMDevHlpROMRegister(pThis->pDevIns, pThis->GCLoadAddress + cbQuart * 2, cbQuart, pThis->pu8EfiRom + cbQuart * 2, cbQuart, PGMPHYS_ROM_FLAGS_SHADOWED | PGMPHYS_ROM_FLAGS_PERMANENT_BINARY, "EFI Firmware Volume (Part 3)"); if (RT_FAILURE(rc)) return rc; rc = PDMDevHlpROMRegister(pThis->pDevIns, pThis->GCLoadAddress + cbQuart * 3, pThis->cbEfiRom - cbQuart * 3, pThis->pu8EfiRom + cbQuart * 3, pThis->cbEfiRom - cbQuart * 3, PGMPHYS_ROM_FLAGS_SHADOWED | PGMPHYS_ROM_FLAGS_PERMANENT_BINARY, "EFI Firmware Volume (Part 4)"); if (RT_FAILURE(rc)) return rc; return VINF_SUCCESS; } static uint8_t efiGetHalfByte(char ch) { uint8_t val; if (ch >= '0' && ch <= '9') val = ch - '0'; else if (ch >= 'A' && ch <= 'F') val = ch - 'A' + 10; else if(ch >= 'a' && ch <= 'f') val = ch - 'a' + 10; else val = 0xff; return val; } static int efiParseDeviceString(PDEVEFI pThis, char* pszDeviceProps) { int rc = 0; uint32_t iStr, iHex, u32OutLen; uint8_t u8Value = 0; /* (shut up gcc) */ bool fUpper = true; u32OutLen = (uint32_t)RTStrNLen(pszDeviceProps, RTSTR_MAX) / 2 + 1; pThis->pu8DeviceProps = (uint8_t*)PDMDevHlpMMHeapAlloc(pThis->pDevIns, u32OutLen); if (!pThis->pu8DeviceProps) return VERR_NO_MEMORY; for (iStr=0, iHex = 0; pszDeviceProps[iStr]; iStr++) { uint8_t u8Hb = efiGetHalfByte(pszDeviceProps[iStr]); if (u8Hb > 0xf) continue; if (fUpper) u8Value = u8Hb << 4; else pThis->pu8DeviceProps[iHex++] = u8Hb | u8Value; Assert(iHex < u32OutLen); fUpper = !fUpper; } Assert(iHex == 0 || fUpper); pThis->u32DevicePropsLen = iHex; return rc; } /** * @copydoc(PDMIBASE::pfnQueryInterface) */ static DECLCALLBACK(void *) devEfi_pfnQueryInterface(PPDMIBASE pInterface, const char *pszIID) { LogFlowFunc(("ENTER: pIBase: %p, pszIID:%p\n", __FUNCTION__, pInterface, pszIID)); PDEVEFI pThis = RT_FROM_MEMBER(pInterface, DEVEFI, Lun0.IBase); PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pThis->Lun0.IBase); return NULL; } /** * @interface_method_impl{PDMDEVREG,pfnConstruct} */ static DECLCALLBACK(int) efiConstruct(PPDMDEVINS pDevIns, int iInstance, PCFGMNODE pCfg) { PDEVEFI pThis = PDMINS_2_DATA(pDevIns, PDEVEFI); int rc; Assert(iInstance == 0); pThis->pDevIns = pDevIns; /* * Validate and read the configuration. */ if (!CFGMR3AreValuesValid(pCfg, "EfiRom\0" "RamSize\0" "RamHoleSize\0" "NumCPUs\0" "UUID\0" "IOAPIC\0" "DmiBIOSFirmwareMajor\0" "DmiBIOSFirmwareMinor\0" "DmiBIOSReleaseDate\0" "DmiBIOSReleaseMajor\0" "DmiBIOSReleaseMinor\0" "DmiBIOSVendor\0" "DmiBIOSVersion\0" "DmiSystemFamily\0" "DmiSystemProduct\0" "DmiSystemSerial\0" "DmiSystemSKU\0" "DmiSystemUuid\0" "DmiSystemVendor\0" "DmiSystemVersion\0" "DmiBoardAssetTag\0" "DmiBoardBoardType\0" "DmiBoardLocInChass\0" "DmiBoardProduct\0" "DmiBoardSerial\0" "DmiBoardVendor\0" "DmiBoardVersion\0" "DmiChassisAssetTag\0" "DmiChassisSerial\0" "DmiChassisVendor\0" "DmiChassisVersion\0" "DmiProcManufacturer\0" "DmiProcVersion\0" "DmiOEMVBoxVer\0" "DmiOEMVBoxRev\0" "DmiUseHostInfo\0" "DmiExposeMemoryTable\0" "DmiExposeProcInf\0" "64BitEntry\0" "BootArgs\0" "DeviceProps\0" "GopMode\0" "UgaHorizontalResolution\0" "UgaVerticalResolution\0")) return PDMDEV_SET_ERROR(pDevIns, VERR_PDM_DEVINS_UNKNOWN_CFG_VALUES, N_("Configuration error: Invalid config value(s) for the EFI device")); /* CPU count (optional). */ rc = CFGMR3QueryU32Def(pCfg, "NumCPUs", &pThis->cCpus, 1); AssertLogRelRCReturn(rc, rc); rc = CFGMR3QueryU8Def(pCfg, "IOAPIC", &pThis->u8IOAPIC, 1); if (RT_FAILURE (rc)) return PDMDEV_SET_ERROR(pDevIns, rc, N_("Configuration error: Failed to read \"IOAPIC\"")); /* * Query the machine's UUID for SMBIOS/DMI use. */ RTUUID uuid; rc = CFGMR3QueryBytes(pCfg, "UUID", &uuid, sizeof(uuid)); if (RT_FAILURE(rc)) return PDMDEV_SET_ERROR(pDevIns, rc, N_("Configuration error: Querying \"UUID\" failed")); /* * Convert the UUID to network byte order. Not entirely straightforward as * parts are MSB already... */ uuid.Gen.u32TimeLow = RT_H2BE_U32(uuid.Gen.u32TimeLow); uuid.Gen.u16TimeMid = RT_H2BE_U16(uuid.Gen.u16TimeMid); uuid.Gen.u16TimeHiAndVersion = RT_H2BE_U16(uuid.Gen.u16TimeHiAndVersion); memcpy(&pThis->aUuid, &uuid, sizeof pThis->aUuid); RTListInit((PRTLISTNODE)&pThis->NVRAM.NvramVariableList.List); /* * RAM sizes */ rc = CFGMR3QueryU64(pCfg, "RamSize", &pThis->cbRam); AssertLogRelRCReturn(rc, rc); rc = CFGMR3QueryU64(pCfg, "RamHoleSize", &pThis->cbRamHole); AssertLogRelRCReturn(rc, rc); pThis->cbBelow4GB = RT_MIN(pThis->cbRam, _4G - pThis->cbRamHole); pThis->cbAbove4GB = pThis->cbRam - pThis->cbBelow4GB; /* * Get the system EFI ROM file name. */ rc = CFGMR3QueryStringAlloc(pCfg, "EfiRom", &pThis->pszEfiRomFile); if (rc == VERR_CFGM_VALUE_NOT_FOUND) { pThis->pszEfiRomFile = (char *)PDMDevHlpMMHeapAlloc(pDevIns, RTPATH_MAX); if (!pThis->pszEfiRomFile) return VERR_NO_MEMORY; rc = RTPathAppPrivateArchTop(pThis->pszEfiRomFile, RTPATH_MAX); AssertRCReturn(rc, rc); rc = RTPathAppend(pThis->pszEfiRomFile, RTPATH_MAX, "VBoxEFI32.fd"); AssertRCReturn(rc, rc); } else if (RT_FAILURE(rc)) return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS, N_("Configuration error: Querying \"EfiRom\" as a string failed")); else if (!*pThis->pszEfiRomFile) { MMR3HeapFree(pThis->pszEfiRomFile); pThis->pszEfiRomFile = NULL; } /* NVRAM processing */ pThis->Lun0.IBase.pfnQueryInterface = devEfi_pfnQueryInterface; #if 0 rc = PDMDevHlpSSMRegisterEx(pDevIns, EFI_SSM_VERSION, sizeof(*pThis), NULL /*pszBefore*/, NULL /*pfnLivePrep*/, NULL /*pfnLiveExec*/, NULL /*pfnLiveDone*/, NULL /*pfnSavePrep*/, efiSaveExec, NULL /*pfnSaveDone*/, NULL /*pfnLoadPrep*/, efiLoadExec, efiLoadDone); #else rc = PDMDevHlpSSMRegister(pDevIns, EFI_SSM_VERSION, sizeof(*pThis), efiSaveExec, efiLoadExec); #endif AssertRCReturn(rc, rc); rc = PDMDevHlpDriverAttach(pDevIns, 0, &pThis->Lun0.IBase, &pThis->Lun0.pDrvBase, "NvramStorage"); if (RT_FAILURE(rc)) return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS, N_("Can't attach Nvram Storage driver")); pThis->Lun0.pNvramDown = (PPDMINVRAM)pThis->Lun0.pDrvBase->pfnQueryInterface(pThis->Lun0.pDrvBase, PDMINVRAM_IID); AssertPtrReturn(pThis->Lun0.pNvramDown, VERR_PDM_MISSING_INTERFACE_BELOW); nvramLoad(pThis); /* * Get boot args. */ rc = CFGMR3QueryString(pCfg, "BootArgs", pThis->szBootArgs, sizeof pThis->szBootArgs); if (rc == VERR_CFGM_VALUE_NOT_FOUND) { strcpy(pThis->szBootArgs, ""); rc = VINF_SUCCESS; } if (RT_FAILURE(rc)) return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS, N_("Configuration error: Querying \"BootArgs\" as a string failed")); LogRel(("EFI boot args: %s\n", pThis->szBootArgs)); /* * Get device props. */ char* pszDeviceProps; rc = CFGMR3QueryStringAlloc(pCfg, "DeviceProps", &pszDeviceProps); if (rc == VERR_CFGM_VALUE_NOT_FOUND) { pszDeviceProps = NULL; rc = VINF_SUCCESS; } if (RT_FAILURE(rc)) return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS, N_("Configuration error: Querying \"DeviceProps\" as a string failed")); if (pszDeviceProps) { LogRel(("EFI device props: %s\n", pszDeviceProps)); rc = efiParseDeviceString(pThis, pszDeviceProps); MMR3HeapFree(pszDeviceProps); if (RT_FAILURE(rc)) return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS, N_("Configuration error: Cannot parse device properties")); } else { pThis->pu8DeviceProps = NULL; pThis->u32DevicePropsLen = 0; } /* * CPU frequencies */ // @todo: we need to have VMM API to access TSC increase speed, for now provide reasonable default pThis->u64TscFrequency = RTMpGetMaxFrequency(0) * 1000 * 1000;// TMCpuTicksPerSecond(PDMDevHlpGetVM(pDevIns)); if (pThis->u64TscFrequency == 0) pThis->u64TscFrequency = UINT64_C(2500000000); /* Multiplier is read from MSR_IA32_PERF_STATUS, and now is hardcoded as 4 */ pThis->u64FsbFrequency = pThis->u64TscFrequency / 4; pThis->u64CpuFrequency = pThis->u64TscFrequency; /* * GOP graphics */ rc = CFGMR3QueryU32(pCfg, "GopMode", &pThis->u32GopMode); AssertRC(rc); if (pThis->u32GopMode == UINT32_MAX) { pThis->u32GopMode = 2; /* 1024x768 */ } /* * Uga graphics */ rc = CFGMR3QueryU32(pCfg, "UgaHorizontalResolution", &pThis->u32UgaHorisontal); AssertRC(rc); if (pThis->u32UgaHorisontal == 0) { pThis->u32UgaHorisontal = 1024; /* 1024x768 */ } rc = CFGMR3QueryU32(pCfg, "UgaVerticalResolution", &pThis->u32UgaVertical); AssertRC(rc); if (pThis->u32UgaVertical == 0) { pThis->u32UgaVertical = 768; /* 1024x768 */ } #ifdef DEVEFI_WITH_VBOXDBG_SCRIPT /* * Zap the debugger script */ RTFileDelete("./DevEFI.VBoxDbg"); #endif /* * Load firmware volume and thunk ROM. */ rc = efiLoadRom(pThis, pCfg); if (RT_FAILURE(rc)) return rc; /* * Register our communication ports. */ rc = PDMDevHlpIOPortRegister(pDevIns, EFI_PORT_BASE, EFI_PORT_COUNT, NULL, efiIOPortWrite, efiIOPortRead, NULL, NULL, "EFI communication ports"); if (RT_FAILURE(rc)) return rc; /* * Plant DMI and MPS tables * XXX I wonder if we really need these tables as there is no SMBIOS header... */ uint16_t cbDmiTablesDummy; rc = FwCommonPlantDMITable(pDevIns, pThis->au8DMIPage, VBOX_DMI_TABLE_SIZE, &pThis->aUuid, pDevIns->pCfg, pThis->cCpus, &pThis->cbDmiTables); AssertRCReturn(rc, rc); if (pThis->u8IOAPIC) FwCommonPlantMpsTable(pDevIns, pThis->au8DMIPage + VBOX_DMI_TABLE_SIZE, _4K - VBOX_DMI_TABLE_SIZE, pThis->cCpus); rc = PDMDevHlpROMRegister(pDevIns, VBOX_DMI_TABLE_BASE, _4K, pThis->au8DMIPage, _4K, PGMPHYS_ROM_FLAGS_PERMANENT_BINARY, "DMI tables"); AssertRCReturn(rc, rc); /* * Call reset to set things up. */ efiReset(pDevIns); return VINF_SUCCESS; } /** * The device registration structure. */ const PDMDEVREG g_DeviceEFI = { /* u32Version */ PDM_DEVREG_VERSION, /* szName */ "efi", /* szRCMod */ "", /* szR0Mod */ "", /* pszDescription */ "Extensible Firmware Interface Device", /* fFlags */ PDM_DEVREG_FLAGS_HOST_BITS_DEFAULT | PDM_DEVREG_FLAGS_GUEST_BITS_32_64, /* fClass */ PDM_DEVREG_CLASS_ARCH_BIOS, /* cMaxInstances */ 1, /* cbInstance */ sizeof(DEVEFI), /* pfnConstruct */ efiConstruct, /* pfnDestruct */ efiDestruct, /* pfnRelocate */ NULL, /* pfnIOCtl */ NULL, /* pfnPowerOn */ NULL, /* pfnReset */ efiReset, /* pfnSuspend */ NULL, /* pfnResume */ NULL, /* pfnAttach */ NULL, /* pfnDetach */ NULL, /* pfnQueryInterface. */ NULL, /* pfnInitComplete. */ efiInitComplete, /* pfnPowerOff */ NULL, /* pfnSoftReset */ NULL, /* u32VersionEnd */ PDM_DEVREG_VERSION };