VirtualBox

source: vbox/trunk/src/VBox/Devices/EFI/DevEFI.cpp@ 88566

Last change on this file since 88566 was 85950, checked in by vboxsync, 4 years ago

Devices/DevEFI: Support querying the configured APIC mode

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 103.0 KB
Line 
1/* $Id: DevEFI.cpp 85950 2020-08-31 11:40:19Z vboxsync $ */
2/** @file
3 * DevEFI - EFI <-> VirtualBox Integration Framework.
4 */
5
6/*
7 * Copyright (C) 2006-2020 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#define LOG_GROUP LOG_GROUP_DEV_EFI
23
24#include <VBox/vmm/pdmdev.h>
25#include <VBox/vmm/pgm.h>
26#include <VBox/vmm/cpum.h>
27#include <VBox/vmm/mm.h>
28#include <VBox/log.h>
29#include <VBox/err.h>
30#include <VBox/param.h>
31#include <VBox/vmm/dbgf.h>
32#include <VBox/vmm/pdmnvram.h>
33
34#include <iprt/asm.h>
35#include <iprt/assert.h>
36#include <iprt/ctype.h>
37#include <iprt/file.h>
38#include <iprt/mem.h>
39#include <iprt/string.h>
40#include <iprt/uuid.h>
41#include <iprt/path.h>
42#include <iprt/string.h>
43#include <iprt/mp.h>
44#include <iprt/list.h>
45#if defined(DEBUG) && defined(IN_RING3)
46# include <iprt/stream.h>
47# define DEVEFI_WITH_VBOXDBG_SCRIPT
48#endif
49#include <iprt/utf16.h>
50
51#include "DevEFI.h"
52#include "FlashCore.h"
53#include "VBoxDD.h"
54#include "VBoxDD2.h"
55#include "../PC/DevFwCommon.h"
56
57/* EFI includes */
58#ifdef IN_RING3
59# ifdef _MSC_VER
60# pragma warning(push)
61# pragma warning(disable:4668)
62# endif
63# include <ProcessorBind.h>
64# ifdef _MSC_VER
65# pragma warning(pop)
66# endif
67# include <Common/UefiBaseTypes.h>
68# include <Common/PiFirmwareVolume.h>
69# include <Common/PiFirmwareFile.h>
70#endif
71
72
73/*********************************************************************************************************************************
74* Structures and Typedefs *
75*********************************************************************************************************************************/
76/**
77 * EFI NVRAM variable.
78 */
79typedef struct EFIVAR
80{
81 /** The list node for the variable. */
82 RTLISTNODE ListNode;
83 /** The unique sequence number of the variable.
84 * This is used to find pCurVar when restoring saved state and therefore only
85 * set when saving. */
86 uint32_t idUniqueSavedState;
87 /** The value attributess. */
88 uint32_t fAttributes;
89 /** The variable name length (not counting the terminator char). */
90 uint32_t cchName;
91 /** The size of the value. This cannot be zero. */
92 uint32_t cbValue;
93 /** The vendor UUID scoping the variable name. */
94 RTUUID uuid;
95 /** The variable name. */
96 char szName[EFI_VARIABLE_NAME_MAX];
97 /** The variable value bytes. */
98 uint8_t abValue[EFI_VARIABLE_VALUE_MAX];
99} EFIVAR;
100/** Pointer to an EFI NVRAM variable. */
101typedef EFIVAR *PEFIVAR;
102/** Pointer to a const EFI NVRAM variable. */
103typedef EFIVAR const *PCEFIVAR;
104/** Pointer to an EFI NVRAM variable pointer. */
105typedef PEFIVAR *PPEFIVAR;
106
107/**
108 * NVRAM state.
109 */
110typedef struct NVRAMDESC
111{
112 /** The current operation. */
113 EFIVAROP enmOp;
114 /** The current status. */
115 uint32_t u32Status;
116 /** The current */
117 uint32_t offOpBuffer;
118 /** The current number of variables. */
119 uint32_t cVariables;
120 /** The list of variables. */
121 RTLISTANCHOR VarList;
122
123 /** The unique variable sequence ID, for the saved state only.
124 * @todo It's part of this structure for hysterical raisins, consider remove it
125 * when changing the saved state format the next time. */
126 uint32_t idUniqueCurVar;
127 /** Variable buffered used both when adding and querying NVRAM variables.
128 * When querying a variable, a copy of it is stored in this buffer and read
129 * from it. When adding, updating or deleting a variable, this buffer is used
130 * to set up the parameters before taking action. */
131 EFIVAR VarOpBuf;
132 /** The current variable. This is only used by EFI_VARIABLE_OP_QUERY_NEXT,
133 * the attribute readers work against the copy in VarOpBuf. */
134 PEFIVAR pCurVar;
135} NVRAMDESC;
136
137
138/**
139 * The EFI device shared state structure.
140 */
141typedef struct DEVEFI
142{
143 /** The flash device containing the NVRAM. */
144 FLASHCORE Flash;
145 /** The 8 I/O ports at 0xEF10 (EFI_PORT_BASE). */
146 IOMIOPORTHANDLE hIoPorts;
147 /** The flash MMIO handle. */
148 IOMMMIOHANDLE hMmioFlash;
149} DEVEFI;
150/** Pointer to the shared EFI state. */
151typedef DEVEFI *PDEVEFI;
152
153/**
154 * The EFI device state structure for ring-3.
155 */
156typedef struct DEVEFIR3
157{
158 /** Pointer back to the device instance. */
159 PPDMDEVINS pDevIns;
160
161 /** EFI message buffer. */
162 char szMsg[VBOX_EFI_DEBUG_BUFFER];
163 /** EFI message buffer index. */
164 uint32_t iMsg;
165
166 /** EFI panic message buffer. */
167 char szPanicMsg[2048];
168 /** EFI panic message buffer index. */
169 uint32_t iPanicMsg;
170
171 struct
172 {
173 /** The current/last image event. */
174 uint8_t uEvt;
175 /** Module path/name offset. */
176 uint8_t offName;
177 /** The offset of the last component in the module path/name. */
178 uint8_t offNameLastComponent;
179 /** Alignment padding. */
180 uint8_t abPadding[5];
181 /** First address associated with the event (image address). */
182 uint64_t uAddr0;
183 /** Second address associated with the event (old image address). */
184 uint64_t uAddr1;
185 /** The size associated with the event (0 if none). */
186 uint64_t cb0;
187 /** The module name. */
188 char szName[256];
189 } ImageEvt;
190
191 /** The system EFI ROM data. */
192 uint8_t const *pu8EfiRom;
193 /** The system EFI ROM data pointer to be passed to RTFileReadAllFree. */
194 uint8_t *pu8EfiRomFree;
195 /** The size of the system EFI ROM. */
196 uint64_t cbEfiRom;
197 /** Offset into the actual ROM within EFI FW volume. */
198 uint64_t offEfiRom;
199 /** The name of the EFI ROM file. */
200 char *pszEfiRomFile;
201 /** Thunk page pointer. */
202 uint8_t *pu8EfiThunk;
203 /** First entry point of the EFI firmware. */
204 RTGCPHYS GCEntryPoint0;
205 /** Second Entry Point (PeiCore)*/
206 RTGCPHYS GCEntryPoint1;
207 /** EFI firmware physical load address. */
208 RTGCPHYS GCLoadAddress;
209 /** Current info selector. */
210 uint32_t iInfoSelector;
211 /** Current info position. */
212 int32_t offInfo;
213
214 /** Number of virtual CPUs. (Config) */
215 uint32_t cCpus;
216
217 /** The size of the DMI tables. */
218 uint16_t cbDmiTables;
219 /** Number of the DMI tables. */
220 uint16_t cNumDmiTables;
221 /** The DMI tables. */
222 uint8_t au8DMIPage[0x1000];
223
224 /** I/O-APIC enabled? */
225 uint8_t u8IOAPIC;
226
227 /** APIC mode to be set up by firmware. */
228 uint8_t u8APIC;
229
230 /** Boot parameters passed to the firmware. */
231 char szBootArgs[256];
232
233 /** Host UUID (for DMI). */
234 RTUUID aUuid;
235
236 /** Device properties buffer. */
237 R3PTRTYPE(uint8_t *) pbDeviceProps;
238 /** Device properties buffer size. */
239 uint32_t cbDeviceProps;
240
241 /** Virtual machine front side bus frequency. */
242 uint64_t u64FsbFrequency;
243 /** Virtual machine time stamp counter frequency. */
244 uint64_t u64TscFrequency;
245 /** Virtual machine CPU frequency. */
246 uint64_t u64CpuFrequency;
247 /** EFI Graphics mode (used as fallback if resolution is not known). */
248 uint32_t u32GraphicsMode;
249 /** EFI Graphics (GOP or UGA) horizontal resolution. */
250 uint32_t u32HorizontalResolution;
251 /** EFI Graphics (GOP or UGA) vertical resolution. */
252 uint32_t u32VerticalResolution;
253 /** Physical address of PCI config space MMIO region */
254 uint64_t u64McfgBase;
255 /** Length of PCI config space MMIO region */
256 uint64_t cbMcfgLength;
257 /** Size of the configured NVRAM device. */
258 uint32_t cbNvram;
259 /** Start address of the NVRAM flash. */
260 RTGCPHYS GCPhysNvram;
261
262 /** NVRAM state variables. */
263 NVRAMDESC NVRAM;
264 /** Filename of the file containing the NVRAM store. */
265 char *pszNvramFile;
266
267 /**
268 * NVRAM port - LUN\#0.
269 */
270 struct
271 {
272 /** The base interface we provide the NVRAM driver. */
273 PDMIBASE IBase;
274 /** The NVRAM driver base interface. */
275 PPDMIBASE pDrvBase;
276 /** The NVRAM interface provided by the driver. */
277 PPDMINVRAMCONNECTOR pNvramDrv;
278 } Lun0;
279} DEVEFIR3;
280/** Pointer to the ring-3 EFI state. */
281typedef DEVEFIR3 *PDEVEFIR3;
282
283
284/**
285 * The EFI device state structure for ring-0.
286 */
287typedef struct DEVEFIR0
288{
289 uint32_t uEmpty;
290} DEVEFIR0;
291/** Pointer to the ring-0 EFI state. */
292typedef DEVEFIR0 *PDEVEFIR0;
293
294
295/**
296 * The EFI device state structure for raw-mode.
297 */
298typedef struct DEVEFIRC
299{
300 uint32_t uEmpty;
301} DEVEFIRC;
302/** Pointer to the raw-mode EFI state. */
303typedef DEVEFIRC *PDEVEFIRC;
304
305
306/** @typedef DEVEFICC
307 * The instance data for the current context. */
308/** @typedef PDEVEFICC
309 * Pointer to the instance data for the current context. */
310#ifdef IN_RING3
311typedef DEVEFIR3 DEVEFICC;
312typedef PDEVEFIR3 PDEVEFICC;
313#elif defined(IN_RING0)
314typedef DEVEFIR0 DEVEFICC;
315typedef PDEVEFIR0 PDEVEFICC;
316#elif defined(IN_RC)
317typedef DEVEFIRC DEVEFICC;
318typedef PDEVEFIRC PDEVEFICC;
319#else
320# error "Not IN_RING3, IN_RING0 or IN_RC"
321#endif
322
323
324/*********************************************************************************************************************************
325* Defined Constants And Macros *
326*********************************************************************************************************************************/
327/** The saved state version. */
328#define EFI_SSM_VERSION 3
329/** The saved state version before working NVRAM support was implemented. */
330#define EFI_SSM_VERSION_PRE_PROPER_NVRAM 2
331/** The saved state version from VBox 4.2. */
332#define EFI_SSM_VERSION_4_2 1
333
334/** Non-volatile EFI variable. */
335#define VBOX_EFI_VARIABLE_NON_VOLATILE UINT32_C(0x00000001)
336/** Non-volatile EFI variable. */
337#define VBOX_EFI_VARIABLE_READ_ONLY UINT32_C(0x00000008)
338
339
340/*********************************************************************************************************************************
341* Global Variables *
342*********************************************************************************************************************************/
343#ifdef IN_RING3
344/** Saved state NVRAMDESC field descriptors. */
345static SSMFIELD const g_aEfiNvramDescField[] =
346{
347 SSMFIELD_ENTRY( NVRAMDESC, enmOp),
348 SSMFIELD_ENTRY( NVRAMDESC, u32Status),
349 SSMFIELD_ENTRY( NVRAMDESC, offOpBuffer),
350 SSMFIELD_ENTRY_IGNORE(NVRAMDESC, VarOpBuf),
351 SSMFIELD_ENTRY( NVRAMDESC, cVariables),
352 SSMFIELD_ENTRY_OLD( idUnquireLast, 4),
353 SSMFIELD_ENTRY_IGNORE(NVRAMDESC, VarList),
354 SSMFIELD_ENTRY( NVRAMDESC, idUniqueCurVar),
355 SSMFIELD_ENTRY_IGNORE(NVRAMDESC, pCurVar),
356 SSMFIELD_ENTRY_TERM()
357};
358
359/** Saved state EFIVAR field descriptors. */
360static SSMFIELD const g_aEfiVariableDescFields[] =
361{
362 SSMFIELD_ENTRY_IGNORE(EFIVAR, ListNode),
363 SSMFIELD_ENTRY( EFIVAR, idUniqueSavedState),
364 SSMFIELD_ENTRY( EFIVAR, uuid),
365 SSMFIELD_ENTRY( EFIVAR, szName),
366 SSMFIELD_ENTRY_OLD( cchName, 4),
367 SSMFIELD_ENTRY( EFIVAR, abValue),
368 SSMFIELD_ENTRY( EFIVAR, cbValue),
369 SSMFIELD_ENTRY( EFIVAR, fAttributes),
370 SSMFIELD_ENTRY_TERM()
371};
372
373/** The EfiSystemNvDataFv GUID for NVRAM storage. */
374static const RTUUID g_UuidNvDataFv = { { 0x8d, 0x2b, 0xf1, 0xff, 0x96, 0x76, 0x8b, 0x4c, 0xa9, 0x85, 0x27, 0x47, 0x07, 0x5b, 0x4f, 0x50} };
375
376# ifdef VBOX_WITH_EFI_IN_DD2
377/** Special file name value for indicating the 32-bit built-in EFI firmware. */
378static const char g_szEfiBuiltin32[] = "VBoxEFI32.fd";
379/** Special file name value for indicating the 64-bit built-in EFI firmware. */
380static const char g_szEfiBuiltin64[] = "VBoxEFI64.fd";
381# endif
382#endif /* IN_RING3 */
383
384
385#ifdef IN_RING3
386
387/**
388 * Flushes the variable list.
389 *
390 * @param pThisCC The EFI state for the current context.
391 */
392static void nvramFlushDeviceVariableList(PDEVEFIR3 pThisCC)
393{
394 while (!RTListIsEmpty(&pThisCC->NVRAM.VarList))
395 {
396 PEFIVAR pEfiVar = RTListNodeGetNext(&pThisCC->NVRAM.VarList, EFIVAR, ListNode);
397 RTListNodeRemove(&pEfiVar->ListNode);
398 RTMemFree(pEfiVar);
399 }
400
401 pThisCC->NVRAM.pCurVar = NULL;
402}
403
404/**
405 * This function looks up variable in NVRAM list.
406 */
407static int nvramLookupVariableByUuidAndName(PDEVEFIR3 pThisCC, char *pszVariableName, PCRTUUID pUuid, PPEFIVAR ppEfiVar)
408{
409 LogFlowFunc(("%RTuuid::'%s'\n", pUuid, pszVariableName));
410 size_t const cchVariableName = strlen(pszVariableName);
411 int rc = VERR_NOT_FOUND;
412
413 /*
414 * Start by checking the last variable queried.
415 */
416 if ( pThisCC->NVRAM.pCurVar
417 && pThisCC->NVRAM.pCurVar->cchName == cchVariableName
418 && memcmp(pThisCC->NVRAM.pCurVar->szName, pszVariableName, cchVariableName + 1) == 0
419 && RTUuidCompare(&pThisCC->NVRAM.pCurVar->uuid, pUuid) == 0
420 )
421 {
422 *ppEfiVar = pThisCC->NVRAM.pCurVar;
423 rc = VINF_SUCCESS;
424 }
425 else
426 {
427 /*
428 * Linear list search.
429 */
430 PEFIVAR pEfiVar;
431 RTListForEach(&pThisCC->NVRAM.VarList, pEfiVar, EFIVAR, ListNode)
432 {
433 Assert(strlen(pEfiVar->szName) == pEfiVar->cchName);
434 if ( pEfiVar->cchName == cchVariableName
435 && memcmp(pEfiVar->szName, pszVariableName, cchVariableName + 1) == 0
436 && RTUuidCompare(&pEfiVar->uuid, pUuid) == 0)
437 {
438 *ppEfiVar = pEfiVar;
439 rc = VINF_SUCCESS;
440 break;
441 }
442 }
443 }
444
445 LogFlowFunc(("rc=%Rrc pEfiVar=%p\n", rc, *ppEfiVar));
446 return rc;
447}
448
449
450/**
451 * Inserts the EFI variable into the list.
452 *
453 * This enforces the desired list ordering and/or insertion policy.
454 *
455 * @param pThisCC The EFI state for the current context.
456 * @param pEfiVar The variable to insert.
457 */
458static void nvramInsertVariable(PDEVEFIR3 pThisCC, PEFIVAR pEfiVar)
459{
460#if 1
461 /*
462 * Sorted by UUID and name.
463 */
464 PEFIVAR pCurVar;
465 RTListForEach(&pThisCC->NVRAM.VarList, pCurVar, EFIVAR, ListNode)
466 {
467 int iDiff = RTUuidCompare(&pEfiVar->uuid, &pCurVar->uuid);
468 if (!iDiff)
469 iDiff = strcmp(pEfiVar->szName, pCurVar->szName);
470 if (iDiff < 0)
471 {
472 RTListNodeInsertBefore(&pCurVar->ListNode, &pEfiVar->ListNode);
473 return;
474 }
475 }
476#endif
477
478 /*
479 * Add it at the end.
480 */
481 RTListAppend(&pThisCC->NVRAM.VarList, &pEfiVar->ListNode);
482}
483
484
485/**
486 * Creates an device internal list of variables.
487 *
488 * @returns VBox status code.
489 * @param pThisCC The EFI state for the current context.
490 */
491static int nvramLoad(PDEVEFIR3 pThisCC)
492{
493 int rc;
494 for (uint32_t iVar = 0; iVar < EFI_VARIABLE_MAX; iVar++)
495 {
496 PEFIVAR pEfiVar = (PEFIVAR)RTMemAllocZ(sizeof(EFIVAR));
497 AssertReturn(pEfiVar, VERR_NO_MEMORY);
498
499 pEfiVar->cchName = sizeof(pEfiVar->szName);
500 pEfiVar->cbValue = sizeof(pEfiVar->abValue);
501 rc = pThisCC->Lun0.pNvramDrv->pfnVarQueryByIndex(pThisCC->Lun0.pNvramDrv, iVar,
502 &pEfiVar->uuid, &pEfiVar->szName[0], &pEfiVar->cchName,
503 &pEfiVar->fAttributes, &pEfiVar->abValue[0], &pEfiVar->cbValue);
504 if (RT_SUCCESS(rc))
505 {
506 /* Some validations. */
507 rc = RTStrValidateEncoding(pEfiVar->szName);
508 size_t cchName = RTStrNLen(pEfiVar->szName, sizeof(pEfiVar->szName));
509 if (cchName != pEfiVar->cchName)
510 rc = VERR_INVALID_PARAMETER;
511 if (pEfiVar->cbValue == 0)
512 rc = VERR_NO_DATA;
513 if (RT_FAILURE(rc))
514 LogRel(("EFI/nvramLoad: Bad variable #%u: cbValue=%#x cchName=%#x (strlen=%#x) szName=%.*Rhxs\n",
515 iVar, pEfiVar->cbValue, pEfiVar->cchName, cchName, pEfiVar->cchName + 1, pEfiVar->szName));
516 }
517 if (RT_FAILURE(rc))
518 {
519 RTMemFree(pEfiVar);
520 if (rc == VERR_NOT_FOUND)
521 rc = VINF_SUCCESS;
522 AssertRC(rc);
523 return rc;
524 }
525
526 /* Append it. */
527 nvramInsertVariable(pThisCC, pEfiVar);
528 pThisCC->NVRAM.cVariables++;
529 }
530
531 AssertLogRelMsgFailed(("EFI: Too many variables.\n"));
532 return VERR_TOO_MUCH_DATA;
533}
534
535
536/**
537 * Let the NVRAM driver store the internal NVRAM variable list.
538 *
539 * @returns VBox status code.
540 * @param pThisCC The EFI state for the current context.
541 */
542static int nvramStore(PDEVEFIR3 pThisCC)
543{
544 /*
545 * Count the non-volatile variables and issue the begin call.
546 */
547 PEFIVAR pEfiVar;
548 uint32_t cNonVolatile = 0;
549 RTListForEach(&pThisCC->NVRAM.VarList, pEfiVar, EFIVAR, ListNode)
550 if (pEfiVar->fAttributes & VBOX_EFI_VARIABLE_NON_VOLATILE)
551 cNonVolatile++;
552 int rc = pThisCC->Lun0.pNvramDrv->pfnVarStoreSeqBegin(pThisCC->Lun0.pNvramDrv, cNonVolatile);
553 if (RT_SUCCESS(rc))
554 {
555 /*
556 * Store each non-volatile variable.
557 */
558 uint32_t idxVar = 0;
559 RTListForEach(&pThisCC->NVRAM.VarList, pEfiVar, EFIVAR, ListNode)
560 {
561 /* Skip volatile variables. */
562 if (!(pEfiVar->fAttributes & VBOX_EFI_VARIABLE_NON_VOLATILE))
563 continue;
564
565 int rc2 = pThisCC->Lun0.pNvramDrv->pfnVarStoreSeqPut(pThisCC->Lun0.pNvramDrv, idxVar,
566 &pEfiVar->uuid, pEfiVar->szName, pEfiVar->cchName,
567 pEfiVar->fAttributes, pEfiVar->abValue, pEfiVar->cbValue);
568 if (RT_FAILURE(rc2) && RT_SUCCESS_NP(rc))
569 {
570 LogRel(("EFI: pfnVarStoreVarByIndex failed: %Rrc\n", rc));
571 rc = rc2;
572 }
573 idxVar++;
574 }
575 Assert(idxVar == cNonVolatile);
576
577 /*
578 * Done.
579 */
580 rc = pThisCC->Lun0.pNvramDrv->pfnVarStoreSeqEnd(pThisCC->Lun0.pNvramDrv, rc);
581 }
582 else
583 LogRel(("EFI: pfnVarStoreBegin failed: %Rrc\n", rc));
584 return rc;
585}
586
587/**
588 * EFI_VARIABLE_OP_QUERY and EFI_VARIABLE_OP_QUERY_NEXT worker that copies the
589 * variable into the VarOpBuf, set pCurVar and u32Status.
590 *
591 * @param pThisCC The EFI state for the current context.
592 * @param pEfiVar The resulting variable. NULL if not found / end.
593 * @param fEnumQuery Set if enumeration query, clear if specific.
594 */
595static void nvramWriteVariableOpQueryCopyResult(PDEVEFIR3 pThisCC, PEFIVAR pEfiVar, bool fEnumQuery)
596{
597 RT_ZERO(pThisCC->NVRAM.VarOpBuf.abValue);
598 if (pEfiVar)
599 {
600 RT_ZERO(pThisCC->NVRAM.VarOpBuf.szName);
601 pThisCC->NVRAM.VarOpBuf.uuid = pEfiVar->uuid;
602 pThisCC->NVRAM.VarOpBuf.cchName = pEfiVar->cchName;
603 memcpy(pThisCC->NVRAM.VarOpBuf.szName, pEfiVar->szName, pEfiVar->cchName); /* no need for + 1. */
604 pThisCC->NVRAM.VarOpBuf.fAttributes = pEfiVar->fAttributes;
605 pThisCC->NVRAM.VarOpBuf.cbValue = pEfiVar->cbValue;
606 memcpy(pThisCC->NVRAM.VarOpBuf.abValue, pEfiVar->abValue, pEfiVar->cbValue);
607 pThisCC->NVRAM.pCurVar = pEfiVar;
608 pThisCC->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_OK;
609 LogFlow(("EFI: Variable query -> %RTuuid::'%s' (%d) abValue=%.*Rhxs\n", &pThisCC->NVRAM.VarOpBuf.uuid,
610 pThisCC->NVRAM.VarOpBuf.szName, pThisCC->NVRAM.VarOpBuf.cchName,
611 pThisCC->NVRAM.VarOpBuf.cbValue, pThisCC->NVRAM.VarOpBuf.abValue));
612 }
613 else
614 {
615 if (fEnumQuery)
616 LogFlow(("EFI: Variable query -> NOT_FOUND \n"));
617 else
618 LogFlow(("EFI: Variable query %RTuuid::'%s' -> NOT_FOUND \n",
619 &pThisCC->NVRAM.VarOpBuf.uuid, pThisCC->NVRAM.VarOpBuf.szName));
620 RT_ZERO(pThisCC->NVRAM.VarOpBuf.szName);
621 pThisCC->NVRAM.VarOpBuf.fAttributes = 0;
622 pThisCC->NVRAM.VarOpBuf.cbValue = 0;
623 pThisCC->NVRAM.VarOpBuf.cchName = 0;
624 pThisCC->NVRAM.pCurVar = NULL;
625 pThisCC->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_NOT_FOUND;
626 }
627}
628
629/**
630 * Implements EFI_VARIABLE_PARAM + EFI_VARIABLE_OP_QUERY.
631 *
632 * @returns IOM strict status code.
633 * @param pThisCC The EFI state for the current context.
634 */
635static int nvramWriteVariableOpQuery(PDEVEFIR3 pThisCC)
636{
637 Log(("EFI_VARIABLE_OP_QUERY: %RTuuid::'%s'\n", &pThisCC->NVRAM.VarOpBuf.uuid, pThisCC->NVRAM.VarOpBuf.szName));
638
639 PEFIVAR pEfiVar;
640 int rc = nvramLookupVariableByUuidAndName(pThisCC,
641 pThisCC->NVRAM.VarOpBuf.szName,
642 &pThisCC->NVRAM.VarOpBuf.uuid,
643 &pEfiVar);
644 nvramWriteVariableOpQueryCopyResult(pThisCC, RT_SUCCESS(rc) ? pEfiVar : NULL, false /*fEnumQuery*/);
645 return VINF_SUCCESS;
646}
647
648/**
649 * Implements EFI_VARIABLE_PARAM + EFI_VARIABLE_OP_QUERY_NEXT.
650 *
651 * This simply walks the list.
652 *
653 * @returns IOM strict status code.
654 * @param pThisCC The EFI state for the current context.
655 */
656static int nvramWriteVariableOpQueryNext(PDEVEFIR3 pThisCC)
657{
658 Log(("EFI_VARIABLE_OP_QUERY_NEXT: pCurVar=%p\n", pThisCC->NVRAM.pCurVar));
659 PEFIVAR pEfiVar = pThisCC->NVRAM.pCurVar;
660 if (pEfiVar)
661 pEfiVar = RTListGetNext(&pThisCC->NVRAM.VarList, pEfiVar, EFIVAR, ListNode);
662 else
663 pEfiVar = RTListGetFirst(&pThisCC->NVRAM.VarList, EFIVAR, ListNode);
664 nvramWriteVariableOpQueryCopyResult(pThisCC, pEfiVar, true /* fEnumQuery */);
665 return VINF_SUCCESS;
666}
667
668/**
669 * Implements EFI_VARIABLE_PARAM + EFI_VARIABLE_OP_ADD.
670 *
671 * @returns IOM strict status code.
672 * @param pThisCC The EFI state for the current context.
673 */
674static int nvramWriteVariableOpAdd(PDEVEFIR3 pThisCC)
675{
676 Log(("EFI_VARIABLE_OP_ADD: %RTuuid::'%s' fAttributes=%#x abValue=%.*Rhxs\n",
677 &pThisCC->NVRAM.VarOpBuf.uuid, pThisCC->NVRAM.VarOpBuf.szName, pThisCC->NVRAM.VarOpBuf.fAttributes,
678 pThisCC->NVRAM.VarOpBuf.cbValue, pThisCC->NVRAM.VarOpBuf.abValue));
679
680 /*
681 * Validate and adjust the input a little before we start.
682 */
683 int rc = RTStrValidateEncoding(pThisCC->NVRAM.VarOpBuf.szName);
684 if (RT_FAILURE(rc))
685 LogRel(("EFI: Badly encoded variable name: %.*Rhxs\n", pThisCC->NVRAM.VarOpBuf.cchName + 1, pThisCC->NVRAM.VarOpBuf.szName));
686 if (RT_FAILURE(rc))
687 {
688 pThisCC->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_ERROR;
689 return VINF_SUCCESS;
690 }
691 pThisCC->NVRAM.VarOpBuf.cchName = (uint32_t)RTStrNLen(pThisCC->NVRAM.VarOpBuf.szName, sizeof(pThisCC->NVRAM.VarOpBuf.szName));
692
693 /*
694 * Look it up and see what to do.
695 */
696 PEFIVAR pEfiVar;
697 rc = nvramLookupVariableByUuidAndName(pThisCC,
698 pThisCC->NVRAM.VarOpBuf.szName,
699 &pThisCC->NVRAM.VarOpBuf.uuid,
700 &pEfiVar);
701 if (RT_SUCCESS(rc))
702 {
703 LogFlowFunc(("Old abValue=%.*Rhxs\n", pEfiVar->cbValue, pEfiVar->abValue));
704#if 0 /** @todo Implement read-only EFI variables. */
705 if (pEfiVar->fAttributes & EFI_VARIABLE_XXXXXXX)
706 {
707 pThisCC->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_RO;
708 break;
709 }
710#endif
711
712 if (pThisCC->NVRAM.VarOpBuf.cbValue == 0)
713 {
714 /*
715 * Delete it.
716 */
717 LogRel(("EFI: Deleting variable %RTuuid::'%s'\n", &pThisCC->NVRAM.VarOpBuf.uuid, pThisCC->NVRAM.VarOpBuf.szName));
718 RTListNodeRemove(&pEfiVar->ListNode);
719 pThisCC->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_OK;
720 pThisCC->NVRAM.cVariables--;
721
722 if (pThisCC->NVRAM.pCurVar == pEfiVar)
723 pThisCC->NVRAM.pCurVar = NULL;
724 RTMemFree(pEfiVar);
725 pEfiVar = NULL;
726 }
727 else
728 {
729 /*
730 * Update/replace it. (The name and UUID are unchanged, of course.)
731 */
732 LogRel(("EFI: Replacing variable %RTuuid::'%s' fAttrib=%#x cbValue=%#x\n", &pThisCC->NVRAM.VarOpBuf.uuid,
733 pThisCC->NVRAM.VarOpBuf.szName, pThisCC->NVRAM.VarOpBuf.fAttributes, pThisCC->NVRAM.VarOpBuf.cbValue));
734 pEfiVar->fAttributes = pThisCC->NVRAM.VarOpBuf.fAttributes;
735 pEfiVar->cbValue = pThisCC->NVRAM.VarOpBuf.cbValue;
736 memcpy(pEfiVar->abValue, pThisCC->NVRAM.VarOpBuf.abValue, pEfiVar->cbValue);
737 pThisCC->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_OK;
738 }
739 }
740 else if (pThisCC->NVRAM.VarOpBuf.cbValue == 0)
741 {
742 /* delete operation, but nothing to delete. */
743 LogFlow(("nvramWriteVariableOpAdd: Delete (not found)\n"));
744 pThisCC->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_OK;
745 }
746 else if (pThisCC->NVRAM.cVariables < EFI_VARIABLE_MAX)
747 {
748 /*
749 * Add a new variable.
750 */
751 LogRel(("EFI: Adding variable %RTuuid::'%s' fAttrib=%#x cbValue=%#x\n", &pThisCC->NVRAM.VarOpBuf.uuid,
752 pThisCC->NVRAM.VarOpBuf.szName, pThisCC->NVRAM.VarOpBuf.fAttributes, pThisCC->NVRAM.VarOpBuf.cbValue));
753 pEfiVar = (PEFIVAR)RTMemAllocZ(sizeof(EFIVAR));
754 if (pEfiVar)
755 {
756 pEfiVar->uuid = pThisCC->NVRAM.VarOpBuf.uuid;
757 pEfiVar->cchName = pThisCC->NVRAM.VarOpBuf.cchName;
758 memcpy(pEfiVar->szName, pThisCC->NVRAM.VarOpBuf.szName, pEfiVar->cchName); /* The buffer is zeroed, so skip '\0'. */
759 pEfiVar->fAttributes = pThisCC->NVRAM.VarOpBuf.fAttributes;
760 pEfiVar->cbValue = pThisCC->NVRAM.VarOpBuf.cbValue;
761 memcpy(pEfiVar->abValue, pThisCC->NVRAM.VarOpBuf.abValue, pEfiVar->cbValue);
762
763 nvramInsertVariable(pThisCC, pEfiVar);
764 pThisCC->NVRAM.cVariables++;
765 pThisCC->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_OK;
766 }
767 else
768 pThisCC->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_ERROR;
769 }
770 else
771 {
772 /*
773 * Too many variables.
774 */
775 LogRelMax(5, ("EFI: Too many variables (%RTuuid::'%s' fAttrib=%#x cbValue=%#x)\n", &pThisCC->NVRAM.VarOpBuf.uuid,
776 pThisCC->NVRAM.VarOpBuf.szName, pThisCC->NVRAM.VarOpBuf.fAttributes, pThisCC->NVRAM.VarOpBuf.cbValue));
777 pThisCC->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_ERROR;
778 Log(("nvramWriteVariableOpAdd: Too many variabled.\n"));
779 }
780
781 /*
782 * Log the value of bugcheck variables.
783 */
784 if ( ( pThisCC->NVRAM.VarOpBuf.cbValue == 4
785 || pThisCC->NVRAM.VarOpBuf.cbValue == 8)
786 && ( strcmp(pThisCC->NVRAM.VarOpBuf.szName, "BugCheckCode") == 0
787 || strcmp(pThisCC->NVRAM.VarOpBuf.szName, "BugCheckParameter0") == 0
788 || strcmp(pThisCC->NVRAM.VarOpBuf.szName, "BugCheckParameter1") == 0
789 || strcmp(pThisCC->NVRAM.VarOpBuf.szName, "BugCheckParameter2") == 0
790 || strcmp(pThisCC->NVRAM.VarOpBuf.szName, "BugCheckParameter3") == 0
791 || strcmp(pThisCC->NVRAM.VarOpBuf.szName, "BugCheckProgress") == 0 ) )
792 {
793 if (pThisCC->NVRAM.VarOpBuf.cbValue == 4)
794 LogRel(("EFI: %RTuuid::'%s' = %#010RX32\n", &pThisCC->NVRAM.VarOpBuf.uuid, pThisCC->NVRAM.VarOpBuf.szName,
795 RT_MAKE_U32_FROM_U8(pThisCC->NVRAM.VarOpBuf.abValue[0], pThisCC->NVRAM.VarOpBuf.abValue[1],
796 pThisCC->NVRAM.VarOpBuf.abValue[2], pThisCC->NVRAM.VarOpBuf.abValue[3])));
797 else
798 LogRel(("EFI: %RTuuid::'%s' = %#018RX64\n", &pThisCC->NVRAM.VarOpBuf.uuid, pThisCC->NVRAM.VarOpBuf.szName,
799 RT_MAKE_U64_FROM_U8(pThisCC->NVRAM.VarOpBuf.abValue[0], pThisCC->NVRAM.VarOpBuf.abValue[1],
800 pThisCC->NVRAM.VarOpBuf.abValue[2], pThisCC->NVRAM.VarOpBuf.abValue[3],
801 pThisCC->NVRAM.VarOpBuf.abValue[4], pThisCC->NVRAM.VarOpBuf.abValue[5],
802 pThisCC->NVRAM.VarOpBuf.abValue[6], pThisCC->NVRAM.VarOpBuf.abValue[7])));
803 }
804
805
806 LogFunc(("cVariables=%u u32Status=%#x\n", pThisCC->NVRAM.cVariables, pThisCC->NVRAM.u32Status));
807 return VINF_SUCCESS;
808}
809
810/**
811 * Implements EFI_VARIABLE_PARAM writes.
812 *
813 * @returns IOM strict status code.
814 * @param pThisCC The EFI state for the current context.
815 * @param u32Value The value being written.
816 */
817static int nvramWriteVariableParam(PDEVEFIR3 pThisCC, uint32_t u32Value)
818{
819 int rc = VINF_SUCCESS;
820 switch (pThisCC->NVRAM.enmOp)
821 {
822 case EFI_VM_VARIABLE_OP_START:
823 switch (u32Value)
824 {
825 case EFI_VARIABLE_OP_QUERY:
826 rc = nvramWriteVariableOpQuery(pThisCC);
827 break;
828
829 case EFI_VARIABLE_OP_QUERY_NEXT:
830 rc = nvramWriteVariableOpQueryNext(pThisCC);
831 break;
832
833 case EFI_VARIABLE_OP_QUERY_REWIND:
834 Log2(("EFI_VARIABLE_OP_QUERY_REWIND\n"));
835 pThisCC->NVRAM.pCurVar = NULL;
836 pThisCC->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_OK;
837 break;
838
839 case EFI_VARIABLE_OP_ADD:
840 rc = nvramWriteVariableOpAdd(pThisCC);
841 break;
842
843 default:
844 pThisCC->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_ERROR;
845 LogRel(("EFI: Unknown EFI_VM_VARIABLE_OP_START value %#x\n", u32Value));
846 break;
847 }
848 break;
849
850 case EFI_VM_VARIABLE_OP_GUID:
851 Log2(("EFI_VM_VARIABLE_OP_GUID[%#x]=%#x\n", pThisCC->NVRAM.offOpBuffer, u32Value));
852 if (pThisCC->NVRAM.offOpBuffer < sizeof(pThisCC->NVRAM.VarOpBuf.uuid))
853 pThisCC->NVRAM.VarOpBuf.uuid.au8[pThisCC->NVRAM.offOpBuffer++] = (uint8_t)u32Value;
854 else
855 {
856 LogRel(("EFI: Out of bounds EFI_VM_VARIABLE_OP_GUID write (%#x).\n", u32Value));
857 pThisCC->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_ERROR;
858 }
859 break;
860
861 case EFI_VM_VARIABLE_OP_ATTRIBUTE:
862 Log2(("EFI_VM_VARIABLE_OP_ATTRIBUTE=%#x\n", u32Value));
863 pThisCC->NVRAM.VarOpBuf.fAttributes = u32Value;
864 break;
865
866 case EFI_VM_VARIABLE_OP_NAME:
867 Log2(("EFI_VM_VARIABLE_OP_NAME[%#x]=%#x\n", pThisCC->NVRAM.offOpBuffer, u32Value));
868 if (pThisCC->NVRAM.offOpBuffer < pThisCC->NVRAM.VarOpBuf.cchName)
869 pThisCC->NVRAM.VarOpBuf.szName[pThisCC->NVRAM.offOpBuffer++] = (uint8_t)u32Value;
870 else if (u32Value == 0)
871 Assert(pThisCC->NVRAM.VarOpBuf.szName[sizeof(pThisCC->NVRAM.VarOpBuf.szName) - 1] == 0);
872 else
873 {
874 LogRel(("EFI: Out of bounds EFI_VM_VARIABLE_OP_NAME write (%#x).\n", u32Value));
875 pThisCC->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_ERROR;
876 }
877 break;
878
879 case EFI_VM_VARIABLE_OP_NAME_LENGTH:
880 Log2(("EFI_VM_VARIABLE_OP_NAME_LENGTH=%#x\n", u32Value));
881 RT_ZERO(pThisCC->NVRAM.VarOpBuf.szName);
882 if (u32Value < sizeof(pThisCC->NVRAM.VarOpBuf.szName))
883 pThisCC->NVRAM.VarOpBuf.cchName = u32Value;
884 else
885 {
886 LogRel(("EFI: Out of bounds EFI_VM_VARIABLE_OP_NAME_LENGTH write (%#x, max %#x).\n",
887 u32Value, sizeof(pThisCC->NVRAM.VarOpBuf.szName) - 1));
888 pThisCC->NVRAM.VarOpBuf.cchName = sizeof(pThisCC->NVRAM.VarOpBuf.szName) - 1;
889 pThisCC->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_ERROR;
890 }
891 Assert(pThisCC->NVRAM.offOpBuffer == 0);
892 break;
893
894 case EFI_VM_VARIABLE_OP_NAME_UTF16:
895 {
896 Log2(("EFI_VM_VARIABLE_OP_NAME_UTF16[%#x]=%#x\n", pThisCC->NVRAM.offOpBuffer, u32Value));
897 /* Currently simplifying this to UCS2, i.e. no surrogates. */
898 if (pThisCC->NVRAM.offOpBuffer == 0)
899 RT_ZERO(pThisCC->NVRAM.VarOpBuf.szName);
900 uint32_t cbUtf8 = (uint32_t)RTStrCpSize(u32Value);
901 if (pThisCC->NVRAM.offOpBuffer + cbUtf8 < sizeof(pThisCC->NVRAM.VarOpBuf.szName))
902 {
903 RTStrPutCp(&pThisCC->NVRAM.VarOpBuf.szName[pThisCC->NVRAM.offOpBuffer], u32Value);
904 pThisCC->NVRAM.offOpBuffer += cbUtf8;
905 }
906 else if (u32Value == 0)
907 Assert(pThisCC->NVRAM.VarOpBuf.szName[sizeof(pThisCC->NVRAM.VarOpBuf.szName) - 1] == 0);
908 else
909 {
910 LogRel(("EFI: Out of bounds EFI_VM_VARIABLE_OP_NAME_UTF16 write (%#x).\n", u32Value));
911 pThisCC->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_ERROR;
912 }
913 break;
914 }
915
916 case EFI_VM_VARIABLE_OP_VALUE:
917 Log2(("EFI_VM_VARIABLE_OP_VALUE[%#x]=%#x\n", pThisCC->NVRAM.offOpBuffer, u32Value));
918 if (pThisCC->NVRAM.offOpBuffer < pThisCC->NVRAM.VarOpBuf.cbValue)
919 pThisCC->NVRAM.VarOpBuf.abValue[pThisCC->NVRAM.offOpBuffer++] = (uint8_t)u32Value;
920 else
921 {
922 LogRel(("EFI: Out of bounds EFI_VM_VARIABLE_OP_VALUE write (%#x).\n", u32Value));
923 pThisCC->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_ERROR;
924 }
925 break;
926
927 case EFI_VM_VARIABLE_OP_VALUE_LENGTH:
928 Log2(("EFI_VM_VARIABLE_OP_VALUE_LENGTH=%#x\n", u32Value));
929 RT_ZERO(pThisCC->NVRAM.VarOpBuf.abValue);
930 if (u32Value <= sizeof(pThisCC->NVRAM.VarOpBuf.abValue))
931 pThisCC->NVRAM.VarOpBuf.cbValue = u32Value;
932 else
933 {
934 LogRel(("EFI: Out of bounds EFI_VM_VARIABLE_OP_VALUE_LENGTH write (%#x, max %#x).\n",
935 u32Value, sizeof(pThisCC->NVRAM.VarOpBuf.abValue)));
936 pThisCC->NVRAM.VarOpBuf.cbValue = sizeof(pThisCC->NVRAM.VarOpBuf.abValue);
937 pThisCC->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_ERROR;
938 }
939 Assert(pThisCC->NVRAM.offOpBuffer == 0);
940 break;
941
942 default:
943 pThisCC->NVRAM.u32Status = EFI_VARIABLE_OP_STATUS_ERROR;
944 LogRel(("EFI: Unexpected variable operation %#x\n", pThisCC->NVRAM.enmOp));
945 break;
946 }
947 return VINF_SUCCESS;
948}
949
950/**
951 * Implements EFI_VARIABLE_OP reads.
952 *
953 * @returns IOM strict status code.
954 * @param pThisCC The EFI state for the current context.
955 * @param u32Value The value being written.
956 */
957static int nvramReadVariableOp(PDEVEFIR3 pThisCC, uint32_t *pu32, unsigned cb)
958{
959 switch (pThisCC->NVRAM.enmOp)
960 {
961 case EFI_VM_VARIABLE_OP_START:
962 *pu32 = pThisCC->NVRAM.u32Status;
963 break;
964
965 case EFI_VM_VARIABLE_OP_GUID:
966 if (pThisCC->NVRAM.offOpBuffer < sizeof(pThisCC->NVRAM.VarOpBuf.uuid) && cb == 1)
967 *pu32 = pThisCC->NVRAM.VarOpBuf.uuid.au8[pThisCC->NVRAM.offOpBuffer++];
968 else
969 {
970 if (cb == 1)
971 LogRel(("EFI: Out of bounds EFI_VM_VARIABLE_OP_GUID read.\n"));
972 else
973 LogRel(("EFI: Invalid EFI_VM_VARIABLE_OP_GUID read size (%d).\n", cb));
974 *pu32 = UINT32_MAX;
975 }
976 break;
977
978 case EFI_VM_VARIABLE_OP_ATTRIBUTE:
979 *pu32 = pThisCC->NVRAM.VarOpBuf.fAttributes;
980 break;
981
982 case EFI_VM_VARIABLE_OP_NAME:
983 /* allow reading terminator char */
984 if (pThisCC->NVRAM.offOpBuffer <= pThisCC->NVRAM.VarOpBuf.cchName && cb == 1)
985 *pu32 = pThisCC->NVRAM.VarOpBuf.szName[pThisCC->NVRAM.offOpBuffer++];
986 else
987 {
988 if (cb == 1)
989 LogRel(("EFI: Out of bounds EFI_VM_VARIABLE_OP_NAME read.\n"));
990 else
991 LogRel(("EFI: Invalid EFI_VM_VARIABLE_OP_NAME read size (%d).\n", cb));
992 *pu32 = UINT32_MAX;
993 }
994 break;
995
996 case EFI_VM_VARIABLE_OP_NAME_LENGTH:
997 *pu32 = pThisCC->NVRAM.VarOpBuf.cchName;
998 break;
999
1000 case EFI_VM_VARIABLE_OP_NAME_UTF16:
1001 /* Lazy bird: ASSUME no surrogate pairs. */
1002 if (pThisCC->NVRAM.offOpBuffer <= pThisCC->NVRAM.VarOpBuf.cchName && cb == 2)
1003 {
1004 char const *psz1 = &pThisCC->NVRAM.VarOpBuf.szName[pThisCC->NVRAM.offOpBuffer];
1005 char const *psz2 = psz1;
1006 RTUNICP Cp;
1007 RTStrGetCpEx(&psz2, &Cp);
1008 *pu32 = Cp;
1009 Log2(("EFI_VM_VARIABLE_OP_NAME_UTF16[%u] => %#x (+%d)\n", pThisCC->NVRAM.offOpBuffer, *pu32, psz2 - psz1));
1010 pThisCC->NVRAM.offOpBuffer += psz2 - psz1;
1011 }
1012 else
1013 {
1014 if (cb == 2)
1015 LogRel(("EFI: Out of bounds EFI_VM_VARIABLE_OP_NAME_UTF16 read.\n"));
1016 else
1017 LogRel(("EFI: Invalid EFI_VM_VARIABLE_OP_NAME_UTF16 read size (%d).\n", cb));
1018 *pu32 = UINT32_MAX;
1019 }
1020 break;
1021
1022 case EFI_VM_VARIABLE_OP_NAME_LENGTH_UTF16:
1023 /* Lazy bird: ASSUME no surrogate pairs. */
1024 *pu32 = (uint32_t)RTStrUniLen(pThisCC->NVRAM.VarOpBuf.szName);
1025 break;
1026
1027 case EFI_VM_VARIABLE_OP_VALUE:
1028 if (pThisCC->NVRAM.offOpBuffer < pThisCC->NVRAM.VarOpBuf.cbValue && cb == 1)
1029 *pu32 = pThisCC->NVRAM.VarOpBuf.abValue[pThisCC->NVRAM.offOpBuffer++];
1030 else
1031 {
1032 if (cb == 1)
1033 LogRel(("EFI: Out of bounds EFI_VM_VARIABLE_OP_VALUE read.\n"));
1034 else
1035 LogRel(("EFI: Invalid EFI_VM_VARIABLE_OP_VALUE read size (%d).\n", cb));
1036 *pu32 = UINT32_MAX;
1037 }
1038 break;
1039
1040 case EFI_VM_VARIABLE_OP_VALUE_LENGTH:
1041 *pu32 = pThisCC->NVRAM.VarOpBuf.cbValue;
1042 break;
1043
1044 default:
1045 *pu32 = UINT32_MAX;
1046 break;
1047 }
1048 return VINF_SUCCESS;
1049}
1050
1051
1052/**
1053 * Checks if the EFI variable value looks like a printable UTF-8 string.
1054 *
1055 * @returns true if it is, false if not.
1056 * @param pEfiVar The variable.
1057 * @param pfZeroTerm Where to return whether the string is zero
1058 * terminated.
1059 */
1060static bool efiInfoNvramIsUtf8(PCEFIVAR pEfiVar, bool *pfZeroTerm)
1061{
1062 if (pEfiVar->cbValue < 2)
1063 return false;
1064 const char *pachValue = (const char *)&pEfiVar->abValue[0];
1065 *pfZeroTerm = pachValue[pEfiVar->cbValue - 1] == 0;
1066
1067 /* Check the length. */
1068 size_t cchValue = RTStrNLen((const char *)pEfiVar->abValue, pEfiVar->cbValue);
1069 if (cchValue != pEfiVar->cbValue - *pfZeroTerm)
1070 return false; /* stray zeros in the value, forget it. */
1071
1072 /* Check that the string is valid UTF-8 and printable. */
1073 const char *pchCur = pachValue;
1074 while ((uintptr_t)(pchCur - pachValue) < cchValue)
1075 {
1076 RTUNICP uc;
1077 int rc = RTStrGetCpEx(&pachValue, &uc);
1078 if (RT_FAILURE(rc))
1079 return false;
1080 /** @todo Missing RTUniCpIsPrintable. */
1081 if (uc < 128 && !RT_C_IS_PRINT(uc))
1082 return false;
1083 }
1084
1085 return true;
1086}
1087
1088
1089/**
1090 * Checks if the EFI variable value looks like a printable UTF-16 string.
1091 *
1092 * @returns true if it is, false if not.
1093 * @param pEfiVar The variable.
1094 * @param pfZeroTerm Where to return whether the string is zero
1095 * terminated.
1096 */
1097static bool efiInfoNvramIsUtf16(PCEFIVAR pEfiVar, bool *pfZeroTerm)
1098{
1099 if (pEfiVar->cbValue < 4 || (pEfiVar->cbValue & 1))
1100 return false;
1101
1102 PCRTUTF16 pwcValue = (PCRTUTF16)&pEfiVar->abValue[0];
1103 size_t cwcValue = pEfiVar->cbValue / sizeof(RTUTF16);
1104 *pfZeroTerm = pwcValue[cwcValue - 1] == 0;
1105 if (!*pfZeroTerm && RTUtf16IsHighSurrogate(pwcValue[cwcValue - 1]))
1106 return false; /* Catch bad string early, before reading a char too many. */
1107 cwcValue -= *pfZeroTerm;
1108 if (cwcValue < 2)
1109 return false;
1110
1111 /* Check that the string is valid UTF-16, printable and spans the whole
1112 value length. */
1113 size_t cAscii = 0;
1114 PCRTUTF16 pwcCur = pwcValue;
1115 while ((uintptr_t)(pwcCur - pwcValue) < cwcValue)
1116 {
1117 RTUNICP uc;
1118 int rc = RTUtf16GetCpEx(&pwcCur, &uc);
1119 if (RT_FAILURE(rc))
1120 return false;
1121 /** @todo Missing RTUniCpIsPrintable. */
1122 if (uc < 128 && !RT_C_IS_PRINT(uc))
1123 return false;
1124 cAscii += uc < 128;
1125 }
1126 if (cAscii < 2)
1127 return false;
1128
1129 return true;
1130}
1131
1132
1133/**
1134 * @implement_callback_method{FNDBGFHANDLERDEV}
1135 */
1136static DECLCALLBACK(void) efiInfoNvram(PPDMDEVINS pDevIns, PCDBGFINFOHLP pHlp, const char *pszArgs)
1137{
1138 RT_NOREF(pszArgs);
1139 PDEVEFIR3 pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PDEVEFIR3);
1140 PDMDevHlpCritSectEnter(pDevIns, pDevIns->pCritSectRoR3, VERR_IGNORED);
1141
1142 pHlp->pfnPrintf(pHlp, "NVRAM variables: %u\n", pThisCC->NVRAM.cVariables);
1143 PCEFIVAR pEfiVar;
1144 RTListForEach(&pThisCC->NVRAM.VarList, pEfiVar, EFIVAR, ListNode)
1145 {
1146 /* Detect UTF-8 and UTF-16 strings. */
1147 bool fZeroTerm = false;
1148 if (efiInfoNvramIsUtf8(pEfiVar, &fZeroTerm))
1149 pHlp->pfnPrintf(pHlp,
1150 "Variable - fAttr=%#04x - '%RTuuid:%s' - cb=%#04x\n"
1151 "String value (UTF-8%s): \"%.*s\"\n",
1152 pEfiVar->fAttributes, &pEfiVar->uuid, pEfiVar->szName, pEfiVar->cbValue,
1153 fZeroTerm ? "" : ",nz", pEfiVar->cbValue, pEfiVar->abValue);
1154 else if (efiInfoNvramIsUtf16(pEfiVar, &fZeroTerm))
1155 pHlp->pfnPrintf(pHlp,
1156 "Variable - fAttr=%#04x - '%RTuuid:%s' - cb=%#04x\n"
1157 "String value (UTF-16%s): \"%.*ls\"\n",
1158 pEfiVar->fAttributes, &pEfiVar->uuid, pEfiVar->szName, pEfiVar->cbValue,
1159 fZeroTerm ? "" : ",nz", pEfiVar->cbValue, pEfiVar->abValue);
1160 else
1161 pHlp->pfnPrintf(pHlp,
1162 "Variable - fAttr=%#04x - '%RTuuid:%s' - cb=%#04x\n"
1163 "%.*Rhxd\n",
1164 pEfiVar->fAttributes, &pEfiVar->uuid, pEfiVar->szName, pEfiVar->cbValue,
1165 pEfiVar->cbValue, pEfiVar->abValue);
1166
1167 }
1168
1169 PDMDevHlpCritSectLeave(pDevIns, pDevIns->pCritSectRoR3);
1170}
1171
1172
1173
1174/**
1175 * Gets the info item size.
1176 *
1177 * @returns Size in bytes, UINT32_MAX on error.
1178 * @param pThisCC The EFI state for the current context.
1179 */
1180static uint32_t efiInfoSize(PDEVEFIR3 pThisCC)
1181{
1182 switch (pThisCC->iInfoSelector)
1183 {
1184 case EFI_INFO_INDEX_VOLUME_BASE:
1185 case EFI_INFO_INDEX_VOLUME_SIZE:
1186 case EFI_INFO_INDEX_TEMPMEM_BASE:
1187 case EFI_INFO_INDEX_TEMPMEM_SIZE:
1188 case EFI_INFO_INDEX_STACK_BASE:
1189 case EFI_INFO_INDEX_STACK_SIZE:
1190 case EFI_INFO_INDEX_GRAPHICS_MODE:
1191 case EFI_INFO_INDEX_VERTICAL_RESOLUTION:
1192 case EFI_INFO_INDEX_HORIZONTAL_RESOLUTION:
1193 return 4;
1194 case EFI_INFO_INDEX_BOOT_ARGS:
1195 return (uint32_t)RTStrNLen(pThisCC->szBootArgs, sizeof(pThisCC->szBootArgs)) + 1;
1196 case EFI_INFO_INDEX_DEVICE_PROPS:
1197 return pThisCC->cbDeviceProps;
1198 case EFI_INFO_INDEX_FSB_FREQUENCY:
1199 case EFI_INFO_INDEX_CPU_FREQUENCY:
1200 case EFI_INFO_INDEX_TSC_FREQUENCY:
1201 case EFI_INFO_INDEX_MCFG_BASE:
1202 case EFI_INFO_INDEX_MCFG_SIZE:
1203 return 8;
1204 case EFI_INFO_INDEX_APIC_MODE:
1205 return 1;
1206 }
1207 return UINT32_MAX;
1208}
1209
1210
1211/**
1212 * efiInfoNextByte for a uint8_t value.
1213 *
1214 * @returns Next (current) byte.
1215 * @param pThisCC The EFI state for the current context.
1216 * @param u8 The value.
1217 */
1218static uint8_t efiInfoNextByteU8(PDEVEFIR3 pThisCC, uint8_t u8)
1219{
1220 uint32_t off = pThisCC->offInfo;
1221 if (off >= 1)
1222 return 0;
1223 return (uint8_t)u8;
1224}
1225
1226
1227/**
1228 * efiInfoNextByte for a uint64_t value.
1229 *
1230 * @returns Next (current) byte.
1231 * @param pThisCC The EFI state for the current context.
1232 * @param u64 The value.
1233 */
1234static uint8_t efiInfoNextByteU64(PDEVEFIR3 pThisCC, uint64_t u64)
1235{
1236 uint64_t off = pThisCC->offInfo;
1237 if (off >= 8)
1238 return 0;
1239 return (uint8_t)(u64 >> (off * 8));
1240}
1241
1242/**
1243 * efiInfoNextByte for a uint32_t value.
1244 *
1245 * @returns Next (current) byte.
1246 * @param pThisCC The EFI state for the current context.
1247 * @param u32 The value.
1248 */
1249static uint8_t efiInfoNextByteU32(PDEVEFIR3 pThisCC, uint32_t u32)
1250{
1251 uint32_t off = pThisCC->offInfo;
1252 if (off >= 4)
1253 return 0;
1254 return (uint8_t)(u32 >> (off * 8));
1255}
1256
1257/**
1258 * efiInfoNextByte for a buffer.
1259 *
1260 * @returns Next (current) byte.
1261 * @param pThisCC The EFI state for the current context.
1262 * @param pvBuf The buffer.
1263 * @param cbBuf The buffer size.
1264 */
1265static uint8_t efiInfoNextByteBuf(PDEVEFIR3 pThisCC, void const *pvBuf, size_t cbBuf)
1266{
1267 uint32_t off = pThisCC->offInfo;
1268 if (off >= cbBuf)
1269 return 0;
1270 return ((uint8_t const *)pvBuf)[off];
1271}
1272
1273/**
1274 * Gets the next info byte.
1275 *
1276 * @returns Next (current) byte.
1277 * @param pThisCC The EFI state for the current context.
1278 */
1279static uint8_t efiInfoNextByte(PDEVEFIR3 pThisCC)
1280{
1281 switch (pThisCC->iInfoSelector)
1282 {
1283
1284 case EFI_INFO_INDEX_VOLUME_BASE: return efiInfoNextByteU64(pThisCC, pThisCC->GCLoadAddress);
1285 case EFI_INFO_INDEX_VOLUME_SIZE: return efiInfoNextByteU64(pThisCC, pThisCC->cbEfiRom);
1286 case EFI_INFO_INDEX_TEMPMEM_BASE: return efiInfoNextByteU32(pThisCC, VBOX_EFI_TOP_OF_STACK); /* just after stack */
1287 case EFI_INFO_INDEX_TEMPMEM_SIZE: return efiInfoNextByteU32(pThisCC, _512K);
1288 case EFI_INFO_INDEX_FSB_FREQUENCY: return efiInfoNextByteU64(pThisCC, pThisCC->u64FsbFrequency);
1289 case EFI_INFO_INDEX_TSC_FREQUENCY: return efiInfoNextByteU64(pThisCC, pThisCC->u64TscFrequency);
1290 case EFI_INFO_INDEX_CPU_FREQUENCY: return efiInfoNextByteU64(pThisCC, pThisCC->u64CpuFrequency);
1291 case EFI_INFO_INDEX_BOOT_ARGS: return efiInfoNextByteBuf(pThisCC, pThisCC->szBootArgs, sizeof(pThisCC->szBootArgs));
1292 case EFI_INFO_INDEX_DEVICE_PROPS: return efiInfoNextByteBuf(pThisCC, pThisCC->pbDeviceProps, pThisCC->cbDeviceProps);
1293 case EFI_INFO_INDEX_GRAPHICS_MODE: return efiInfoNextByteU32(pThisCC, pThisCC->u32GraphicsMode);
1294 case EFI_INFO_INDEX_HORIZONTAL_RESOLUTION: return efiInfoNextByteU32(pThisCC, pThisCC->u32HorizontalResolution);
1295 case EFI_INFO_INDEX_VERTICAL_RESOLUTION: return efiInfoNextByteU32(pThisCC, pThisCC->u32VerticalResolution);
1296
1297 /* Keep in sync with value in EfiThunk.asm */
1298 case EFI_INFO_INDEX_STACK_BASE: return efiInfoNextByteU32(pThisCC, VBOX_EFI_TOP_OF_STACK - _128K); /* 2M - 128 K */
1299 case EFI_INFO_INDEX_STACK_SIZE: return efiInfoNextByteU32(pThisCC, _128K);
1300 case EFI_INFO_INDEX_MCFG_BASE: return efiInfoNextByteU64(pThisCC, pThisCC->u64McfgBase);
1301 case EFI_INFO_INDEX_MCFG_SIZE: return efiInfoNextByteU64(pThisCC, pThisCC->cbMcfgLength);
1302 case EFI_INFO_INDEX_APIC_MODE: return efiInfoNextByteU8(pThisCC, pThisCC->u8APIC);
1303
1304 default:
1305 PDMDevHlpDBGFStop(pThisCC->pDevIns, RT_SRC_POS, "%#x", pThisCC->iInfoSelector);
1306 return 0;
1307 }
1308}
1309
1310
1311#ifdef IN_RING3
1312static void efiVBoxDbgScript(const char *pszFormat, ...)
1313{
1314# ifdef DEVEFI_WITH_VBOXDBG_SCRIPT
1315 PRTSTREAM pStrm;
1316 int rc2 = RTStrmOpen("./DevEFI.VBoxDbg", "a", &pStrm);
1317 if (RT_SUCCESS(rc2))
1318 {
1319 va_list va;
1320 va_start(va, pszFormat);
1321 RTStrmPrintfV(pStrm, pszFormat, va);
1322 va_end(va);
1323 RTStrmClose(pStrm);
1324 }
1325# else
1326 RT_NOREF(pszFormat);
1327# endif
1328}
1329#endif /* IN_RING3 */
1330
1331
1332/**
1333 * Handles writes to the image event port.
1334 *
1335 * @returns VBox status suitable for I/O port write handler.
1336 *
1337 * @param pThisCC The EFI state for the current context.
1338 * @param u32 The value being written.
1339 * @param cb The size of the value.
1340 */
1341static int efiPortImageEventWrite(PDEVEFIR3 pThisCC, uint32_t u32, unsigned cb)
1342{
1343 RT_NOREF(cb);
1344 switch (u32 & EFI_IMAGE_EVT_CMD_MASK)
1345 {
1346 case EFI_IMAGE_EVT_CMD_START_LOAD32:
1347 case EFI_IMAGE_EVT_CMD_START_LOAD64:
1348 case EFI_IMAGE_EVT_CMD_START_UNLOAD32:
1349 case EFI_IMAGE_EVT_CMD_START_UNLOAD64:
1350 case EFI_IMAGE_EVT_CMD_START_RELOC32:
1351 case EFI_IMAGE_EVT_CMD_START_RELOC64:
1352 AssertBreak(EFI_IMAGE_EVT_GET_PAYLOAD(u32) == 0);
1353
1354 /* Reset the state. */
1355 RT_ZERO(pThisCC->ImageEvt);
1356 pThisCC->ImageEvt.uEvt = (uint8_t)u32; Assert(pThisCC->ImageEvt.uEvt == u32);
1357 return VINF_SUCCESS;
1358
1359 case EFI_IMAGE_EVT_CMD_COMPLETE:
1360 {
1361# ifdef IN_RING3
1362 AssertBreak(EFI_IMAGE_EVT_GET_PAYLOAD(u32) == 0);
1363
1364 /* For now, just log it. */
1365 static uint64_t s_cImageEvtLogged = 0;
1366 if (s_cImageEvtLogged < 2048)
1367 {
1368 s_cImageEvtLogged++;
1369 switch (pThisCC->ImageEvt.uEvt)
1370 {
1371 /* ASSUMES the name ends with .pdb and the image file ends with .efi! */
1372 case EFI_IMAGE_EVT_CMD_START_LOAD32:
1373 LogRel(("EFI: VBoxDbg> loadimage32 '%.*s.efi' %#llx LB %#llx\n",
1374 pThisCC->ImageEvt.offName - 4, pThisCC->ImageEvt.szName, pThisCC->ImageEvt.uAddr0, pThisCC->ImageEvt.cb0));
1375 if (pThisCC->ImageEvt.offName > 4)
1376 efiVBoxDbgScript("loadimage32 '%.*s.efi' %#llx\n",
1377 pThisCC->ImageEvt.offName - 4, pThisCC->ImageEvt.szName, pThisCC->ImageEvt.uAddr0);
1378 break;
1379 case EFI_IMAGE_EVT_CMD_START_LOAD64:
1380 LogRel(("EFI: VBoxDbg> loadimage64 '%.*s.efi' %#llx LB %#llx\n",
1381 pThisCC->ImageEvt.offName - 4, pThisCC->ImageEvt.szName, pThisCC->ImageEvt.uAddr0, pThisCC->ImageEvt.cb0));
1382 if (pThisCC->ImageEvt.offName > 4)
1383 efiVBoxDbgScript("loadimage64 '%.*s.efi' %#llx\n",
1384 pThisCC->ImageEvt.offName - 4, pThisCC->ImageEvt.szName, pThisCC->ImageEvt.uAddr0);
1385 break;
1386 case EFI_IMAGE_EVT_CMD_START_UNLOAD32:
1387 case EFI_IMAGE_EVT_CMD_START_UNLOAD64:
1388 {
1389 LogRel(("EFI: VBoxDbg> unload '%.*s.efi' # %#llx LB %#llx\n",
1390 pThisCC->ImageEvt.offName - 4 - pThisCC->ImageEvt.offNameLastComponent,
1391 &pThisCC->ImageEvt.szName[pThisCC->ImageEvt.offNameLastComponent],
1392 pThisCC->ImageEvt.uAddr0, pThisCC->ImageEvt.cb0));
1393 if (pThisCC->ImageEvt.offName > 4)
1394 efiVBoxDbgScript("unload '%.*s.efi'\n",
1395 pThisCC->ImageEvt.offName - 4 - pThisCC->ImageEvt.offNameLastComponent,
1396 &pThisCC->ImageEvt.szName[pThisCC->ImageEvt.offNameLastComponent]);
1397 break;
1398 }
1399 case EFI_IMAGE_EVT_CMD_START_RELOC32:
1400 case EFI_IMAGE_EVT_CMD_START_RELOC64:
1401 {
1402 LogRel(("EFI: relocate module to %#llx from %#llx\n",
1403 pThisCC->ImageEvt.uAddr0, pThisCC->ImageEvt.uAddr1));
1404 break;
1405 }
1406 }
1407 }
1408 return VINF_SUCCESS;
1409# else
1410 return VINF_IOM_R3_IOPORT_WRITE;
1411# endif
1412 }
1413
1414 case EFI_IMAGE_EVT_CMD_ADDR0:
1415 AssertBreak(EFI_IMAGE_EVT_GET_PAYLOAD(u32) <= UINT16_MAX);
1416 pThisCC->ImageEvt.uAddr0 <<= 16;
1417 pThisCC->ImageEvt.uAddr0 |= EFI_IMAGE_EVT_GET_PAYLOAD_U16(u32);
1418 return VINF_SUCCESS;
1419
1420 case EFI_IMAGE_EVT_CMD_ADDR1:
1421 AssertBreak(EFI_IMAGE_EVT_GET_PAYLOAD(u32) <= UINT16_MAX);
1422 pThisCC->ImageEvt.uAddr1 <<= 16;
1423 pThisCC->ImageEvt.uAddr1 |= EFI_IMAGE_EVT_GET_PAYLOAD_U16(u32);
1424 return VINF_SUCCESS;
1425
1426 case EFI_IMAGE_EVT_CMD_SIZE0:
1427 AssertBreak(EFI_IMAGE_EVT_GET_PAYLOAD(u32) <= UINT16_MAX);
1428 pThisCC->ImageEvt.cb0 <<= 16;
1429 pThisCC->ImageEvt.cb0 |= EFI_IMAGE_EVT_GET_PAYLOAD_U16(u32);
1430 return VINF_SUCCESS;
1431
1432 case EFI_IMAGE_EVT_CMD_NAME:
1433 AssertBreak(EFI_IMAGE_EVT_GET_PAYLOAD(u32) <= 0x7f);
1434 if (pThisCC->ImageEvt.offName < sizeof(pThisCC->ImageEvt.szName) - 1)
1435 {
1436 char ch = EFI_IMAGE_EVT_GET_PAYLOAD_U8(u32);
1437 if (ch == '\\')
1438 ch = '/';
1439 pThisCC->ImageEvt.szName[pThisCC->ImageEvt.offName++] = ch;
1440 if (ch == '/' || ch == ':')
1441 pThisCC->ImageEvt.offNameLastComponent = pThisCC->ImageEvt.offName;
1442 }
1443 else
1444 Log(("EFI: Image name overflow\n"));
1445 return VINF_SUCCESS;
1446 }
1447
1448 Log(("EFI: Unknown image event: %#x (cb=%d)\n", u32, cb));
1449 return VINF_SUCCESS;
1450}
1451
1452
1453/**
1454 * @callback_method_impl{FNIOMIOPORTNEWIN}
1455 *
1456 * @note The @a offPort parameter is absolute!
1457 */
1458static DECLCALLBACK(VBOXSTRICTRC) efiR3IoPortRead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT offPort, uint32_t *pu32, unsigned cb)
1459{
1460 RT_NOREF(pvUser);
1461 PDEVEFIR3 pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PDEVEFIR3);
1462 Log4(("EFI in: %x %x\n", offPort, cb));
1463
1464 switch (offPort)
1465 {
1466 case EFI_INFO_PORT:
1467 if (pThisCC->offInfo == -1 && cb == 4)
1468 {
1469 pThisCC->offInfo = 0;
1470 uint32_t cbInfo = *pu32 = efiInfoSize(pThisCC);
1471 if (cbInfo == UINT32_MAX)
1472 return PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS, "iInfoSelector=%#x (%d)\n",
1473 pThisCC->iInfoSelector, pThisCC->iInfoSelector);
1474 }
1475 else
1476 {
1477 if (cb != 1)
1478 return VERR_IOM_IOPORT_UNUSED;
1479 *pu32 = efiInfoNextByte(pThisCC);
1480 pThisCC->offInfo++;
1481 }
1482 return VINF_SUCCESS;
1483
1484 case EFI_PANIC_PORT:
1485# ifdef IN_RING3
1486 LogRel(("EFI panic port read!\n"));
1487 /* Insert special code here on panic reads */
1488 return PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS, "EFI Panic: panic port read!\n");
1489# else
1490 /* Reschedule to R3 */
1491 return VINF_IOM_R3_IOPORT_READ;
1492# endif
1493
1494 case EFI_PORT_VARIABLE_OP:
1495 return nvramReadVariableOp(pThisCC, pu32, cb);
1496
1497 case EFI_PORT_VARIABLE_PARAM:
1498 case EFI_PORT_DEBUG_POINT:
1499 case EFI_PORT_IMAGE_EVENT:
1500 *pu32 = UINT32_MAX;
1501 return VINF_SUCCESS;
1502 }
1503
1504 return VERR_IOM_IOPORT_UNUSED;
1505}
1506
1507
1508/**
1509 * Translates a debug point value into a string for logging.
1510 *
1511 * @returns read-only string
1512 * @param enmDbgPoint Valid debug point value.
1513 */
1514static const char *efiDbgPointName(EFIDBGPOINT enmDbgPoint)
1515{
1516 switch (enmDbgPoint)
1517 {
1518 case EFIDBGPOINT_SEC_PREMEM: return "SEC_PREMEM";
1519 case EFIDBGPOINT_SEC_POSTMEM: return "SEC_POSTMEM";
1520 case EFIDBGPOINT_DXE_CORE: return "DXE_CORE";
1521 case EFIDBGPOINT_SMM: return "SMM";
1522 case EFIDBGPOINT_SMI_ENTER: return "SMI_ENTER";
1523 case EFIDBGPOINT_SMI_EXIT: return "SMI_EXIT";
1524 case EFIDBGPOINT_GRAPHICS: return "GRAPHICS";
1525 case EFIDBGPOINT_DXE_AP: return "DXE_AP";
1526 default:
1527 AssertFailed();
1528 return "Unknown";
1529 }
1530}
1531
1532
1533/**
1534 * @callback_method_impl{FNIOMIOPORTNEWOUT}
1535 *
1536 * @note The @a offPort parameter is absolute!
1537 */
1538static DECLCALLBACK(VBOXSTRICTRC) efiR3IoPortWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT offPort, uint32_t u32, unsigned cb)
1539{
1540 RT_NOREF(pvUser);
1541 PDEVEFIR3 pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PDEVEFIR3);
1542 VBOXSTRICTRC rc = VINF_SUCCESS;
1543 Log4(("efi: out %x %x %d\n", offPort, u32, cb));
1544
1545 switch (offPort)
1546 {
1547 case EFI_INFO_PORT:
1548 Log2(("EFI_INFO_PORT: iInfoSelector=%#x\n", u32));
1549 pThisCC->iInfoSelector = u32;
1550 pThisCC->offInfo = -1;
1551 break;
1552
1553 case EFI_DEBUG_PORT:
1554 {
1555 /* The raw version. */
1556 switch (u32)
1557 {
1558 case '\r': Log3(("efi: <return>\n")); break;
1559 case '\n': Log3(("efi: <newline>\n")); break;
1560 case '\t': Log3(("efi: <tab>\n")); break;
1561 default: Log3(("efi: %c (%02x)\n", u32, u32)); break;
1562 }
1563 /* The readable, buffered version. */
1564 if (u32 == '\n' || u32 == '\r')
1565 {
1566 Assert(pThisCC->iMsg < sizeof(pThisCC->szMsg));
1567 pThisCC->szMsg[pThisCC->iMsg] = '\0';
1568 if (pThisCC->iMsg)
1569 LogRel2(("efi: %s\n", pThisCC->szMsg));
1570 pThisCC->iMsg = 0;
1571 }
1572 else
1573 {
1574 if (pThisCC->iMsg >= sizeof(pThisCC->szMsg) - 1)
1575 {
1576 pThisCC->szMsg[pThisCC->iMsg] = '\0';
1577 LogRel2(("efi: %s\n", pThisCC->szMsg));
1578 pThisCC->iMsg = 0;
1579 }
1580 pThisCC->szMsg[pThisCC->iMsg] = (char)u32;
1581 pThisCC->szMsg[++pThisCC->iMsg] = '\0';
1582 }
1583 break;
1584 }
1585
1586 case EFI_PANIC_PORT:
1587 {
1588 switch (u32)
1589 {
1590 case EFI_PANIC_CMD_BAD_ORG: /* Legacy */
1591 case EFI_PANIC_CMD_THUNK_TRAP:
1592#ifdef IN_RING3
1593 LogRel(("EFI: Panic! Unexpected trap!!\n"));
1594# ifdef VBOX_STRICT
1595 return PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS, "EFI Panic: Unexpected trap during early bootstrap!\n");
1596# else
1597 AssertReleaseMsgFailed(("Unexpected trap during early EFI bootstrap!!\n"));
1598# endif
1599 break;
1600#else
1601 return VINF_IOM_R3_IOPORT_WRITE;
1602#endif
1603
1604 case EFI_PANIC_CMD_START_MSG:
1605 LogRel(("Receiving EFI panic...\n"));
1606 pThisCC->iPanicMsg = 0;
1607 pThisCC->szPanicMsg[0] = '\0';
1608 break;
1609
1610 case EFI_PANIC_CMD_END_MSG:
1611#ifdef IN_RING3
1612 LogRel(("EFI: Panic! %s\n", pThisCC->szPanicMsg));
1613# ifdef VBOX_STRICT
1614 return PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS, "EFI Panic: %s\n", pThisCC->szPanicMsg);
1615# else
1616 return VERR_INTERNAL_ERROR;
1617# endif
1618#else
1619 return VINF_IOM_R3_IOPORT_WRITE;
1620#endif
1621
1622
1623 default:
1624 if ( u32 >= EFI_PANIC_CMD_MSG_FIRST
1625 && u32 <= EFI_PANIC_CMD_MSG_LAST)
1626 {
1627 /* Add the message char to the buffer. */
1628 uint32_t i = pThisCC->iPanicMsg;
1629 if (i + 1 < sizeof(pThisCC->szPanicMsg))
1630 {
1631 char ch = EFI_PANIC_CMD_MSG_GET_CHAR(u32);
1632 if ( ch == '\n'
1633 && i > 0
1634 && pThisCC->szPanicMsg[i - 1] == '\r')
1635 i--;
1636 pThisCC->szPanicMsg[i] = ch;
1637 pThisCC->szPanicMsg[i + 1] = '\0';
1638 pThisCC->iPanicMsg = i + 1;
1639 }
1640 }
1641 else
1642 Log(("EFI: Unknown panic command: %#x (cb=%d)\n", u32, cb));
1643 break;
1644 }
1645 break;
1646 }
1647
1648 case EFI_PORT_VARIABLE_OP:
1649 {
1650 /* clear buffer index */
1651 if (u32 >= (uint32_t)EFI_VM_VARIABLE_OP_MAX)
1652 {
1653 Log(("EFI: Invalid variable op %#x\n", u32));
1654 u32 = EFI_VM_VARIABLE_OP_ERROR;
1655 }
1656 pThisCC->NVRAM.offOpBuffer = 0;
1657 pThisCC->NVRAM.enmOp = (EFIVAROP)u32;
1658 Log2(("EFI_VARIABLE_OP: enmOp=%#x (%d)\n", u32, u32));
1659 break;
1660 }
1661
1662 case EFI_PORT_VARIABLE_PARAM:
1663 rc = nvramWriteVariableParam(pThisCC, u32);
1664 break;
1665
1666 case EFI_PORT_DEBUG_POINT:
1667# ifdef IN_RING3
1668 if (u32 > EFIDBGPOINT_INVALID && u32 < EFIDBGPOINT_END)
1669 {
1670 /* For now, just log it. */
1671 LogRelMax(1024, ("EFI: debug point %s\n", efiDbgPointName((EFIDBGPOINT)u32)));
1672 rc = VINF_SUCCESS;
1673 }
1674 else
1675 rc = PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS, "Invalid debug point %#x\n", u32);
1676 break;
1677# else
1678 return VINF_IOM_R3_IOPORT_WRITE;
1679# endif
1680
1681 case EFI_PORT_IMAGE_EVENT:
1682 rc = efiPortImageEventWrite(pThisCC, u32, cb);
1683 break;
1684
1685 default:
1686 Log(("EFI: Write to reserved port %RTiop: %#x (cb=%d)\n", offPort, u32, cb));
1687 break;
1688 }
1689 return rc;
1690}
1691
1692#endif /* IN_RING3 */
1693
1694/**
1695 * @callback_method_impl{FNIOMMMIONEWWRITE, Flash memory write}
1696 */
1697static DECLCALLBACK(VBOXSTRICTRC) efiR3NvMmioWrite(PPDMDEVINS pDevIns, void *pvUser, RTGCPHYS off, void const *pv, unsigned cb)
1698{
1699 PDEVEFI pThis = PDMDEVINS_2_DATA(pDevIns, PDEVEFI);
1700 RT_NOREF(pvUser);
1701
1702 return flashWrite(&pThis->Flash, off, pv, cb);
1703}
1704
1705
1706/**
1707 * @callback_method_impl{FNIOMMMIONEWREAD, Flash memory read}
1708 */
1709static DECLCALLBACK(VBOXSTRICTRC) efiR3NvMmioRead(PPDMDEVINS pDevIns, void *pvUser, RTGCPHYS off, void *pv, unsigned cb)
1710{
1711 PDEVEFI pThis = PDMDEVINS_2_DATA(pDevIns, PDEVEFI);
1712 RT_NOREF(pvUser);
1713
1714 return flashRead(&pThis->Flash, off, pv, cb);
1715}
1716
1717#ifdef IN_RING3
1718
1719static DECLCALLBACK(int) efiSaveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
1720{
1721 PDEVEFI pThis = PDMDEVINS_2_DATA(pDevIns, PDEVEFI);
1722 LogFlow(("efiSaveExec:\n"));
1723
1724 return flashR3SaveExec(&pThis->Flash, pDevIns, pSSM);
1725}
1726
1727static DECLCALLBACK(int) efiLoadExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uVersion, uint32_t uPass)
1728{
1729 PDEVEFI pThis = PDMDEVINS_2_DATA(pDevIns, PDEVEFI);
1730 PDEVEFIR3 pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PDEVEFIR3);
1731 PCPDMDEVHLPR3 pHlp = pDevIns->pHlpR3;
1732 LogFlow(("efiLoadExec: uVersion=%d uPass=%d\n", uVersion, uPass));
1733
1734 /*
1735 * Validate input.
1736 */
1737 if (uPass != SSM_PASS_FINAL)
1738 return VERR_SSM_UNEXPECTED_PASS;
1739 if ( uVersion != EFI_SSM_VERSION
1740 && uVersion != EFI_SSM_VERSION_PRE_PROPER_NVRAM
1741 && uVersion != EFI_SSM_VERSION_4_2
1742 )
1743 return VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION;
1744
1745 int rc;
1746 if (uVersion > EFI_SSM_VERSION_PRE_PROPER_NVRAM)
1747 rc = flashR3LoadExec(&pThis->Flash, pDevIns, pSSM);
1748 else
1749 {
1750 /*
1751 * Kill the current variables before loading anything.
1752 */
1753 nvramFlushDeviceVariableList(pThisCC);
1754
1755 /*
1756 * Load the NVRAM state.
1757 */
1758 rc = pHlp->pfnSSMGetStructEx(pSSM, &pThisCC->NVRAM, sizeof(NVRAMDESC), 0, g_aEfiNvramDescField, NULL);
1759 AssertRCReturn(rc, rc);
1760 pThisCC->NVRAM.pCurVar = NULL;
1761
1762 rc = pHlp->pfnSSMGetStructEx(pSSM, &pThisCC->NVRAM.VarOpBuf, sizeof(EFIVAR), 0, g_aEfiVariableDescFields, NULL);
1763 AssertRCReturn(rc, rc);
1764
1765 /*
1766 * Load variables.
1767 */
1768 pThisCC->NVRAM.pCurVar = NULL;
1769 Assert(RTListIsEmpty(&pThisCC->NVRAM.VarList));
1770 RTListInit(&pThisCC->NVRAM.VarList);
1771 for (uint32_t i = 0; i < pThisCC->NVRAM.cVariables; i++)
1772 {
1773 PEFIVAR pEfiVar = (PEFIVAR)RTMemAllocZ(sizeof(EFIVAR));
1774 AssertReturn(pEfiVar, VERR_NO_MEMORY);
1775
1776 rc = pHlp->pfnSSMGetStructEx(pSSM, pEfiVar, sizeof(EFIVAR), 0, g_aEfiVariableDescFields, NULL);
1777 if (RT_SUCCESS(rc))
1778 {
1779 if ( pEfiVar->cbValue > sizeof(pEfiVar->abValue)
1780 || pEfiVar->cbValue == 0)
1781 {
1782 rc = VERR_SSM_DATA_UNIT_FORMAT_CHANGED;
1783 LogRel(("EFI: Loaded invalid variable value length %#x\n", pEfiVar->cbValue));
1784 }
1785 uint32_t cchVarName = (uint32_t)RTStrNLen(pEfiVar->szName, sizeof(pEfiVar->szName));
1786 if (cchVarName >= sizeof(pEfiVar->szName))
1787 {
1788 rc = VERR_SSM_DATA_UNIT_FORMAT_CHANGED;
1789 LogRel(("EFI: Loaded variable name is unterminated.\n"));
1790 }
1791 if (pEfiVar->cchName > cchVarName) /* No check for 0 here, busted load code in 4.2, so now storing 0 here. */
1792 {
1793 rc = VERR_SSM_DATA_UNIT_FORMAT_CHANGED;
1794 LogRel(("EFI: Loaded invalid variable name length %#x (cchVarName=%#x)\n", pEfiVar->cchName, cchVarName));
1795 }
1796 if (RT_SUCCESS(rc))
1797 pEfiVar->cchName = cchVarName;
1798 }
1799 AssertRCReturnStmt(rc, RTMemFree(pEfiVar), rc);
1800
1801 /* Add it (not using nvramInsertVariable to preserve saved order),
1802 updating the current variable pointer while we're here. */
1803#if 1
1804 RTListAppend(&pThisCC->NVRAM.VarList, &pEfiVar->ListNode);
1805#else
1806 nvramInsertVariable(pThisCC, pEfiVar);
1807#endif
1808 if (pThisCC->NVRAM.idUniqueCurVar == pEfiVar->idUniqueSavedState)
1809 pThisCC->NVRAM.pCurVar = pEfiVar;
1810 }
1811 }
1812
1813 return rc;
1814}
1815
1816
1817/**
1818 * @copydoc(PDMIBASE::pfnQueryInterface)
1819 */
1820static DECLCALLBACK(void *) devEfiQueryInterface(PPDMIBASE pInterface, const char *pszIID)
1821{
1822 LogFlowFunc(("ENTER: pIBase=%p pszIID=%p\n", pInterface, pszIID));
1823 PDEVEFIR3 pThisCC = RT_FROM_MEMBER(pInterface, DEVEFIR3, Lun0.IBase);
1824
1825 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pThisCC->Lun0.IBase);
1826 return NULL;
1827}
1828
1829
1830/**
1831 * Write to CMOS memory.
1832 * This is used by the init complete code.
1833 */
1834static void cmosWrite(PPDMDEVINS pDevIns, unsigned off, uint32_t u32Val)
1835{
1836 Assert(off < 128);
1837 Assert(u32Val < 256);
1838
1839 int rc = PDMDevHlpCMOSWrite(pDevIns, off, u32Val);
1840 AssertRC(rc);
1841}
1842
1843/**
1844 * Init complete notification.
1845 *
1846 * @returns VBOX status code.
1847 * @param pDevIns The device instance.
1848 */
1849static DECLCALLBACK(int) efiInitComplete(PPDMDEVINS pDevIns)
1850{
1851 PDEVEFIR3 pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PDEVEFIR3);
1852
1853 PVM pVM = PDMDevHlpGetVM(pDevIns);
1854 uint64_t const cbRamSize = MMR3PhysGetRamSize(pVM);
1855 uint32_t const cbBelow4GB = MMR3PhysGetRamSizeBelow4GB(pVM);
1856 uint64_t const cbAbove4GB = MMR3PhysGetRamSizeAbove4GB(pVM);
1857 NOREF(cbAbove4GB);
1858
1859 /*
1860 * Memory sizes.
1861 */
1862 uint32_t u32Low = 0;
1863 uint32_t u32Chunks = 0;
1864 if (cbRamSize > 16 * _1M)
1865 {
1866 u32Low = RT_MIN(cbBelow4GB, UINT32_C(0xfe000000));
1867 u32Chunks = (u32Low - 16U * _1M) / _64K;
1868 }
1869 cmosWrite(pDevIns, 0x34, RT_BYTE1(u32Chunks));
1870 cmosWrite(pDevIns, 0x35, RT_BYTE2(u32Chunks));
1871
1872 if (u32Low < cbRamSize)
1873 {
1874 uint64_t u64 = cbRamSize - u32Low;
1875 u32Chunks = (uint32_t)(u64 / _64K);
1876 cmosWrite(pDevIns, 0x5b, RT_BYTE1(u32Chunks));
1877 cmosWrite(pDevIns, 0x5c, RT_BYTE2(u32Chunks));
1878 cmosWrite(pDevIns, 0x5d, RT_BYTE3(u32Chunks));
1879 cmosWrite(pDevIns, 0x5e, RT_BYTE4(u32Chunks));
1880 }
1881
1882 /*
1883 * Number of CPUs.
1884 */
1885 cmosWrite(pDevIns, 0x60, pThisCC->cCpus & 0xff);
1886
1887 return VINF_SUCCESS;
1888}
1889
1890
1891/**
1892 * @interface_method_impl{PDMDEVREG,pfnMemSetup}
1893 */
1894static DECLCALLBACK(void) efiMemSetup(PPDMDEVINS pDevIns, PDMDEVMEMSETUPCTX enmCtx)
1895{
1896 RT_NOREF(enmCtx);
1897 PDEVEFIR3 pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PDEVEFIR3);
1898
1899 /*
1900 * Re-shadow the Firmware Volume and make it RAM/RAM.
1901 */
1902 uint32_t cPages = RT_ALIGN_64(pThisCC->cbEfiRom, PAGE_SIZE) >> PAGE_SHIFT;
1903 RTGCPHYS GCPhys = pThisCC->GCLoadAddress;
1904 while (cPages > 0)
1905 {
1906 uint8_t abPage[PAGE_SIZE];
1907
1908 /* Read the (original) ROM page and write it back to the RAM page. */
1909 int rc = PDMDevHlpROMProtectShadow(pDevIns, GCPhys, PAGE_SIZE, PGMROMPROT_READ_ROM_WRITE_RAM);
1910 AssertLogRelRC(rc);
1911
1912 rc = PDMDevHlpPhysRead(pDevIns, GCPhys, abPage, PAGE_SIZE);
1913 AssertLogRelRC(rc);
1914 if (RT_FAILURE(rc))
1915 memset(abPage, 0xcc, sizeof(abPage));
1916
1917 rc = PDMDevHlpPhysWrite(pDevIns, GCPhys, abPage, PAGE_SIZE);
1918 AssertLogRelRC(rc);
1919
1920 /* Switch to the RAM/RAM mode. */
1921 rc = PDMDevHlpROMProtectShadow(pDevIns, GCPhys, PAGE_SIZE, PGMROMPROT_READ_RAM_WRITE_RAM);
1922 AssertLogRelRC(rc);
1923
1924 /* Advance */
1925 GCPhys += PAGE_SIZE;
1926 cPages--;
1927 }
1928}
1929
1930
1931/**
1932 * @interface_method_impl{PDMDEVREG,pfnReset}
1933 */
1934static DECLCALLBACK(void) efiReset(PPDMDEVINS pDevIns)
1935{
1936 PDEVEFI pThis = PDMDEVINS_2_DATA(pDevIns, PDEVEFI);
1937 PDEVEFIR3 pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PDEVEFIR3);
1938 LogFlow(("efiReset\n"));
1939
1940 pThisCC->iInfoSelector = 0;
1941 pThisCC->offInfo = -1;
1942
1943 pThisCC->iMsg = 0;
1944 pThisCC->szMsg[0] = '\0';
1945 pThisCC->iPanicMsg = 0;
1946 pThisCC->szPanicMsg[0] = '\0';
1947
1948 flashR3Reset(&pThis->Flash);
1949
1950#ifdef DEVEFI_WITH_VBOXDBG_SCRIPT
1951 /*
1952 * Zap the debugger script
1953 */
1954 RTFileDelete("./DevEFI.VBoxDbg");
1955#endif
1956}
1957
1958
1959/**
1960 * @interface_method_impl{PDMDEVREG,pfnPowerOff}
1961 */
1962static DECLCALLBACK(void) efiPowerOff(PPDMDEVINS pDevIns)
1963{
1964 PDEVEFIR3 pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PDEVEFIR3);
1965
1966 if (pThisCC->Lun0.pNvramDrv)
1967 nvramStore(pThisCC);
1968}
1969
1970
1971
1972/**
1973 * Destruct a device instance.
1974 *
1975 * Most VM resources are freed by the VM. This callback is provided so that any non-VM
1976 * resources can be freed correctly.
1977 *
1978 * @param pDevIns The device instance data.
1979 */
1980static DECLCALLBACK(int) efiDestruct(PPDMDEVINS pDevIns)
1981{
1982 PDMDEV_CHECK_VERSIONS_RETURN_QUIET(pDevIns);
1983 PDEVEFI pThis = PDMDEVINS_2_DATA(pDevIns, PDEVEFI);
1984 PDEVEFIR3 pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PDEVEFIR3);
1985
1986 nvramFlushDeviceVariableList(pThisCC);
1987
1988 if (pThisCC->pszNvramFile)
1989 {
1990 int rc = flashR3SaveToFile(&pThis->Flash, pDevIns, pThisCC->pszNvramFile);
1991 if (RT_FAILURE(rc))
1992 LogRel(("EFI: Failed to save flash file to '%s': %Rrc\n", pThisCC->pszNvramFile, rc));
1993 }
1994
1995 flashR3Destruct(&pThis->Flash, pDevIns);
1996
1997 if (pThisCC->pszNvramFile)
1998 {
1999 PDMDevHlpMMHeapFree(pDevIns, pThisCC->pszNvramFile);
2000 pThisCC->pszNvramFile = NULL;
2001 }
2002
2003 if (pThisCC->pu8EfiRomFree)
2004 {
2005 RTFileReadAllFree(pThisCC->pu8EfiRomFree, (size_t)pThisCC->cbEfiRom + pThisCC->offEfiRom);
2006 pThisCC->pu8EfiRomFree = NULL;
2007 }
2008
2009 /*
2010 * Free MM heap pointers (waste of time, but whatever).
2011 */
2012 if (pThisCC->pszEfiRomFile)
2013 {
2014 PDMDevHlpMMHeapFree(pDevIns, pThisCC->pszEfiRomFile);
2015 pThisCC->pszEfiRomFile = NULL;
2016 }
2017
2018 if (pThisCC->pu8EfiThunk)
2019 {
2020 PDMDevHlpMMHeapFree(pDevIns, pThisCC->pu8EfiThunk);
2021 pThisCC->pu8EfiThunk = NULL;
2022 }
2023
2024 if (pThisCC->pbDeviceProps)
2025 {
2026 PDMDevHlpMMHeapFree(pDevIns, pThisCC->pbDeviceProps);
2027 pThisCC->pbDeviceProps = NULL;
2028 pThisCC->cbDeviceProps = 0;
2029 }
2030
2031 return VINF_SUCCESS;
2032}
2033
2034
2035#if 0 /* unused */
2036/**
2037 * Helper that searches for a FFS file of a given type.
2038 *
2039 * @returns Pointer to the FFS file header if found, NULL if not.
2040 *
2041 * @param pFfsFile Pointer to the FFS file header to start searching at.
2042 * @param pbEnd The end of the firmware volume.
2043 * @param FileType The file type to look for.
2044 * @param pcbFfsFile Where to store the FFS file size (includes header).
2045 */
2046DECLINLINE(EFI_FFS_FILE_HEADER const *)
2047efiFwVolFindFileByType(EFI_FFS_FILE_HEADER const *pFfsFile, uint8_t const *pbEnd, EFI_FV_FILETYPE FileType, uint32_t *pcbFile)
2048{
2049# define FFS_SIZE(hdr) RT_MAKE_U32_FROM_U8((hdr)->Size[0], (hdr)->Size[1], (hdr)->Size[2], 0)
2050 while ((uintptr_t)pFfsFile < (uintptr_t)pbEnd)
2051 {
2052 if (pFfsFile->Type == FileType)
2053 {
2054 *pcbFile = FFS_SIZE(pFfsFile);
2055 LogFunc(("Found %RTuuid of type:%d\n", &pFfsFile->Name, FileType));
2056 return pFfsFile;
2057 }
2058 pFfsFile = (EFI_FFS_FILE_HEADER *)((uintptr_t)pFfsFile + RT_ALIGN(FFS_SIZE(pFfsFile), 8));
2059 }
2060# undef FFS_SIZE
2061 return NULL;
2062}
2063#endif /* unused */
2064
2065
2066/**
2067 * Parse EFI ROM headers and find entry points.
2068 *
2069 * @returns VBox status code.
2070 * @param pDevIns The device instance.
2071 * @param pThis The shared device state.
2072 * @param pThisCC The device state for the current context.
2073 */
2074static int efiParseFirmware(PPDMDEVINS pDevIns, PDEVEFI pThis, PDEVEFIR3 pThisCC)
2075{
2076 EFI_FIRMWARE_VOLUME_HEADER const *pFwVolHdr = (EFI_FIRMWARE_VOLUME_HEADER const *)pThisCC->pu8EfiRom;
2077
2078 /*
2079 * Validate firmware volume header.
2080 */
2081 AssertLogRelMsgReturn(pFwVolHdr->Signature == RT_MAKE_U32_FROM_U8('_', 'F', 'V', 'H'),
2082 ("%#x, expected %#x\n", pFwVolHdr->Signature, RT_MAKE_U32_FROM_U8('_', 'F', 'V', 'H')),
2083 VERR_INVALID_MAGIC);
2084 AssertLogRelMsgReturn(pFwVolHdr->Revision == EFI_FVH_REVISION,
2085 ("%#x, expected %#x\n", pFwVolHdr->Signature, EFI_FVH_REVISION),
2086 VERR_VERSION_MISMATCH);
2087 /** @todo check checksum, see PE spec vol. 3 */
2088 AssertLogRelMsgReturn(pFwVolHdr->FvLength <= pThisCC->cbEfiRom,
2089 ("%#llx, expected %#llx\n", pFwVolHdr->FvLength, pThisCC->cbEfiRom),
2090 VERR_INVALID_PARAMETER);
2091 AssertLogRelMsgReturn( pFwVolHdr->BlockMap[0].Length > 0
2092 && pFwVolHdr->BlockMap[0].NumBlocks > 0,
2093 ("%#x, %x\n", pFwVolHdr->BlockMap[0].Length, pFwVolHdr->BlockMap[0].NumBlocks),
2094 VERR_INVALID_PARAMETER);
2095
2096 AssertLogRelMsgReturn(!(pThisCC->cbEfiRom & PAGE_OFFSET_MASK), ("%RX64\n", pThisCC->cbEfiRom), VERR_INVALID_PARAMETER);
2097
2098 LogRel(("Found EFI FW Volume, %u bytes (%u %u-byte blocks)\n", pFwVolHdr->FvLength, pFwVolHdr->BlockMap[0].NumBlocks, pFwVolHdr->BlockMap[0].Length));
2099
2100 /** @todo Make this more dynamic, this assumes that the NV storage area comes first (always the case for our builds). */
2101 AssertLogRelMsgReturn(!memcmp(&pFwVolHdr->FileSystemGuid, &g_UuidNvDataFv, sizeof(g_UuidNvDataFv)),
2102 ("Expected EFI_SYSTEM_NV_DATA_FV_GUID as an identifier"),
2103 VERR_INVALID_MAGIC);
2104
2105 /* Found NVRAM storage, configure flash device. */
2106 pThisCC->offEfiRom = pFwVolHdr->FvLength;
2107 pThisCC->cbNvram = pFwVolHdr->FvLength;
2108 pThisCC->GCPhysNvram = UINT32_C(0xfffff000) - pThisCC->cbEfiRom + PAGE_SIZE;
2109 pThisCC->cbEfiRom -= pThisCC->cbNvram;
2110
2111 int rc = flashR3Init(&pThis->Flash, pThisCC->pDevIns, 0xA289 /*Intel*/, pThisCC->cbNvram, pFwVolHdr->BlockMap[0].Length);
2112 if (RT_FAILURE(rc))
2113 return rc;
2114
2115 /* If the file does not exist we initialize the NVRAM from the loaded ROM file. */
2116 if (!pThisCC->pszNvramFile || !RTPathExists(pThisCC->pszNvramFile))
2117 rc = flashR3LoadFromBuf(&pThis->Flash, pThisCC->pu8EfiRom, pThisCC->cbNvram);
2118 else
2119 rc = flashR3LoadFromFile(&pThis->Flash, pDevIns, pThisCC->pszNvramFile);
2120 if (RT_FAILURE(rc))
2121 return rc;
2122
2123 pThisCC->GCLoadAddress = pThisCC->GCPhysNvram + pThisCC->cbNvram;
2124
2125 return VINF_SUCCESS;
2126}
2127
2128/**
2129 * Load EFI ROM file into the memory.
2130 *
2131 * @returns VBox status code.
2132 * @param pDevIns The device instance.
2133 * @param pThis The shared Efi state.
2134 * @param pThisCC The device state for the current context.
2135 * @param pCfg Configuration node handle for the device.
2136 */
2137static int efiLoadRom(PPDMDEVINS pDevIns, PDEVEFI pThis, PDEVEFIR3 pThisCC, PCFGMNODE pCfg)
2138{
2139 RT_NOREF(pCfg);
2140
2141 /*
2142 * Read the entire firmware volume into memory.
2143 */
2144 int rc;
2145#ifdef VBOX_WITH_EFI_IN_DD2
2146 if (RTStrCmp(pThisCC->pszEfiRomFile, g_szEfiBuiltin32) == 0)
2147 {
2148 pThisCC->pu8EfiRomFree = NULL;
2149 pThisCC->pu8EfiRom = g_abEfiFirmware32;
2150 pThisCC->cbEfiRom = g_cbEfiFirmware32;
2151 }
2152 else if (RTStrCmp(pThisCC->pszEfiRomFile, g_szEfiBuiltin64) == 0)
2153 {
2154 pThisCC->pu8EfiRomFree = NULL;
2155 pThisCC->pu8EfiRom = g_abEfiFirmware64;
2156 pThisCC->cbEfiRom = g_cbEfiFirmware64;
2157 }
2158 else
2159#endif
2160 {
2161 void *pvFile;
2162 size_t cbFile;
2163 rc = RTFileReadAllEx(pThisCC->pszEfiRomFile,
2164 0 /*off*/,
2165 RTFOFF_MAX /*cbMax*/,
2166 RTFILE_RDALL_O_DENY_WRITE,
2167 &pvFile,
2168 &cbFile);
2169 if (RT_FAILURE(rc))
2170 return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS,
2171 N_("Loading the EFI firmware volume '%s' failed with rc=%Rrc"),
2172 pThisCC->pszEfiRomFile, rc);
2173 pThisCC->pu8EfiRomFree = (uint8_t *)pvFile;
2174 pThisCC->pu8EfiRom = (uint8_t *)pvFile;
2175 pThisCC->cbEfiRom = cbFile;
2176 }
2177
2178 /*
2179 * Validate firmware volume and figure out the load address as well as the SEC entry point.
2180 */
2181 rc = efiParseFirmware(pDevIns, pThis, pThisCC);
2182 if (RT_FAILURE(rc))
2183 return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS,
2184 N_("Parsing the EFI firmware volume '%s' failed with rc=%Rrc"),
2185 pThisCC->pszEfiRomFile, rc);
2186
2187 /*
2188 * Map the firmware volume into memory as shadowed ROM.
2189 *
2190 * This is a little complicated due to saved state legacy. We used to have a
2191 * 2MB image w/o any flash portion, divided into four 512KB mappings.
2192 *
2193 * We've now increased the size of the firmware to 4MB, but for saved state
2194 * compatibility reasons need to use the same mappings and names (!!) for the
2195 * top 2MB.
2196 */
2197 /** @todo fix PGMR3PhysRomRegister so it doesn't mess up in SUPLib when mapping a big ROM image. */
2198#if 1
2199 static const char * const s_apszNames[16] =
2200 {
2201 "EFI Firmware Volume", "EFI Firmware Volume (Part 2)", "EFI Firmware Volume (Part 3)", "EFI Firmware Volume (Part 4)",
2202 "EFI Firmware Volume (Part 5)", "EFI Firmware Volume (Part 6)", "EFI Firmware Volume (Part 7)", "EFI Firmware Volume (Part 8)",
2203 "EFI Firmware Volume (Part 9)", "EFI Firmware Volume (Part 10)", "EFI Firmware Volume (Part 11)", "EFI Firmware Volume (Part 12)",
2204 "EFI Firmware Volume (Part 13)", "EFI Firmware Volume (Part 14)", "EFI Firmware Volume (Part 15)", "EFI Firmware Volume (Part 16)",
2205 };
2206 AssertLogRelMsgReturn(pThisCC->cbEfiRom < RT_ELEMENTS(s_apszNames) * _512K,
2207 ("EFI firmware image too big: %#RX64, max %#zx\n",
2208 pThisCC->cbEfiRom, RT_ELEMENTS(s_apszNames) * _512K),
2209 VERR_IMAGE_TOO_BIG);
2210
2211 uint32_t const cbChunk = pThisCC->cbNvram + pThisCC->cbEfiRom >= _2M ? _512K
2212 : (uint32_t)RT_ALIGN_64((pThisCC->cbNvram + pThisCC->cbEfiRom) / 4, PAGE_SIZE);
2213 uint32_t cbLeft = pThisCC->cbEfiRom; /* ASSUMES NVRAM comes first! */
2214 uint32_t off = pThisCC->offEfiRom + cbLeft; /* ASSUMES NVRAM comes first! */
2215 RTGCPHYS64 GCPhys = pThisCC->GCLoadAddress + cbLeft;
2216 AssertLogRelMsg(GCPhys == _4G, ("%RGp\n", GCPhys));
2217
2218 /* Compatibility mappings at the top (note that this isn't entirely the same
2219 algorithm, but it will produce the same results for a power of two sized image): */
2220 unsigned i = 4;
2221 while (i-- > 0)
2222 {
2223 uint32_t const cb = RT_MIN(cbLeft, cbChunk);
2224 cbLeft -= cb;
2225 GCPhys -= cb;
2226 off -= cb;
2227 rc = PDMDevHlpROMRegister(pDevIns, GCPhys, cb, pThisCC->pu8EfiRom + off, cb,
2228 PGMPHYS_ROM_FLAGS_SHADOWED | PGMPHYS_ROM_FLAGS_PERMANENT_BINARY, s_apszNames[i]);
2229 AssertRCReturn(rc, rc);
2230 }
2231
2232 /* The rest (if any) is mapped in descending order of address and increasing name order: */
2233 if (cbLeft > 0)
2234 {
2235 Assert(cbChunk == _512K);
2236 for (i = 4; cbLeft > 0; i++)
2237 {
2238 uint32_t const cb = RT_MIN(cbLeft, cbChunk);
2239 cbLeft -= cb;
2240 GCPhys -= cb;
2241 off -= cb;
2242 /** @todo Add flag to prevent saved state loading from bitching about these regions. */
2243 rc = PDMDevHlpROMRegister(pDevIns, GCPhys, cb, pThisCC->pu8EfiRom + off, cb,
2244 PGMPHYS_ROM_FLAGS_SHADOWED | PGMPHYS_ROM_FLAGS_PERMANENT_BINARY
2245 | PGMPHYS_ROM_FLAGS_MAYBE_MISSING_FROM_STATE, s_apszNames[i]);
2246 AssertRCReturn(rc, rc);
2247 }
2248 Assert(i <= RT_ELEMENTS(s_apszNames));
2249 }
2250
2251 /* Not sure what the purpose of this one is... */
2252 rc = PDMDevHlpROMProtectShadow(pDevIns, pThisCC->GCLoadAddress, (uint32_t)cbChunk, PGMROMPROT_READ_RAM_WRITE_IGNORE);
2253 AssertRCReturn(rc, rc);
2254
2255#else
2256 RTGCPHYS cbQuart = RT_ALIGN_64(pThisCC->cbEfiRom / 4, PAGE_SIZE);
2257 rc = PDMDevHlpROMRegister(pDevIns,
2258 pThisCC->GCLoadAddress,
2259 cbQuart,
2260 pThisCC->pu8EfiRom + pThisCC->uEfiRomOfs,
2261 cbQuart,
2262 PGMPHYS_ROM_FLAGS_SHADOWED | PGMPHYS_ROM_FLAGS_PERMANENT_BINARY,
2263 "EFI Firmware Volume");
2264 AssertRCReturn(rc, rc);
2265 rc = PDMDevHlpROMProtectShadow(pDevIns, pThisCC->GCLoadAddress, (uint32_t)cbQuart, PGMROMPROT_READ_RAM_WRITE_IGNORE);
2266 AssertRCReturn(rc, rc);
2267 rc = PDMDevHlpROMRegister(pDevIns,
2268 pThisCC->GCLoadAddress + cbQuart,
2269 cbQuart,
2270 pThisCC->pu8EfiRom + pThisCC->uEfiRomOfs + cbQuart,
2271 cbQuart,
2272 PGMPHYS_ROM_FLAGS_SHADOWED | PGMPHYS_ROM_FLAGS_PERMANENT_BINARY,
2273 "EFI Firmware Volume (Part 2)");
2274 if (RT_FAILURE(rc))
2275 return rc;
2276 rc = PDMDevHlpROMRegister(pDevIns,
2277 pThisCC->GCLoadAddress + cbQuart * 2,
2278 cbQuart,
2279 pThisCC->pu8EfiRom + pThisCC->uEfiRomOfs + cbQuart * 2,
2280 cbQuart,
2281 PGMPHYS_ROM_FLAGS_SHADOWED | PGMPHYS_ROM_FLAGS_PERMANENT_BINARY,
2282 "EFI Firmware Volume (Part 3)");
2283 if (RT_FAILURE(rc))
2284 return rc;
2285 rc = PDMDevHlpROMRegister(pDevIns,
2286 pThisCC->GCLoadAddress + cbQuart * 3,
2287 pThisCC->cbEfiRom - cbQuart * 3,
2288 pThisCC->pu8EfiRom + pThisCC->uEfiRomOfs + cbQuart * 3,
2289 pThisCC->cbEfiRom - cbQuart * 3,
2290 PGMPHYS_ROM_FLAGS_SHADOWED | PGMPHYS_ROM_FLAGS_PERMANENT_BINARY,
2291 "EFI Firmware Volume (Part 4)");
2292 if (RT_FAILURE(rc))
2293 return rc;
2294#endif
2295
2296 /*
2297 * Register MMIO region for flash device.
2298 */
2299 rc = PDMDevHlpMmioCreateEx(pDevIns, pThisCC->cbNvram, IOMMMIO_FLAGS_READ_PASSTHRU | IOMMMIO_FLAGS_WRITE_PASSTHRU,
2300 NULL /*pPciDev*/, UINT32_MAX, efiR3NvMmioWrite, efiR3NvMmioRead, NULL, NULL /*pvUser*/,
2301 "Flash Memory", &pThis->hMmioFlash);
2302 AssertRCReturn(rc, rc);
2303 rc = PDMDevHlpMmioMap(pDevIns, pThis->hMmioFlash, pThisCC->GCPhysNvram);
2304 AssertRCReturn(rc, rc);
2305
2306 LogRel(("EFI: Registered %uKB flash at %RGp\n", pThisCC->cbNvram / _1K, pThisCC->GCPhysNvram));
2307 return VINF_SUCCESS;
2308}
2309
2310static uint8_t efiGetHalfByte(char ch)
2311{
2312 uint8_t val;
2313
2314 if (ch >= '0' && ch <= '9')
2315 val = ch - '0';
2316 else if (ch >= 'A' && ch <= 'F')
2317 val = ch - 'A' + 10;
2318 else if(ch >= 'a' && ch <= 'f')
2319 val = ch - 'a' + 10;
2320 else
2321 val = 0xff;
2322
2323 return val;
2324}
2325
2326
2327/**
2328 * Converts a hex string into a binary data blob located at
2329 * pThisCC->pbDeviceProps, size returned as pThisCC->cbDeviceProps.
2330 *
2331 * @returns VERR_NO_MEMORY or VINF_SUCCESS.
2332 * @param pThisCC The device state for the current context.
2333 * @param pszDeviceProps The device property hex string to decode.
2334 */
2335static int efiParseDeviceString(PDEVEFIR3 pThisCC, const char *pszDeviceProps)
2336{
2337 uint32_t const cbOut = (uint32_t)RTStrNLen(pszDeviceProps, RTSTR_MAX) / 2 + 1;
2338 pThisCC->pbDeviceProps = (uint8_t *)PDMDevHlpMMHeapAlloc(pThisCC->pDevIns, cbOut);
2339 if (!pThisCC->pbDeviceProps)
2340 return VERR_NO_MEMORY;
2341
2342 uint32_t iHex = 0;
2343 bool fUpper = true;
2344 uint8_t u8Value = 0; /* (shut up gcc) */
2345 for (uint32_t iStr = 0; pszDeviceProps[iStr]; iStr++)
2346 {
2347 uint8_t u8Hb = efiGetHalfByte(pszDeviceProps[iStr]);
2348 if (u8Hb > 0xf)
2349 continue;
2350
2351 if (fUpper)
2352 u8Value = u8Hb << 4;
2353 else
2354 pThisCC->pbDeviceProps[iHex++] = u8Hb | u8Value;
2355
2356 Assert(iHex < cbOut);
2357 fUpper = !fUpper;
2358 }
2359
2360 Assert(iHex == 0 || fUpper);
2361 pThisCC->cbDeviceProps = iHex;
2362
2363 return VINF_SUCCESS;
2364}
2365
2366
2367/**
2368 * @interface_method_impl{PDMDEVREG,pfnConstruct}
2369 */
2370static DECLCALLBACK(int) efiConstruct(PPDMDEVINS pDevIns, int iInstance, PCFGMNODE pCfg)
2371{
2372 PDMDEV_CHECK_VERSIONS_RETURN(pDevIns);
2373 PDEVEFI pThis = PDMDEVINS_2_DATA(pDevIns, PDEVEFI);
2374 PDEVEFIR3 pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PDEVEFIR3);
2375 PCPDMDEVHLPR3 pHlp = pDevIns->pHlpR3;
2376 int rc;
2377
2378 RT_NOREF(iInstance);
2379 Assert(iInstance == 0);
2380
2381 /*
2382 * Initalize the basic variables so that the destructor always works.
2383 */
2384 pThisCC->pDevIns = pDevIns;
2385 RTListInit(&pThisCC->NVRAM.VarList);
2386 pThisCC->Lun0.IBase.pfnQueryInterface = devEfiQueryInterface;
2387
2388 /*
2389 * Validate and read the configuration.
2390 */
2391 PDMDEV_VALIDATE_CONFIG_RETURN(pDevIns,
2392 "EfiRom|"
2393 "NumCPUs|"
2394 "McfgBase|"
2395 "McfgLength|"
2396 "UUID|"
2397 "UuidLe|"
2398 "IOAPIC|"
2399 "APIC|"
2400 "DmiBIOSFirmwareMajor|"
2401 "DmiBIOSFirmwareMinor|"
2402 "DmiBIOSReleaseDate|"
2403 "DmiBIOSReleaseMajor|"
2404 "DmiBIOSReleaseMinor|"
2405 "DmiBIOSVendor|"
2406 "DmiBIOSVersion|"
2407 "DmiSystemFamily|"
2408 "DmiSystemProduct|"
2409 "DmiSystemSerial|"
2410 "DmiSystemSKU|"
2411 "DmiSystemUuid|"
2412 "DmiSystemVendor|"
2413 "DmiSystemVersion|"
2414 "DmiBoardAssetTag|"
2415 "DmiBoardBoardType|"
2416 "DmiBoardLocInChass|"
2417 "DmiBoardProduct|"
2418 "DmiBoardSerial|"
2419 "DmiBoardVendor|"
2420 "DmiBoardVersion|"
2421 "DmiChassisAssetTag|"
2422 "DmiChassisSerial|"
2423 "DmiChassisType|"
2424 "DmiChassisVendor|"
2425 "DmiChassisVersion|"
2426 "DmiProcManufacturer|"
2427 "DmiProcVersion|"
2428 "DmiOEMVBoxVer|"
2429 "DmiOEMVBoxRev|"
2430 "DmiUseHostInfo|"
2431 "DmiExposeMemoryTable|"
2432 "DmiExposeProcInf|"
2433 "64BitEntry|"
2434 "BootArgs|"
2435 "DeviceProps|"
2436 "GopMode|" // legacy
2437 "GraphicsMode|"
2438 "UgaHorizontalResolution|" // legacy
2439 "UgaVerticalResolution|" // legacy
2440 "GraphicsResolution|"
2441 "NvramFile", "");
2442
2443 /* CPU count (optional). */
2444 rc = pHlp->pfnCFGMQueryU32Def(pCfg, "NumCPUs", &pThisCC->cCpus, 1);
2445 AssertLogRelRCReturn(rc, rc);
2446
2447 rc = pHlp->pfnCFGMQueryU64Def(pCfg, "McfgBase", &pThisCC->u64McfgBase, 0);
2448 if (RT_FAILURE(rc))
2449 return PDMDEV_SET_ERROR(pDevIns, rc,
2450 N_("Configuration error: Querying \"\" as integer failed"));
2451 rc = pHlp->pfnCFGMQueryU64Def(pCfg, "McfgLength", &pThisCC->cbMcfgLength, 0);
2452 if (RT_FAILURE(rc))
2453 return PDMDEV_SET_ERROR(pDevIns, rc,
2454 N_("Configuration error: Querying \"McfgLength\" as integer failed"));
2455
2456 rc = pHlp->pfnCFGMQueryU8Def(pCfg, "IOAPIC", &pThisCC->u8IOAPIC, 1);
2457 if (RT_FAILURE (rc))
2458 return PDMDEV_SET_ERROR(pDevIns, rc,
2459 N_("Configuration error: Failed to read \"IOAPIC\""));
2460
2461 rc = pHlp->pfnCFGMQueryU8Def(pCfg, "APIC", &pThisCC->u8APIC, 1);
2462 if (RT_FAILURE (rc))
2463 return PDMDEV_SET_ERROR(pDevIns, rc,
2464 N_("Configuration error: Failed to read \"APIC\""));
2465
2466 /*
2467 * Query the machine's UUID for SMBIOS/DMI use.
2468 */
2469 RTUUID uuid;
2470 rc = pHlp->pfnCFGMQueryBytes(pCfg, "UUID", &uuid, sizeof(uuid));
2471 if (RT_FAILURE(rc))
2472 return PDMDEV_SET_ERROR(pDevIns, rc,
2473 N_("Configuration error: Querying \"UUID\" failed"));
2474
2475 bool fUuidLe;
2476 rc = pHlp->pfnCFGMQueryBoolDef(pCfg, "UuidLe", &fUuidLe, false);
2477 if (RT_FAILURE(rc))
2478 return PDMDEV_SET_ERROR(pDevIns, rc,
2479 N_("Configuration error: Querying \"UuidLe\" failed"));
2480
2481 if (!fUuidLe)
2482 {
2483 /*
2484 * UUIDs are stored little endian actually (see chapter 7.2.1 System — UUID
2485 * of the DMI/SMBIOS spec) but to not force reactivation of existing guests we have
2486 * to carry this bug along... (see also DevPcBios.cpp when changing this)
2487 *
2488 * Convert the UUID to network byte order. Not entirely straightforward as
2489 * parts are MSB already...
2490 */
2491 uuid.Gen.u32TimeLow = RT_H2BE_U32(uuid.Gen.u32TimeLow);
2492 uuid.Gen.u16TimeMid = RT_H2BE_U16(uuid.Gen.u16TimeMid);
2493 uuid.Gen.u16TimeHiAndVersion = RT_H2BE_U16(uuid.Gen.u16TimeHiAndVersion);
2494 }
2495 memcpy(&pThisCC->aUuid, &uuid, sizeof pThisCC->aUuid);
2496
2497 /*
2498 * Get the system EFI ROM file name.
2499 */
2500#ifdef VBOX_WITH_EFI_IN_DD2
2501 rc = pHlp->pfnCFGMQueryStringAllocDef(pCfg, "EfiRom", &pThisCC->pszEfiRomFile, g_szEfiBuiltin32);
2502 if (RT_FAILURE(rc))
2503#else
2504 rc = pHlp->pfnCFGMQueryStringAlloc(pCfg, "EfiRom", &pThisCC->pszEfiRomFile);
2505 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
2506 {
2507 pThisCC->pszEfiRomFile = (char *)PDMDevHlpMMHeapAlloc(pDevIns, RTPATH_MAX);
2508 AssertReturn(pThisCC->pszEfiRomFile, VERR_NO_MEMORY);
2509 rc = RTPathAppPrivateArchTop(pThisCC->pszEfiRomFile, RTPATH_MAX);
2510 AssertRCReturn(rc, rc);
2511 rc = RTPathAppend(pThisCC->pszEfiRomFile, RTPATH_MAX, "VBoxEFI32.fd");
2512 AssertRCReturn(rc, rc);
2513 }
2514 else if (RT_FAILURE(rc))
2515#endif
2516 return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS,
2517 N_("Configuration error: Querying \"EfiRom\" as a string failed"));
2518
2519 /*
2520 * NVRAM processing.
2521 */
2522 rc = PDMDevHlpSSMRegister(pDevIns, EFI_SSM_VERSION, sizeof(*pThisCC), efiSaveExec, efiLoadExec);
2523 AssertRCReturn(rc, rc);
2524
2525 rc = PDMDevHlpDriverAttach(pDevIns, 0, &pThisCC->Lun0.IBase, &pThisCC->Lun0.pDrvBase, "NvramStorage");
2526 if (RT_SUCCESS(rc))
2527 {
2528 pThisCC->Lun0.pNvramDrv = PDMIBASE_QUERY_INTERFACE(pThisCC->Lun0.pDrvBase, PDMINVRAMCONNECTOR);
2529 AssertPtrReturn(pThisCC->Lun0.pNvramDrv, VERR_PDM_MISSING_INTERFACE_BELOW);
2530
2531 rc = nvramLoad(pThisCC);
2532 AssertRCReturn(rc, rc);
2533 }
2534 else if (rc == VERR_PDM_NO_ATTACHED_DRIVER)
2535 {
2536 pThisCC->Lun0.pNvramDrv = NULL;
2537 rc = VINF_SUCCESS; /* Missing driver is no error condition. */
2538 }
2539 else
2540 return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS, N_("Can't attach Nvram Storage driver"));
2541
2542 /*
2543 * Get boot args.
2544 */
2545 rc = pHlp->pfnCFGMQueryStringDef(pCfg, "BootArgs", pThisCC->szBootArgs, sizeof(pThisCC->szBootArgs), "");
2546 if (RT_FAILURE(rc))
2547 return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS,
2548 N_("Configuration error: Querying \"BootArgs\" as a string failed"));
2549
2550 //strcpy(pThisCC->szBootArgs, "-v keepsyms=1 io=0xf debug=0x2a");
2551 //strcpy(pThisCC->szBootArgs, "-v keepsyms=1 debug=0x2a");
2552 LogRel(("EFI: boot args = %s\n", pThisCC->szBootArgs));
2553
2554 /*
2555 * Get device props.
2556 */
2557 char *pszDeviceProps;
2558 rc = pHlp->pfnCFGMQueryStringAllocDef(pCfg, "DeviceProps", &pszDeviceProps, NULL);
2559 if (RT_FAILURE(rc))
2560 return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS,
2561 N_("Configuration error: Querying \"DeviceProps\" as a string failed"));
2562 if (pszDeviceProps)
2563 {
2564 LogRel(("EFI: device props = %s\n", pszDeviceProps));
2565 rc = efiParseDeviceString(pThisCC, pszDeviceProps);
2566 PDMDevHlpMMHeapFree(pDevIns, pszDeviceProps);
2567 if (RT_FAILURE(rc))
2568 return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS,
2569 N_("Configuration error: Cannot parse device properties"));
2570 }
2571 else
2572 {
2573 pThisCC->pbDeviceProps = NULL;
2574 pThisCC->cbDeviceProps = 0;
2575 }
2576
2577 /*
2578 * CPU frequencies.
2579 */
2580 pThisCC->u64TscFrequency = TMCpuTicksPerSecond(PDMDevHlpGetVM(pDevIns));
2581 pThisCC->u64CpuFrequency = pThisCC->u64TscFrequency;
2582 pThisCC->u64FsbFrequency = CPUMGetGuestScalableBusFrequency(PDMDevHlpGetVM(pDevIns));
2583
2584 /*
2585 * EFI graphics mode (with new EFI VGA code used only as a fallback, for
2586 * old EFI VGA code the only way to select the GOP mode).
2587 */
2588 rc = pHlp->pfnCFGMQueryU32Def(pCfg, "GraphicsMode", &pThisCC->u32GraphicsMode, UINT32_MAX);
2589 if (RT_FAILURE(rc))
2590 return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS,
2591 N_("Configuration error: Querying \"GraphicsMode\" as a 32-bit int failed"));
2592 if (pThisCC->u32GraphicsMode == UINT32_MAX)
2593 {
2594 /* get the legacy value if nothing else was specified */
2595 rc = pHlp->pfnCFGMQueryU32Def(pCfg, "GopMode", &pThisCC->u32GraphicsMode, UINT32_MAX);
2596 if (RT_FAILURE(rc))
2597 return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS,
2598 N_("Configuration error: Querying \"GopMode\" as a 32-bit int failed"));
2599 }
2600 if (pThisCC->u32GraphicsMode == UINT32_MAX)
2601 pThisCC->u32GraphicsMode = 2; /* 1024x768, at least typically */
2602
2603 /*
2604 * EFI graphics resolution, defaults to 1024x768 (used to be UGA only, now
2605 * is the main config setting as the mode number is so hard to predict).
2606 */
2607 char szResolution[16];
2608 rc = pHlp->pfnCFGMQueryStringDef(pCfg, "GraphicsResolution", szResolution, sizeof(szResolution), "");
2609 if (RT_FAILURE(rc))
2610 return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS,
2611 N_("Configuration error: Querying \"GraphicsResolution\" as a string failed"));
2612 if (szResolution[0])
2613 {
2614 const char *pszX = RTStrStr(szResolution, "x");
2615 if (pszX)
2616 {
2617 pThisCC->u32HorizontalResolution = RTStrToUInt32(szResolution);
2618 pThisCC->u32VerticalResolution = RTStrToUInt32(pszX + 1);
2619 }
2620 }
2621 else
2622 {
2623 /* get the legacy values if nothing else was specified */
2624 rc = pHlp->pfnCFGMQueryU32Def(pCfg, "UgaHorizontalResolution", &pThisCC->u32HorizontalResolution, 0);
2625 AssertRCReturn(rc, rc);
2626 rc = pHlp->pfnCFGMQueryU32Def(pCfg, "UgaVerticalResolution", &pThisCC->u32VerticalResolution, 0);
2627 AssertRCReturn(rc, rc);
2628 }
2629 if (pThisCC->u32HorizontalResolution == 0 || pThisCC->u32VerticalResolution == 0)
2630 {
2631 pThisCC->u32HorizontalResolution = 1024;
2632 pThisCC->u32VerticalResolution = 768;
2633 }
2634
2635 pThisCC->pszNvramFile = NULL;
2636 rc = pHlp->pfnCFGMQueryStringAlloc(pCfg, "NvramFile", &pThisCC->pszNvramFile);
2637 if (RT_FAILURE(rc) && rc != VERR_CFGM_VALUE_NOT_FOUND)
2638 return PDMDEV_SET_ERROR(pDevIns, rc,
2639 N_("Configuration error: Querying \"NvramFile\" as a string failed"));
2640
2641 /*
2642 * Load firmware volume and thunk ROM.
2643 */
2644 rc = efiLoadRom(pDevIns, pThis, pThisCC, pCfg);
2645 if (RT_FAILURE(rc))
2646 return rc;
2647
2648 /*
2649 * Register our I/O ports.
2650 */
2651 rc = PDMDevHlpIoPortCreateFlagsAndMap(pDevIns, EFI_PORT_BASE, EFI_PORT_COUNT, IOM_IOPORT_F_ABS,
2652 efiR3IoPortWrite, efiR3IoPortRead,
2653 "EFI communication ports", NULL /*paExtDescs*/, &pThis->hIoPorts);
2654 AssertRCReturn(rc, rc);
2655
2656 /*
2657 * Plant DMI and MPS tables in the ROM region.
2658 */
2659 rc = FwCommonPlantDMITable(pDevIns, pThisCC->au8DMIPage, VBOX_DMI_TABLE_SIZE, &pThisCC->aUuid,
2660 pDevIns->pCfg, pThisCC->cCpus, &pThisCC->cbDmiTables, &pThisCC->cNumDmiTables,
2661 true /*fUefi*/);
2662 AssertRCReturn(rc, rc);
2663
2664 /*
2665 * NB: VBox/Devices/EFI/Firmware/VBoxPkg/VBoxSysTables/VBoxSysTables.c scans memory for
2666 * the SMBIOS header. The header must be placed in a range that EFI will scan.
2667 */
2668 FwCommonPlantSmbiosAndDmiHdrs(pDevIns, pThisCC->au8DMIPage + VBOX_DMI_TABLE_SIZE,
2669 pThisCC->cbDmiTables, pThisCC->cNumDmiTables);
2670
2671 if (pThisCC->u8IOAPIC)
2672 {
2673 FwCommonPlantMpsTable(pDevIns,
2674 pThisCC->au8DMIPage /* aka VBOX_DMI_TABLE_BASE */ + VBOX_DMI_TABLE_SIZE + VBOX_DMI_HDR_SIZE,
2675 _4K - VBOX_DMI_TABLE_SIZE - VBOX_DMI_HDR_SIZE, pThisCC->cCpus);
2676 FwCommonPlantMpsFloatPtr(pDevIns, VBOX_DMI_TABLE_BASE + VBOX_DMI_TABLE_SIZE + VBOX_DMI_HDR_SIZE);
2677 }
2678
2679 rc = PDMDevHlpROMRegister(pDevIns, VBOX_DMI_TABLE_BASE, _4K, pThisCC->au8DMIPage, _4K,
2680 PGMPHYS_ROM_FLAGS_PERMANENT_BINARY, "DMI tables");
2681
2682 AssertRCReturn(rc, rc);
2683
2684 /*
2685 * Register info handlers.
2686 */
2687 rc = PDMDevHlpDBGFInfoRegister(pDevIns, "nvram", "Dumps the NVRAM variables.\n", efiInfoNvram);
2688 AssertRCReturn(rc, rc);
2689
2690 /*
2691 * Call reset to set things up.
2692 */
2693 efiReset(pDevIns);
2694
2695 return VINF_SUCCESS;
2696}
2697
2698#else /* IN_RING3 */
2699
2700
2701/**
2702 * @callback_method_impl{PDMDEVREGR0,pfnConstruct}
2703 */
2704static DECLCALLBACK(int) efiRZConstruct(PPDMDEVINS pDevIns)
2705{
2706 PDMDEV_CHECK_VERSIONS_RETURN(pDevIns);
2707 PDEVEFI pThis = PDMDEVINS_2_DATA(pDevIns, PDEVEFI);
2708
2709# if 1
2710 int rc = PDMDevHlpMmioSetUpContext(pDevIns, pThis->hMmioFlash, efiR3NvMmioWrite, efiR3NvMmioRead, NULL /*pvUser*/);
2711 AssertRCReturn(rc, rc);
2712# else
2713 RT_NOREF(pDevIns, pThis); (void)&efiR3NvMmioRead; (void)&efiR3NvMmioWrite;
2714# endif
2715
2716 return VINF_SUCCESS;
2717}
2718
2719
2720#endif /* IN_RING3 */
2721
2722/**
2723 * The device registration structure.
2724 */
2725const PDMDEVREG g_DeviceEFI =
2726{
2727 /* .u32Version = */ PDM_DEVREG_VERSION,
2728 /* .uReserved0 = */ 0,
2729 /* .szName = */ "efi",
2730 /* .fFlags = */ PDM_DEVREG_FLAGS_DEFAULT_BITS | PDM_DEVREG_FLAGS_RZ | PDM_DEVREG_FLAGS_NEW_STYLE,
2731 /* .fClass = */ PDM_DEVREG_CLASS_ARCH_BIOS,
2732 /* .cMaxInstances = */ 1,
2733 /* .uSharedVersion = */ 42,
2734 /* .cbInstanceShared = */ sizeof(DEVEFI),
2735 /* .cbInstanceCC = */ sizeof(DEVEFICC),
2736 /* .cbInstanceRC = */ sizeof(DEVEFIRC),
2737 /* .cMaxPciDevices = */ 0,
2738 /* .cMaxMsixVectors = */ 0,
2739 /* .pszDescription = */ "Extensible Firmware Interface Device.\n"
2740 "LUN#0 - NVRAM port",
2741#if defined(IN_RING3)
2742 /* .pszRCMod = */ "VBoxDDRC.rc",
2743 /* .pszR0Mod = */ "VBoxDDR0.r0",
2744 /* .pfnConstruct = */ efiConstruct,
2745 /* .pfnDestruct = */ efiDestruct,
2746 /* .pfnRelocate = */ NULL,
2747 /* .pfnMemSetup = */ efiMemSetup,
2748 /* .pfnPowerOn = */ NULL,
2749 /* .pfnReset = */ efiReset,
2750 /* .pfnSuspend = */ NULL,
2751 /* .pfnResume = */ NULL,
2752 /* .pfnAttach = */ NULL,
2753 /* .pfnDetach = */ NULL,
2754 /* .pfnQueryInterface = */ NULL,
2755 /* .pfnInitComplete = */ efiInitComplete,
2756 /* .pfnPowerOff = */ efiPowerOff,
2757 /* .pfnSoftReset = */ NULL,
2758 /* .pfnReserved0 = */ NULL,
2759 /* .pfnReserved1 = */ NULL,
2760 /* .pfnReserved2 = */ NULL,
2761 /* .pfnReserved3 = */ NULL,
2762 /* .pfnReserved4 = */ NULL,
2763 /* .pfnReserved5 = */ NULL,
2764 /* .pfnReserved6 = */ NULL,
2765 /* .pfnReserved7 = */ NULL,
2766#elif defined(IN_RING0)
2767 /* .pfnEarlyConstruct = */ NULL,
2768 /* .pfnConstruct = */ efiRZConstruct,
2769 /* .pfnDestruct = */ NULL,
2770 /* .pfnFinalDestruct = */ NULL,
2771 /* .pfnRequest = */ NULL,
2772 /* .pfnReserved0 = */ NULL,
2773 /* .pfnReserved1 = */ NULL,
2774 /* .pfnReserved2 = */ NULL,
2775 /* .pfnReserved3 = */ NULL,
2776 /* .pfnReserved4 = */ NULL,
2777 /* .pfnReserved5 = */ NULL,
2778 /* .pfnReserved6 = */ NULL,
2779 /* .pfnReserved7 = */ NULL,
2780#elif defined(IN_RC)
2781 /* .pfnConstruct = */ efiRZConstruct,
2782 /* .pfnReserved0 = */ NULL,
2783 /* .pfnReserved1 = */ NULL,
2784 /* .pfnReserved2 = */ NULL,
2785 /* .pfnReserved3 = */ NULL,
2786 /* .pfnReserved4 = */ NULL,
2787 /* .pfnReserved5 = */ NULL,
2788 /* .pfnReserved6 = */ NULL,
2789 /* .pfnReserved7 = */ NULL,
2790#else
2791# error "Not in IN_RING3, IN_RING0 or IN_RC!"
2792#endif
2793 /* .u32VersionEnd = */ PDM_DEVREG_VERSION
2794};
2795
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette