]> git.proxmox.com Git - mirror_edk2.git/blob - IntelFrameworkModulePkg/Universal/BdsDxe/BootMaint/BBSsupport.c
Report the setting variable failure to platform through the status code when core...
[mirror_edk2.git] / IntelFrameworkModulePkg / Universal / BdsDxe / BootMaint / BBSsupport.c
1 /** @file
2 This function deal with the legacy boot option, it create, delete
3 and manage the legacy boot option, all legacy boot option is getting from
4 the legacy BBS table.
5
6 Copyright (c) 2004 - 2014, Intel Corporation. All rights reserved.<BR>
7 This program and the accompanying materials
8 are licensed and made available under the terms and conditions of the BSD License
9 which accompanies this distribution. The full text of the license may be found at
10 http://opensource.org/licenses/bsd-license.php
11
12 THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
13 WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
14
15 **/
16
17 #include "BBSsupport.h"
18
19 #pragma pack(1)
20 typedef struct {
21 BBS_TABLE BbsEntry;
22 UINT16 BbsIndex;
23 } LEGACY_BOOT_OPTION_BBS_DATA;
24 #pragma pack()
25
26 /**
27 Re-order the Boot Option according to the DevOrder.
28
29 The routine re-orders the Boot Option in BootOption array according to
30 the order specified by DevOrder.
31
32 @param DevOrder Pointer to buffer containing the BBS Index,
33 high 8-bit value 0xFF indicating a disabled boot option
34 @param DevOrderCount Count of the BBS Index
35 @param EnBootOption Callee allocated buffer containing the enabled Boot Option Numbers
36 @param EnBootOptionCount Count of the enabled Boot Option Numbers
37 @param DisBootOption Callee allocated buffer containing the disabled Boot Option Numbers
38 @param DisBootOptionCount Count of the disabled Boot Option Numbers
39 **/
40 VOID
41 OrderLegacyBootOption4SameType (
42 UINT16 *DevOrder,
43 UINTN DevOrderCount,
44 UINT16 **EnBootOption,
45 UINTN *EnBootOptionCount,
46 UINT16 **DisBootOption,
47 UINTN *DisBootOptionCount
48 )
49 {
50 EFI_STATUS Status;
51 UINT16 *NewBootOption;
52 UINT16 *BootOrder;
53 UINTN BootOrderSize;
54 UINTN Index;
55 UINTN StartPosition;
56
57 BDS_COMMON_OPTION *BootOption;
58
59 CHAR16 OptionName[sizeof ("Boot####")];
60 UINT16 *BbsIndexArray;
61 UINT16 *DeviceTypeArray;
62 LIST_ENTRY List;
63
64 BootOrder = BdsLibGetVariableAndSize (
65 L"BootOrder",
66 &gEfiGlobalVariableGuid,
67 &BootOrderSize
68 );
69 ASSERT (BootOrder != NULL);
70
71 BbsIndexArray = AllocatePool (BootOrderSize);
72 DeviceTypeArray = AllocatePool (BootOrderSize);
73 *EnBootOption = AllocatePool (BootOrderSize);
74 *DisBootOption = AllocatePool (BootOrderSize);
75 *DisBootOptionCount = 0;
76 *EnBootOptionCount = 0;
77 Index = 0;
78
79 ASSERT (BbsIndexArray != NULL);
80 ASSERT (DeviceTypeArray != NULL);
81 ASSERT (*EnBootOption != NULL);
82 ASSERT (*DisBootOption != NULL);
83
84 for (Index = 0; Index < BootOrderSize / sizeof (UINT16); Index++) {
85
86 UnicodeSPrint (OptionName, sizeof (OptionName), L"Boot%04x", BootOrder[Index]);
87 InitializeListHead (&List);
88 BootOption = BdsLibVariableToOption (&List, OptionName);
89 ASSERT (BootOption != NULL);
90
91 if ((DevicePathType (BootOption->DevicePath) == BBS_DEVICE_PATH) &&
92 (DevicePathSubType (BootOption->DevicePath) == BBS_BBS_DP)) {
93 //
94 // Legacy Boot Option
95 //
96 ASSERT (BootOption->LoadOptionsSize == sizeof (LEGACY_BOOT_OPTION_BBS_DATA));
97
98 DeviceTypeArray[Index] = ((BBS_BBS_DEVICE_PATH *) BootOption->DevicePath)->DeviceType;
99 BbsIndexArray [Index] = ((LEGACY_BOOT_OPTION_BBS_DATA *) BootOption->LoadOptions)->BbsIndex;
100 } else {
101 DeviceTypeArray[Index] = BBS_TYPE_UNKNOWN;
102 BbsIndexArray [Index] = 0xFFFF;
103 }
104 FreePool (BootOption->DevicePath);
105 FreePool (BootOption->Description);
106 FreePool (BootOption->LoadOptions);
107 FreePool (BootOption);
108 }
109
110 //
111 // Record the corresponding Boot Option Numbers according to the DevOrder
112 // Record the EnBootOption and DisBootOption according to the DevOrder
113 //
114 StartPosition = BootOrderSize / sizeof (UINT16);
115 NewBootOption = AllocatePool (DevOrderCount * sizeof (UINT16));
116 ASSERT (NewBootOption != NULL);
117 while (DevOrderCount-- != 0) {
118 for (Index = 0; Index < BootOrderSize / sizeof (UINT16); Index++) {
119 if (BbsIndexArray[Index] == (DevOrder[DevOrderCount] & 0xFF)) {
120 StartPosition = MIN (StartPosition, Index);
121 NewBootOption[DevOrderCount] = BootOrder[Index];
122
123 if ((DevOrder[DevOrderCount] & 0xFF00) == 0xFF00) {
124 (*DisBootOption)[*DisBootOptionCount] = BootOrder[Index];
125 (*DisBootOptionCount)++;
126 } else {
127 (*EnBootOption)[*EnBootOptionCount] = BootOrder[Index];
128 (*EnBootOptionCount)++;
129 }
130 break;
131 }
132 }
133 }
134
135 //
136 // Overwrite the old BootOption
137 //
138 CopyMem (&BootOrder[StartPosition], NewBootOption, (*DisBootOptionCount + *EnBootOptionCount) * sizeof (UINT16));
139 Status = gRT->SetVariable (
140 L"BootOrder",
141 &gEfiGlobalVariableGuid,
142 EFI_VARIABLE_BOOTSERVICE_ACCESS | EFI_VARIABLE_RUNTIME_ACCESS | EFI_VARIABLE_NON_VOLATILE,
143 BootOrderSize,
144 BootOrder
145 );
146 //
147 // Changing content without increasing its size with current variable implementation shouldn't fail.
148 //
149 ASSERT_EFI_ERROR (Status);
150
151 FreePool (NewBootOption);
152 FreePool (DeviceTypeArray);
153 FreePool (BbsIndexArray);
154 }
155
156 /**
157 Group the legacy boot options in the BootOption.
158
159 The routine assumes the boot options in the beginning that covers all the device
160 types are ordered properly and re-position the following boot options just after
161 the corresponding boot options with the same device type.
162 For example:
163 1. Input = [Harddisk1 CdRom2 Efi1 Harddisk0 CdRom0 CdRom1 Harddisk2 Efi0]
164 Assuming [Harddisk1 CdRom2 Efi1] is ordered properly
165 Output = [Harddisk1 Harddisk0 Harddisk2 CdRom2 CdRom0 CdRom1 Efi1 Efi0]
166
167 2. Input = [Efi1 Efi0 CdRom1 Harddisk0 Harddisk1 Harddisk2 CdRom0 CdRom2]
168 Assuming [Efi1 Efi0 CdRom1 Harddisk0] is ordered properly
169 Output = [Efi1 Efi0 CdRom1 CdRom0 CdRom2 Harddisk0 Harddisk1 Harddisk2]
170
171 **/
172 VOID
173 GroupMultipleLegacyBootOption4SameType (
174 VOID
175 )
176 {
177 EFI_STATUS Status;
178 UINTN Index;
179 UINTN DeviceIndex;
180 UINTN DeviceTypeIndex[7];
181 UINTN *NextIndex;
182 UINT16 OptionNumber;
183 UINT16 *BootOrder;
184 UINTN BootOrderSize;
185 CHAR16 OptionName[sizeof ("Boot####")];
186 BDS_COMMON_OPTION *BootOption;
187 LIST_ENTRY List;
188
189 SetMem (DeviceTypeIndex, sizeof (DeviceTypeIndex), 0xff);
190
191 BootOrder = BdsLibGetVariableAndSize (
192 L"BootOrder",
193 &gEfiGlobalVariableGuid,
194 &BootOrderSize
195 );
196
197 for (Index = 0; Index < BootOrderSize / sizeof (UINT16); Index++) {
198 UnicodeSPrint (OptionName, sizeof (OptionName), L"Boot%04x", BootOrder[Index]);
199 InitializeListHead (&List);
200 BootOption = BdsLibVariableToOption (&List, OptionName);
201 ASSERT (BootOption != NULL);
202
203 if ((DevicePathType (BootOption->DevicePath) == BBS_DEVICE_PATH) &&
204 (DevicePathSubType (BootOption->DevicePath) == BBS_BBS_DP)) {
205 //
206 // Legacy Boot Option
207 //
208 ASSERT ((((BBS_BBS_DEVICE_PATH *) BootOption->DevicePath)->DeviceType & 0xF) < sizeof (DeviceTypeIndex) / sizeof (DeviceTypeIndex[0]));
209 NextIndex = &DeviceTypeIndex[((BBS_BBS_DEVICE_PATH *) BootOption->DevicePath)->DeviceType & 0xF];
210
211 if (*NextIndex == (UINTN) -1) {
212 //
213 // *NextIndex is the Index in BootOrder to put the next Option Number for the same type
214 //
215 *NextIndex = Index + 1;
216 } else {
217 //
218 // insert the current boot option before *NextIndex, causing [*Next .. Index] shift right one position
219 //
220 OptionNumber = BootOrder[Index];
221 CopyMem (&BootOrder[*NextIndex + 1], &BootOrder[*NextIndex], (Index - *NextIndex) * sizeof (UINT16));
222 BootOrder[*NextIndex] = OptionNumber;
223
224 //
225 // Update the DeviceTypeIndex array to reflect the right shift operation
226 //
227 for (DeviceIndex = 0; DeviceIndex < sizeof (DeviceTypeIndex) / sizeof (DeviceTypeIndex[0]); DeviceIndex++) {
228 if (DeviceTypeIndex[DeviceIndex] != (UINTN) -1 && DeviceTypeIndex[DeviceIndex] >= *NextIndex) {
229 DeviceTypeIndex[DeviceIndex]++;
230 }
231 }
232 }
233 }
234 FreePool (BootOption->DevicePath);
235 FreePool (BootOption->Description);
236 FreePool (BootOption->LoadOptions);
237 FreePool (BootOption);
238 }
239
240 Status = gRT->SetVariable (
241 L"BootOrder",
242 &gEfiGlobalVariableGuid,
243 EFI_VARIABLE_BOOTSERVICE_ACCESS | EFI_VARIABLE_RUNTIME_ACCESS | EFI_VARIABLE_NON_VOLATILE,
244 BootOrderSize,
245 BootOrder
246 );
247 //
248 // Changing content without increasing its size with current variable implementation shouldn't fail.
249 //
250 ASSERT_EFI_ERROR (Status);
251 FreePool (BootOrder);
252 }
253