]> git.proxmox.com Git - mirror_edk2.git/blame - ArmPlatformPkg/Bds/Bds.c
ArmPkg/BdsLib: Fixed memory leak
[mirror_edk2.git] / ArmPlatformPkg / Bds / Bds.c
CommitLineData
705b0c03 1/** @file\r
2*\r
3* Copyright (c) 2011, ARM Limited. All rights reserved.\r
4* \r
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
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
22#define EFI_SET_TIMER_TO_SECOND 10000000\r
23\r
24EFI_HANDLE mImageHandle;\r
25\r
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
42 Status = GetEnvironmentVariable (ConsoleVarName, NULL, NULL, (VOID**)&DevicePathInstances);\r
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
127 \r
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
158 }\r
159 FreePool (Buffer);\r
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
483bc3d3 184 Status = GetConsoleDevicePathFromVariable (L"ConErr", (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
199 return EFI_SUCCESS;\r
200}\r
201\r
202EFI_STATUS\r
203DefineDefaultBootEntries (\r
204 VOID\r
205 )\r
206{\r
207 BDS_LOAD_OPTION* BdsLoadOption;\r
208 UINTN Size;\r
209 EFI_STATUS Status;\r
210 EFI_DEVICE_PATH_FROM_TEXT_PROTOCOL* EfiDevicePathFromTextProtocol;\r
211 EFI_DEVICE_PATH* BootDevicePath;\r
212 ARM_BDS_LOADER_ARGUMENTS* BootArguments;\r
213 ARM_BDS_LOADER_TYPE BootType;\r
214 EFI_DEVICE_PATH* InitrdPath;\r
215 UINTN CmdLineSize;\r
216 UINTN InitrdSize;\r
217\r
218 //\r
219 // If Boot Order does not exist then create a default entry\r
220 //\r
221 Size = 0;\r
222 Status = gRT->GetVariable (L"BootOrder", &gEfiGlobalVariableGuid, NULL, &Size, NULL);\r
223 if (Status == EFI_NOT_FOUND) {\r
224 if ((PcdGetPtr(PcdDefaultBootDevicePath) == NULL) || (StrLen ((CHAR16*)PcdGetPtr(PcdDefaultBootDevicePath)) == 0)) {\r
225 return EFI_UNSUPPORTED;\r
226 }\r
227\r
228 Status = gBS->LocateProtocol (&gEfiDevicePathFromTextProtocolGuid, NULL, (VOID **)&EfiDevicePathFromTextProtocol);\r
229 if (EFI_ERROR(Status)) {\r
230 // You must provide an implementation of DevicePathFromTextProtocol in your firmware (eg: DevicePathDxe)\r
231 DEBUG((EFI_D_ERROR,"Error: Bds requires DevicePathFromTextProtocol\n"));\r
232 return Status;\r
233 }\r
234 BootDevicePath = EfiDevicePathFromTextProtocol->ConvertTextToDevicePath ((CHAR16*)PcdGetPtr(PcdDefaultBootDevicePath));\r
235\r
236 DEBUG_CODE_BEGIN();\r
237 // We convert back to the text representation of the device Path to see if the initial text is correct\r
238 EFI_DEVICE_PATH_TO_TEXT_PROTOCOL* DevicePathToTextProtocol;\r
239 CHAR16* DevicePathTxt;\r
240\r
241 Status = gBS->LocateProtocol(&gEfiDevicePathToTextProtocolGuid, NULL, (VOID **)&DevicePathToTextProtocol);\r
242 ASSERT_EFI_ERROR(Status);\r
243 DevicePathTxt = DevicePathToTextProtocol->ConvertDevicePathToText (BootDevicePath, TRUE, TRUE);\r
244\r
245 ASSERT (StrCmp ((CHAR16*)PcdGetPtr(PcdDefaultBootDevicePath), DevicePathTxt) == 0);\r
246\r
247 FreePool (DevicePathTxt);\r
248 DEBUG_CODE_END();\r
249\r
250 // Create the entry is the Default values are correct\r
251 if (BootDevicePath != NULL) {\r
252 BootType = (ARM_BDS_LOADER_TYPE)PcdGet32 (PcdDefaultBootType);\r
253\r
254 if ((BootType == BDS_LOADER_KERNEL_LINUX_ATAG) || (BootType == BDS_LOADER_KERNEL_LINUX_FDT)) {\r
255 CmdLineSize = AsciiStrSize ((CHAR8*)PcdGetPtr(PcdDefaultBootArgument));\r
256 InitrdPath = EfiDevicePathFromTextProtocol->ConvertTextToDevicePath ((CHAR16*)PcdGetPtr(PcdDefaultBootInitrdPath));\r
257 InitrdSize = GetDevicePathSize (InitrdPath);\r
258\r
259 BootArguments = (ARM_BDS_LOADER_ARGUMENTS*)AllocatePool (sizeof(ARM_BDS_LOADER_ARGUMENTS) + CmdLineSize + InitrdSize);\r
260 BootArguments->LinuxArguments.CmdLineSize = CmdLineSize;\r
261 BootArguments->LinuxArguments.InitrdSize = InitrdSize;\r
262\r
263 CopyMem ((VOID*)(BootArguments + 1), (CHAR8*)PcdGetPtr(PcdDefaultBootArgument), CmdLineSize);\r
264 CopyMem ((VOID*)((UINTN)(BootArguments + 1) + CmdLineSize), InitrdPath, InitrdSize);\r
265 } else {\r
266 BootArguments = NULL;\r
267 }\r
268\r
269 BootOptionCreate (LOAD_OPTION_ACTIVE | LOAD_OPTION_CATEGORY_BOOT,\r
270 (CHAR16*)PcdGetPtr(PcdDefaultBootDescription),\r
271 BootDevicePath,\r
272 BootType,\r
273 BootArguments,\r
274 &BdsLoadOption\r
275 );\r
276 FreePool (BdsLoadOption);\r
277 } else {\r
278 Status = EFI_UNSUPPORTED;\r
279 }\r
280 }\r
281\r
282 return EFI_SUCCESS;\r
283}\r
284\r
285EFI_STATUS\r
286StartDefaultBootOnTimeout (\r
287 VOID\r
288 )\r
289{\r
290 UINTN Size;\r
291 UINT16 Timeout;\r
292 UINT16 *TimeoutPtr;\r
293 EFI_EVENT WaitList[2];\r
294 UINTN WaitIndex;\r
295 UINT16 *BootOrder;\r
296 UINTN BootOrderSize;\r
297 UINTN Index;\r
298 CHAR16 BootVariableName[9];\r
299 EFI_STATUS Status;\r
300 EFI_INPUT_KEY Key;\r
301\r
302 Size = sizeof(UINT16);\r
303 Timeout = (UINT16)PcdGet16 (PcdPlatformBootTimeOut);\r
304 TimeoutPtr = &Timeout;\r
305 GetEnvironmentVariable (L"Timeout", &Timeout, &Size, (VOID**)&TimeoutPtr);\r
306\r
307 if (Timeout != 0xFFFF) {\r
308 if (Timeout > 0) {\r
309 // Create the waiting events (keystroke and 1sec timer)\r
310 gBS->CreateEvent (EVT_TIMER, 0, NULL, NULL, &WaitList[0]);\r
311 gBS->SetTimer (WaitList[0], TimerPeriodic, EFI_SET_TIMER_TO_SECOND);\r
312 WaitList[1] = gST->ConIn->WaitForKey;\r
313\r
314 // Start the timer\r
315 WaitIndex = 0;\r
316 Print(L"The default boot selection will start in ");\r
317 while ((Timeout > 0) && (WaitIndex == 0)) {\r
318 Print(L"%3d seconds",Timeout);\r
319 gBS->WaitForEvent (2, WaitList, &WaitIndex);\r
320 if (WaitIndex == 0) {\r
321 Print(L"\b\b\b\b\b\b\b\b\b\b\b");\r
322 Timeout--;\r
323 }\r
324 }\r
325 // Discard key in the buffer\r
326 do {\r
327 Status = gST->ConIn->ReadKeyStroke (gST->ConIn, &Key);\r
328 } while(!EFI_ERROR(Status));\r
329 gBS->CloseEvent (WaitList[0]);\r
330 Print(L"\n\r");\r
331 }\r
332\r
333 // In case of Timeout we start the default boot selection\r
334 if (Timeout == 0) {\r
335 // Get the Boot Option Order from the environment variable (a default value should have been created)\r
336 GetEnvironmentVariable (L"BootOrder", NULL, &BootOrderSize, (VOID**)&BootOrder);\r
337\r
338 for (Index = 0; Index < BootOrderSize / sizeof (UINT16); Index++) {\r
339 UnicodeSPrint (BootVariableName, 9 * sizeof(CHAR16), L"Boot%04X", BootOrder[Index]);\r
340 Status = BdsStartBootOption (BootVariableName);\r
341 if(!EFI_ERROR(Status)){\r
342 // Boot option returned successfully, hence don't need to start next boot option\r
343 break;\r
344 }\r
345 // In case of success, we should not return from this call.\r
346 }\r
347 }\r
348 }\r
349 return EFI_SUCCESS;\r
350}\r
351\r
352/**\r
353 This function uses policy data from the platform to determine what operating \r
354 system or system utility should be loaded and invoked. This function call \r
355 also optionally make the use of user input to determine the operating system \r
356 or system utility to be loaded and invoked. When the DXE Core has dispatched \r
357 all the drivers on the dispatch queue, this function is called. This \r
358 function will attempt to connect the boot devices required to load and invoke \r
359 the selected operating system or system utility. During this process, \r
360 additional firmware volumes may be discovered that may contain addition DXE \r
361 drivers that can be dispatched by the DXE Core. If a boot device cannot be \r
362 fully connected, this function calls the DXE Service Dispatch() to allow the \r
363 DXE drivers from any newly discovered firmware volumes to be dispatched. \r
364 Then the boot device connection can be attempted again. If the same boot \r
365 device connection operation fails twice in a row, then that boot device has \r
366 failed, and should be skipped. This function should never return.\r
367\r
368 @param This The EFI_BDS_ARCH_PROTOCOL instance.\r
369\r
370 @return None.\r
371\r
372**/\r
373VOID\r
374EFIAPI\r
375BdsEntry (\r
376 IN EFI_BDS_ARCH_PROTOCOL *This\r
377 )\r
378{\r
379 UINTN Size;\r
380 EFI_STATUS Status;\r
381\r
382 PERF_END (NULL, "DXE", NULL, 0);\r
383\r
384 //\r
385 // Declare the Firmware Vendor\r
386 //\r
387 if (FixedPcdGetPtr(PcdFirmwareVendor) != NULL) {\r
388 Size = 0x100;\r
389 gST->FirmwareVendor = AllocateRuntimePool (Size);\r
390 ASSERT (gST->FirmwareVendor != NULL);\r
391 UnicodeSPrint (gST->FirmwareVendor, Size, L"%a EFI %a %a", PcdGetPtr(PcdFirmwareVendor), __DATE__, __TIME__);\r
392 }\r
393\r
394 // If BootNext environment variable is defined then we just load it !\r
395 Status = BdsStartBootOption (L"BootNext");\r
396 if (Status != EFI_NOT_FOUND) {\r
397 // BootNext has not been succeeded launched\r
398 if (EFI_ERROR(Status)) {\r
399 Print(L"Fail to start BootNext.\n");\r
400 }\r
401\r
402 // Delete the BootNext environment variable\r
403 gRT->SetVariable (L"BootNext", &gEfiGlobalVariableGuid,\r
404 EFI_VARIABLE_NON_VOLATILE | EFI_VARIABLE_BOOTSERVICE_ACCESS | EFI_VARIABLE_RUNTIME_ACCESS,\r
405 0, NULL);\r
406 }\r
407\r
408 // If Boot Order does not exist then create a default entry\r
409 DefineDefaultBootEntries ();\r
410\r
411 // Now we need to setup the EFI System Table with information about the console devices.\r
412 InitializeConsole ();\r
413\r
414 // Timer before initiating the default boot selection\r
415 StartDefaultBootOnTimeout ();\r
416\r
417 // Start the Boot Menu\r
418 Status = BootMenuMain ();\r
419 ASSERT_EFI_ERROR (Status);\r
420\r
421}\r
422\r
423EFI_BDS_ARCH_PROTOCOL gBdsProtocol = {\r
424 BdsEntry,\r
425};\r
426\r
427EFI_STATUS\r
428EFIAPI\r
429BdsInitialize (\r
430 IN EFI_HANDLE ImageHandle,\r
431 IN EFI_SYSTEM_TABLE *SystemTable\r
432 )\r
433{\r
434 EFI_STATUS Status;\r
435\r
436 mImageHandle = ImageHandle;\r
437\r
438 Status = gBS->InstallMultipleProtocolInterfaces (\r
439 &ImageHandle,\r
440 &gEfiBdsArchProtocolGuid, &gBdsProtocol,\r
441 NULL\r
442 );\r
443 ASSERT_EFI_ERROR (Status);\r
444\r
445 return Status;\r
446}\r