]> git.proxmox.com Git - mirror_edk2.git/blame - ArmPlatformPkg/Bds/Bds.c
ArmPlatformPkg/Bds: Clarify assert message with debug information
[mirror_edk2.git] / ArmPlatformPkg / Bds / Bds.c
CommitLineData
705b0c03 1/** @file\r
2*\r
b51cffe1 3* Copyright (c) 2011-2015, ARM Limited. All rights reserved.\r
705b0c03 4*\r
c0b2e477 5* This program and the accompanying materials\r
6* are licensed and made available under the terms and conditions of the BSD License\r
7* which accompanies this distribution. The full text of the license may be found at\r
8* http://opensource.org/licenses/bsd-license.php\r
9*\r
10* THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,\r
11* WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.\r
705b0c03 12*\r
13**/\r
14\r
15#include "BdsInternal.h"\r
16\r
17#include <Library/PcdLib.h>\r
18#include <Library/PerformanceLib.h>\r
19\r
20#include <Protocol/Bds.h>\r
21\r
ae190039
OM
22#include <Guid/EventGroup.h>\r
23\r
705b0c03 24#define EFI_SET_TIMER_TO_SECOND 10000000\r
25\r
705b0c03 26STATIC\r
27EFI_STATUS\r
28GetConsoleDevicePathFromVariable (\r
29 IN CHAR16* ConsoleVarName,\r
30 IN CHAR16* DefaultConsolePaths,\r
31 OUT EFI_DEVICE_PATH** DevicePaths\r
32 )\r
33{\r
34 EFI_STATUS Status;\r
35 UINTN Size;\r
36 EFI_DEVICE_PATH_PROTOCOL* DevicePathInstances;\r
37 EFI_DEVICE_PATH_PROTOCOL* DevicePathInstance;\r
38 CHAR16* DevicePathStr;\r
39 CHAR16* NextDevicePathStr;\r
40 EFI_DEVICE_PATH_FROM_TEXT_PROTOCOL *EfiDevicePathFromTextProtocol;\r
41\r
c0b2e477 42 Status = GetGlobalEnvironmentVariable (ConsoleVarName, NULL, NULL, (VOID**)&DevicePathInstances);\r
705b0c03 43 if (EFI_ERROR(Status)) {\r
483bc3d3 44 // In case no default console device path has been defined we assume a driver handles the console (eg: SimpleTextInOutSerial)\r
45 if ((DefaultConsolePaths == NULL) || (DefaultConsolePaths[0] == L'\0')) {\r
46 *DevicePaths = NULL;\r
47 return EFI_SUCCESS;\r
48 }\r
49\r
705b0c03 50 Status = gBS->LocateProtocol (&gEfiDevicePathFromTextProtocolGuid, NULL, (VOID **)&EfiDevicePathFromTextProtocol);\r
51 ASSERT_EFI_ERROR(Status);\r
52\r
53 DevicePathInstances = NULL;\r
54\r
55 // Extract the Device Path instances from the multi-device path string\r
56 while ((DefaultConsolePaths != NULL) && (DefaultConsolePaths[0] != L'\0')) {\r
57 NextDevicePathStr = StrStr (DefaultConsolePaths, L";");\r
58 if (NextDevicePathStr == NULL) {\r
59 DevicePathStr = DefaultConsolePaths;\r
60 DefaultConsolePaths = NULL;\r
61 } else {\r
62 DevicePathStr = (CHAR16*)AllocateCopyPool ((NextDevicePathStr - DefaultConsolePaths + 1) * sizeof(CHAR16), DefaultConsolePaths);\r
63 *(DevicePathStr + (NextDevicePathStr - DefaultConsolePaths)) = L'\0';\r
64 DefaultConsolePaths = NextDevicePathStr;\r
65 if (DefaultConsolePaths[0] == L';') {\r
66 DefaultConsolePaths++;\r
67 }\r
68 }\r
69\r
70 DevicePathInstance = EfiDevicePathFromTextProtocol->ConvertTextToDevicePath (DevicePathStr);\r
71 ASSERT(DevicePathInstance != NULL);\r
72 DevicePathInstances = AppendDevicePathInstance (DevicePathInstances, DevicePathInstance);\r
73\r
74 if (NextDevicePathStr != NULL) {\r
75 FreePool (DevicePathStr);\r
76 }\r
77 FreePool (DevicePathInstance);\r
78 }\r
79\r
80 // Set the environment variable with this device path multi-instances\r
81 Size = GetDevicePathSize (DevicePathInstances);\r
82 if (Size > 0) {\r
83 gRT->SetVariable (\r
84 ConsoleVarName,\r
85 &gEfiGlobalVariableGuid,\r
86 EFI_VARIABLE_NON_VOLATILE | EFI_VARIABLE_BOOTSERVICE_ACCESS | EFI_VARIABLE_RUNTIME_ACCESS,\r
87 Size,\r
88 DevicePathInstances\r
89 );\r
90 } else {\r
91 Status = EFI_INVALID_PARAMETER;\r
92 }\r
93 }\r
94\r
95 if (!EFI_ERROR(Status)) {\r
96 *DevicePaths = DevicePathInstances;\r
97 }\r
483bc3d3 98 return Status;\r
705b0c03 99}\r
100\r
101STATIC\r
102EFI_STATUS\r
103InitializeConsolePipe (\r
104 IN EFI_DEVICE_PATH *ConsoleDevicePaths,\r
105 IN EFI_GUID *Protocol,\r
106 OUT EFI_HANDLE *Handle,\r
107 OUT VOID* *Interface\r
108 )\r
109{\r
110 EFI_STATUS Status;\r
111 UINTN Size;\r
112 UINTN NoHandles;\r
113 EFI_HANDLE *Buffer;\r
114 EFI_DEVICE_PATH_PROTOCOL* DevicePath;\r
115\r
116 // Connect all the Device Path Consoles\r
483bc3d3 117 while (ConsoleDevicePaths != NULL) {\r
705b0c03 118 DevicePath = GetNextDevicePathInstance (&ConsoleDevicePaths, &Size);\r
119\r
120 Status = BdsConnectDevicePath (DevicePath, Handle, NULL);\r
121 DEBUG_CODE_BEGIN();\r
122 if (EFI_ERROR(Status)) {\r
123 // We convert back to the text representation of the device Path\r
124 EFI_DEVICE_PATH_TO_TEXT_PROTOCOL* DevicePathToTextProtocol;\r
125 CHAR16* DevicePathTxt;\r
126 EFI_STATUS Status;\r
c0b2e477 127\r
705b0c03 128 Status = gBS->LocateProtocol(&gEfiDevicePathToTextProtocolGuid, NULL, (VOID **)&DevicePathToTextProtocol);\r
129 if (!EFI_ERROR(Status)) {\r
130 DevicePathTxt = DevicePathToTextProtocol->ConvertDevicePathToText (DevicePath, TRUE, TRUE);\r
131\r
132 DEBUG((EFI_D_ERROR,"Fail to start the console with the Device Path '%s'. (Error '%r')\n", DevicePathTxt, Status));\r
133\r
134 FreePool (DevicePathTxt);\r
135 }\r
136 }\r
137 DEBUG_CODE_END();\r
138\r
139 // If the console splitter driver is not supported by the platform then use the first Device Path\r
140 // instance for the console interface.\r
141 if (!EFI_ERROR(Status) && (*Interface == NULL)) {\r
142 Status = gBS->HandleProtocol (*Handle, Protocol, Interface);\r
143 }\r
483bc3d3 144 }\r
705b0c03 145\r
146 // No Device Path has been defined for this console interface. We take the first protocol implementation\r
147 if (*Interface == NULL) {\r
148 Status = gBS->LocateHandleBuffer (ByProtocol, Protocol, NULL, &NoHandles, &Buffer);\r
149 if (EFI_ERROR (Status)) {\r
150 BdsConnectAllDrivers();\r
151 Status = gBS->LocateHandleBuffer (ByProtocol, Protocol, NULL, &NoHandles, &Buffer);\r
152 }\r
153\r
154 if (!EFI_ERROR(Status)) {\r
155 *Handle = Buffer[0];\r
156 Status = gBS->HandleProtocol (*Handle, Protocol, Interface);\r
157 ASSERT_EFI_ERROR(Status);\r
3a2b37f0 158 FreePool (Buffer);\r
705b0c03 159 }\r
705b0c03 160 } else {\r
161 Status = EFI_SUCCESS;\r
162 }\r
163\r
164 return Status;\r
165}\r
166\r
167EFI_STATUS\r
168InitializeConsole (\r
169 VOID\r
170 )\r
171{\r
172 EFI_STATUS Status;\r
173 EFI_DEVICE_PATH* ConOutDevicePaths;\r
174 EFI_DEVICE_PATH* ConInDevicePaths;\r
175 EFI_DEVICE_PATH* ConErrDevicePaths;\r
176\r
177 // By getting the Console Device Paths from the environment variables before initializing the console pipe, we\r
178 // create the 3 environment variables (ConIn, ConOut, ConErr) that allows to initialize all the console interface\r
179 // of newly installed console drivers\r
483bc3d3 180 Status = GetConsoleDevicePathFromVariable (L"ConOut", (CHAR16*)PcdGetPtr(PcdDefaultConOutPaths), &ConOutDevicePaths);\r
705b0c03 181 ASSERT_EFI_ERROR (Status);\r
483bc3d3 182 Status = GetConsoleDevicePathFromVariable (L"ConIn", (CHAR16*)PcdGetPtr(PcdDefaultConInPaths), &ConInDevicePaths);\r
705b0c03 183 ASSERT_EFI_ERROR (Status);\r
30d1bcd1 184 Status = GetConsoleDevicePathFromVariable (L"ErrOut", (CHAR16*)PcdGetPtr(PcdDefaultConOutPaths), &ConErrDevicePaths);\r
705b0c03 185 ASSERT_EFI_ERROR (Status);\r
186\r
187 // Initialize the Consoles\r
188 Status = InitializeConsolePipe (ConOutDevicePaths, &gEfiSimpleTextOutProtocolGuid, &gST->ConsoleOutHandle, (VOID **)&gST->ConOut);\r
189 ASSERT_EFI_ERROR (Status);\r
190 Status = InitializeConsolePipe (ConInDevicePaths, &gEfiSimpleTextInProtocolGuid, &gST->ConsoleInHandle, (VOID **)&gST->ConIn);\r
191 ASSERT_EFI_ERROR (Status);\r
192 Status = InitializeConsolePipe (ConErrDevicePaths, &gEfiSimpleTextOutProtocolGuid, &gST->StandardErrorHandle, (VOID **)&gST->StdErr);\r
193 if (EFI_ERROR(Status)) {\r
194 // In case of error, we reuse the console output for the error output\r
195 gST->StandardErrorHandle = gST->ConsoleOutHandle;\r
196 gST->StdErr = gST->ConOut;\r
197 }\r
198\r
b148591a 199 // Free Memory allocated for reading the UEFI Variables\r
200 if (ConOutDevicePaths) {\r
201 FreePool (ConOutDevicePaths);\r
202 }\r
203 if (ConInDevicePaths) {\r
204 FreePool (ConInDevicePaths);\r
205 }\r
206 if (ConErrDevicePaths) {\r
207 FreePool (ConErrDevicePaths);\r
208 }\r
209\r
705b0c03 210 return EFI_SUCCESS;\r
211}\r
212\r
213EFI_STATUS\r
214DefineDefaultBootEntries (\r
215 VOID\r
216 )\r
217{\r
218 BDS_LOAD_OPTION* BdsLoadOption;\r
219 UINTN Size;\r
220 EFI_STATUS Status;\r
221 EFI_DEVICE_PATH_FROM_TEXT_PROTOCOL* EfiDevicePathFromTextProtocol;\r
222 EFI_DEVICE_PATH* BootDevicePath;\r
90a44ec4
OM
223 UINT8* OptionalData;\r
224 UINTN OptionalDataSize;\r
705b0c03 225 ARM_BDS_LOADER_ARGUMENTS* BootArguments;\r
226 ARM_BDS_LOADER_TYPE BootType;\r
227 EFI_DEVICE_PATH* InitrdPath;\r
705b0c03 228 UINTN InitrdSize;\r
9fc9aa46
OM
229 UINTN CmdLineSize;\r
230 UINTN CmdLineAsciiSize;\r
231 CHAR16* DefaultBootArgument;\r
232 CHAR8* AsciiDefaultBootArgument;\r
705b0c03 233\r
234 //\r
235 // If Boot Order does not exist then create a default entry\r
236 //\r
237 Size = 0;\r
238 Status = gRT->GetVariable (L"BootOrder", &gEfiGlobalVariableGuid, NULL, &Size, NULL);\r
239 if (Status == EFI_NOT_FOUND) {\r
240 if ((PcdGetPtr(PcdDefaultBootDevicePath) == NULL) || (StrLen ((CHAR16*)PcdGetPtr(PcdDefaultBootDevicePath)) == 0)) {\r
241 return EFI_UNSUPPORTED;\r
242 }\r
243\r
244 Status = gBS->LocateProtocol (&gEfiDevicePathFromTextProtocolGuid, NULL, (VOID **)&EfiDevicePathFromTextProtocol);\r
245 if (EFI_ERROR(Status)) {\r
246 // You must provide an implementation of DevicePathFromTextProtocol in your firmware (eg: DevicePathDxe)\r
247 DEBUG((EFI_D_ERROR,"Error: Bds requires DevicePathFromTextProtocol\n"));\r
248 return Status;\r
249 }\r
250 BootDevicePath = EfiDevicePathFromTextProtocol->ConvertTextToDevicePath ((CHAR16*)PcdGetPtr(PcdDefaultBootDevicePath));\r
251\r
252 DEBUG_CODE_BEGIN();\r
253 // We convert back to the text representation of the device Path to see if the initial text is correct\r
254 EFI_DEVICE_PATH_TO_TEXT_PROTOCOL* DevicePathToTextProtocol;\r
255 CHAR16* DevicePathTxt;\r
256\r
257 Status = gBS->LocateProtocol(&gEfiDevicePathToTextProtocolGuid, NULL, (VOID **)&DevicePathToTextProtocol);\r
258 ASSERT_EFI_ERROR(Status);\r
259 DevicePathTxt = DevicePathToTextProtocol->ConvertDevicePathToText (BootDevicePath, TRUE, TRUE);\r
260\r
b51cffe1
OM
261 if (StrCmp ((CHAR16*)PcdGetPtr (PcdDefaultBootDevicePath), DevicePathTxt) != 0) {\r
262 DEBUG ((EFI_D_ERROR, "Device Path given: '%s' Device Path expected: '%s'\n",\r
263 (CHAR16*)PcdGetPtr (PcdDefaultBootDevicePath), DevicePathTxt));\r
264 ASSERT_EFI_ERROR (EFI_INVALID_PARAMETER);\r
265 }\r
705b0c03 266\r
267 FreePool (DevicePathTxt);\r
268 DEBUG_CODE_END();\r
269\r
270 // Create the entry is the Default values are correct\r
271 if (BootDevicePath != NULL) {\r
272 BootType = (ARM_BDS_LOADER_TYPE)PcdGet32 (PcdDefaultBootType);\r
273\r
9fc9aa46
OM
274 // We do not support NULL pointer\r
275 ASSERT (PcdGetPtr (PcdDefaultBootArgument) != NULL);\r
276\r
277 //\r
278 // Logic to handle ASCII or Unicode default parameters\r
279 //\r
280 if (*(CHAR8*)PcdGetPtr (PcdDefaultBootArgument) == '\0') {\r
281 CmdLineSize = 0;\r
282 CmdLineAsciiSize = 0;\r
283 DefaultBootArgument = NULL;\r
284 AsciiDefaultBootArgument = NULL;\r
285 } else if (IsUnicodeString ((CHAR16*)PcdGetPtr (PcdDefaultBootArgument))) {\r
286 // The command line is a Unicode string\r
287 DefaultBootArgument = (CHAR16*)PcdGetPtr (PcdDefaultBootArgument);\r
288 CmdLineSize = StrSize (DefaultBootArgument);\r
289\r
290 // Initialize ASCII variables\r
291 CmdLineAsciiSize = CmdLineSize / 2;\r
292 AsciiDefaultBootArgument = AllocatePool (CmdLineAsciiSize);\r
293 if (AsciiDefaultBootArgument == NULL) {\r
294 return EFI_OUT_OF_RESOURCES;\r
295 }\r
296 UnicodeStrToAsciiStr ((CHAR16*)PcdGetPtr (PcdDefaultBootArgument), AsciiDefaultBootArgument);\r
297 } else {\r
298 // The command line is a ASCII string\r
299 AsciiDefaultBootArgument = (CHAR8*)PcdGetPtr (PcdDefaultBootArgument);\r
300 CmdLineAsciiSize = AsciiStrSize (AsciiDefaultBootArgument);\r
301\r
302 // Initialize ASCII variables\r
303 CmdLineSize = CmdLineAsciiSize * 2;\r
304 DefaultBootArgument = AllocatePool (CmdLineSize);\r
305 if (DefaultBootArgument == NULL) {\r
306 return EFI_OUT_OF_RESOURCES;\r
307 }\r
308 AsciiStrToUnicodeStr (AsciiDefaultBootArgument, DefaultBootArgument);\r
309 }\r
310\r
705b0c03 311 if ((BootType == BDS_LOADER_KERNEL_LINUX_ATAG) || (BootType == BDS_LOADER_KERNEL_LINUX_FDT)) {\r
705b0c03 312 InitrdPath = EfiDevicePathFromTextProtocol->ConvertTextToDevicePath ((CHAR16*)PcdGetPtr(PcdDefaultBootInitrdPath));\r
313 InitrdSize = GetDevicePathSize (InitrdPath);\r
314\r
90a44ec4
OM
315 OptionalDataSize = sizeof(ARM_BDS_LOADER_ARGUMENTS) + CmdLineAsciiSize + InitrdSize;\r
316 BootArguments = (ARM_BDS_LOADER_ARGUMENTS*)AllocatePool (OptionalDataSize);\r
9fc9aa46
OM
317 if (BootArguments == NULL) {\r
318 return EFI_OUT_OF_RESOURCES;\r
319 }\r
320 BootArguments->LinuxArguments.CmdLineSize = CmdLineAsciiSize;\r
705b0c03 321 BootArguments->LinuxArguments.InitrdSize = InitrdSize;\r
322\r
9fc9aa46
OM
323 CopyMem ((VOID*)(BootArguments + 1), AsciiDefaultBootArgument, CmdLineAsciiSize);\r
324 CopyMem ((VOID*)((UINTN)(BootArguments + 1) + CmdLineAsciiSize), InitrdPath, InitrdSize);\r
90a44ec4
OM
325\r
326 OptionalData = (UINT8*)BootArguments;\r
705b0c03 327 } else {\r
06044819
OM
328 OptionalData = (UINT8*)DefaultBootArgument;\r
329 OptionalDataSize = CmdLineSize;\r
705b0c03 330 }\r
331\r
332 BootOptionCreate (LOAD_OPTION_ACTIVE | LOAD_OPTION_CATEGORY_BOOT,\r
333 (CHAR16*)PcdGetPtr(PcdDefaultBootDescription),\r
334 BootDevicePath,\r
335 BootType,\r
90a44ec4
OM
336 OptionalData,\r
337 OptionalDataSize,\r
705b0c03 338 &BdsLoadOption\r
339 );\r
340 FreePool (BdsLoadOption);\r
9fc9aa46
OM
341\r
342 if (DefaultBootArgument == (CHAR16*)PcdGetPtr (PcdDefaultBootArgument)) {\r
343 FreePool (AsciiDefaultBootArgument);\r
0aa39d28 344 } else if (DefaultBootArgument != NULL) {\r
9fc9aa46
OM
345 FreePool (DefaultBootArgument);\r
346 }\r
705b0c03 347 } else {\r
348 Status = EFI_UNSUPPORTED;\r
349 }\r
350 }\r
351\r
66982010 352 return Status;\r
705b0c03 353}\r
354\r
355EFI_STATUS\r
356StartDefaultBootOnTimeout (\r
357 VOID\r
358 )\r
359{\r
360 UINTN Size;\r
361 UINT16 Timeout;\r
362 UINT16 *TimeoutPtr;\r
363 EFI_EVENT WaitList[2];\r
364 UINTN WaitIndex;\r
365 UINT16 *BootOrder;\r
366 UINTN BootOrderSize;\r
367 UINTN Index;\r
368 CHAR16 BootVariableName[9];\r
b34e4db3 369 EFI_STATUS Status;\r
370 EFI_INPUT_KEY Key;\r
705b0c03 371\r
372 Size = sizeof(UINT16);\r
373 Timeout = (UINT16)PcdGet16 (PcdPlatformBootTimeOut);\r
13354807
HL
374 Status = GetGlobalEnvironmentVariable (L"Timeout", &Timeout, &Size, (VOID**)&TimeoutPtr);\r
375 if (!EFI_ERROR (Status)) {\r
376 Timeout = *TimeoutPtr;\r
377 FreePool (TimeoutPtr);\r
378 }\r
705b0c03 379\r
380 if (Timeout != 0xFFFF) {\r
381 if (Timeout > 0) {\r
382 // Create the waiting events (keystroke and 1sec timer)\r
383 gBS->CreateEvent (EVT_TIMER, 0, NULL, NULL, &WaitList[0]);\r
384 gBS->SetTimer (WaitList[0], TimerPeriodic, EFI_SET_TIMER_TO_SECOND);\r
385 WaitList[1] = gST->ConIn->WaitForKey;\r
386\r
387 // Start the timer\r
388 WaitIndex = 0;\r
389 Print(L"The default boot selection will start in ");\r
390 while ((Timeout > 0) && (WaitIndex == 0)) {\r
391 Print(L"%3d seconds",Timeout);\r
392 gBS->WaitForEvent (2, WaitList, &WaitIndex);\r
393 if (WaitIndex == 0) {\r
394 Print(L"\b\b\b\b\b\b\b\b\b\b\b");\r
395 Timeout--;\r
396 }\r
397 }\r
398 // Discard key in the buffer\r
399 do {\r
91c38d4e 400 Status = gST->ConIn->ReadKeyStroke (gST->ConIn, &Key);\r
705b0c03 401 } while(!EFI_ERROR(Status));\r
402 gBS->CloseEvent (WaitList[0]);\r
403 Print(L"\n\r");\r
404 }\r
405\r
406 // In case of Timeout we start the default boot selection\r
407 if (Timeout == 0) {\r
408 // Get the Boot Option Order from the environment variable (a default value should have been created)\r
c0b2e477 409 GetGlobalEnvironmentVariable (L"BootOrder", NULL, &BootOrderSize, (VOID**)&BootOrder);\r
705b0c03 410\r
411 for (Index = 0; Index < BootOrderSize / sizeof (UINT16); Index++) {\r
412 UnicodeSPrint (BootVariableName, 9 * sizeof(CHAR16), L"Boot%04X", BootOrder[Index]);\r
413 Status = BdsStartBootOption (BootVariableName);\r
414 if(!EFI_ERROR(Status)){\r
91c38d4e
RC
415 // Boot option returned successfully, hence don't need to start next boot option\r
416 break;\r
705b0c03 417 }\r
418 // In case of success, we should not return from this call.\r
419 }\r
b148591a 420 FreePool (BootOrder);\r
705b0c03 421 }\r
422 }\r
423 return EFI_SUCCESS;\r
424}\r
425\r
ae190039
OM
426/**\r
427 An empty function to pass error checking of CreateEventEx ().\r
428\r
429 @param Event Event whose notification function is being invoked.\r
430 @param Context Pointer to the notification function's context,\r
431 which is implementation-dependent.\r
432\r
433**/\r
ae190039
OM
434VOID\r
435EFIAPI\r
436EmptyCallbackFunction (\r
437 IN EFI_EVENT Event,\r
438 IN VOID *Context\r
439 )\r
440{\r
441 return;\r
442}\r
443\r
705b0c03 444/**\r
c0b2e477 445 This function uses policy data from the platform to determine what operating\r
446 system or system utility should be loaded and invoked. This function call\r
447 also optionally make the use of user input to determine the operating system\r
448 or system utility to be loaded and invoked. When the DXE Core has dispatched\r
449 all the drivers on the dispatch queue, this function is called. This\r
450 function will attempt to connect the boot devices required to load and invoke\r
451 the selected operating system or system utility. During this process,\r
452 additional firmware volumes may be discovered that may contain addition DXE\r
453 drivers that can be dispatched by the DXE Core. If a boot device cannot be\r
454 fully connected, this function calls the DXE Service Dispatch() to allow the\r
455 DXE drivers from any newly discovered firmware volumes to be dispatched.\r
456 Then the boot device connection can be attempted again. If the same boot\r
457 device connection operation fails twice in a row, then that boot device has\r
705b0c03 458 failed, and should be skipped. This function should never return.\r
459\r
460 @param This The EFI_BDS_ARCH_PROTOCOL instance.\r
461\r
462 @return None.\r
463\r
464**/\r
465VOID\r
466EFIAPI\r
467BdsEntry (\r
468 IN EFI_BDS_ARCH_PROTOCOL *This\r
469 )\r
470{\r
471 UINTN Size;\r
472 EFI_STATUS Status;\r
0e29bf5c 473 UINT16 *BootNext;\r
474 UINTN BootNextSize;\r
475 CHAR16 BootVariableName[9];\r
ae190039
OM
476 EFI_EVENT EndOfDxeEvent;\r
477\r
478 //\r
479 // Signal EndOfDxe PI Event\r
480 //\r
481 Status = gBS->CreateEventEx (\r
482 EVT_NOTIFY_SIGNAL,\r
483 TPL_NOTIFY,\r
484 EmptyCallbackFunction,\r
485 NULL,\r
486 &gEfiEndOfDxeEventGroupGuid,\r
487 &EndOfDxeEvent\r
488 );\r
489 if (!EFI_ERROR (Status)) {\r
490 gBS->SignalEvent (EndOfDxeEvent);\r
491 }\r
705b0c03 492\r
493 PERF_END (NULL, "DXE", NULL, 0);\r
494\r
495 //\r
496 // Declare the Firmware Vendor\r
497 //\r
498 if (FixedPcdGetPtr(PcdFirmwareVendor) != NULL) {\r
499 Size = 0x100;\r
500 gST->FirmwareVendor = AllocateRuntimePool (Size);\r
501 ASSERT (gST->FirmwareVendor != NULL);\r
502 UnicodeSPrint (gST->FirmwareVendor, Size, L"%a EFI %a %a", PcdGetPtr(PcdFirmwareVendor), __DATE__, __TIME__);\r
503 }\r
504\r
53cba4fe 505 //\r
506 // Fixup Table CRC after we updated Firmware Vendor\r
507 //\r
508 gST->Hdr.CRC32 = 0;\r
509 Status = gBS->CalculateCrc32 ((VOID*)gST, gST->Hdr.HeaderSize, &gST->Hdr.CRC32);\r
510 ASSERT_EFI_ERROR (Status);\r
511\r
705b0c03 512 // If BootNext environment variable is defined then we just load it !\r
0e29bf5c 513 BootNextSize = sizeof(UINT16);\r
c0b2e477 514 Status = GetGlobalEnvironmentVariable (L"BootNext", NULL, &BootNextSize, (VOID**)&BootNext);\r
0e29bf5c 515 if (!EFI_ERROR(Status)) {\r
516 ASSERT(BootNextSize == sizeof(UINT16));\r
517\r
518 // Generate the requested Boot Entry variable name\r
519 UnicodeSPrint (BootVariableName, 9 * sizeof(CHAR16), L"Boot%04X", *BootNext);\r
520\r
521 // Set BootCurrent variable\r
522 gRT->SetVariable (L"BootCurrent", &gEfiGlobalVariableGuid,\r
523 EFI_VARIABLE_BOOTSERVICE_ACCESS | EFI_VARIABLE_RUNTIME_ACCESS,\r
524 BootNextSize, BootNext);\r
525\r
526 FreePool (BootNext);\r
527\r
528 // Start the requested Boot Entry\r
529 Status = BdsStartBootOption (BootVariableName);\r
530 if (Status != EFI_NOT_FOUND) {\r
531 // BootNext has not been succeeded launched\r
532 if (EFI_ERROR(Status)) {\r
533 Print(L"Fail to start BootNext.\n");\r
534 }\r
535\r
536 // Delete the BootNext environment variable\r
537 gRT->SetVariable (L"BootNext", &gEfiGlobalVariableGuid,\r
538 EFI_VARIABLE_NON_VOLATILE | EFI_VARIABLE_BOOTSERVICE_ACCESS | EFI_VARIABLE_RUNTIME_ACCESS,\r
539 0, NULL);\r
705b0c03 540 }\r
541\r
0e29bf5c 542 // Clear BootCurrent variable\r
543 gRT->SetVariable (L"BootCurrent", &gEfiGlobalVariableGuid,\r
544 EFI_VARIABLE_BOOTSERVICE_ACCESS | EFI_VARIABLE_RUNTIME_ACCESS,\r
705b0c03 545 0, NULL);\r
546 }\r
547\r
548 // If Boot Order does not exist then create a default entry\r
549 DefineDefaultBootEntries ();\r
550\r
551 // Now we need to setup the EFI System Table with information about the console devices.\r
552 InitializeConsole ();\r
553\r
53cba4fe 554 //\r
555 // Update the CRC32 in the EFI System Table header\r
556 //\r
557 gST->Hdr.CRC32 = 0;\r
558 Status = gBS->CalculateCrc32 ((VOID*)gST, gST->Hdr.HeaderSize, &gST->Hdr.CRC32);\r
559 ASSERT_EFI_ERROR (Status);\r
560\r
705b0c03 561 // Timer before initiating the default boot selection\r
562 StartDefaultBootOnTimeout ();\r
563\r
564 // Start the Boot Menu\r
565 Status = BootMenuMain ();\r
566 ASSERT_EFI_ERROR (Status);\r
567\r
568}\r
569\r
570EFI_BDS_ARCH_PROTOCOL gBdsProtocol = {\r
571 BdsEntry,\r
572};\r
573\r
574EFI_STATUS\r
575EFIAPI\r
576BdsInitialize (\r
577 IN EFI_HANDLE ImageHandle,\r
578 IN EFI_SYSTEM_TABLE *SystemTable\r
579 )\r
580{\r
581 EFI_STATUS Status;\r
582\r
705b0c03 583 Status = gBS->InstallMultipleProtocolInterfaces (\r
584 &ImageHandle,\r
585 &gEfiBdsArchProtocolGuid, &gBdsProtocol,\r
586 NULL\r
587 );\r
588 ASSERT_EFI_ERROR (Status);\r
589\r
590 return Status;\r
591}\r