]> git.proxmox.com Git - mirror_edk2.git/blame - MdeModulePkg/Universal/DriverSampleDxe/DriverSample.c
1. Update the implementation of HII ConfigRouting Protocol in HiiDataBase module...
[mirror_edk2.git] / MdeModulePkg / Universal / DriverSampleDxe / DriverSample.c
CommitLineData
93e3992d 1/** @file\r
53cf48e0 2This is an example of how a driver might export data to the HII protocol to be\r
3later utilized by the Setup Protocol\r
4\r
8d00a0f1 5Copyright (c) 2004 - 2008, Intel Corporation\r
93e3992d 6All rights reserved. This program and the accompanying materials\r
7are licensed and made available under the terms and conditions of the BSD License\r
8which accompanies this distribution. The full text of the license may be found at\r
9http://opensource.org/licenses/bsd-license.php\r
10\r
11THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,\r
12WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.\r
13\r
93e3992d 14**/\r
15\r
16\r
17#include "DriverSample.h"\r
18\r
7e3bcccb 19#define DISPLAY_ONLY_MY_ITEM 0x0002\r
93e3992d 20\r
21EFI_GUID mFormSetGuid = FORMSET_GUID;\r
22EFI_GUID mInventoryGuid = INVENTORY_GUID;\r
23\r
24CHAR16 VariableName[] = L"MyIfrNVData";\r
25\r
2f3065c0
LG
26EFI_HANDLE DriverHandle[2] = {NULL, NULL};\r
27DRIVER_SAMPLE_PRIVATE_DATA *PrivateData = NULL;\r
28\r
29HII_VENDOR_DEVICE_PATH mHiiVendorDevicePath0 = {\r
f6f910dd 30 {\r
31 {\r
32 HARDWARE_DEVICE_PATH,\r
33 HW_VENDOR_DP,\r
34 {\r
35 (UINT8) (sizeof (VENDOR_DEVICE_PATH)),\r
36 (UINT8) ((sizeof (VENDOR_DEVICE_PATH)) >> 8)\r
37 }\r
38 },\r
39 //\r
40 // {C153B68D-EBFC-488e-B110-662867745B87}\r
41 //\r
42 { 0xc153b68d, 0xebfc, 0x488e, { 0xb1, 0x10, 0x66, 0x28, 0x67, 0x74, 0x5b, 0x87 } }\r
43 },\r
44 {\r
45 END_DEVICE_PATH_TYPE,\r
46 END_ENTIRE_DEVICE_PATH_SUBTYPE,\r
47 { \r
48 (UINT8) (END_DEVICE_PATH_LENGTH),\r
49 (UINT8) ((END_DEVICE_PATH_LENGTH) >> 8)\r
50 }\r
51 }\r
52};\r
53\r
2f3065c0 54HII_VENDOR_DEVICE_PATH mHiiVendorDevicePath1 = {\r
f6f910dd 55 {\r
56 {\r
57 HARDWARE_DEVICE_PATH,\r
58 HW_VENDOR_DP,\r
59 {\r
60 (UINT8) (sizeof (VENDOR_DEVICE_PATH)),\r
61 (UINT8) ((sizeof (VENDOR_DEVICE_PATH)) >> 8)\r
62 }\r
63 },\r
64 //\r
65 // {06F37F07-0C48-40e9-8436-0A08A0BB76B0}\r
66 //\r
67 { 0x6f37f07, 0xc48, 0x40e9, { 0x84, 0x36, 0xa, 0x8, 0xa0, 0xbb, 0x76, 0xb0 } }\r
68 },\r
69 {\r
70 END_DEVICE_PATH_TYPE,\r
71 END_ENTIRE_DEVICE_PATH_SUBTYPE,\r
72 { \r
73 (UINT8) (END_DEVICE_PATH_LENGTH),\r
74 (UINT8) ((END_DEVICE_PATH_LENGTH) >> 8)\r
75 }\r
76 }\r
77};\r
78\r
7064c0a5 79/**\r
80 Encode the password using a simple algorithm.\r
81 \r
82 @param Password The string to be encoded.\r
83 @param MaxSize The size of the string.\r
84 \r
85**/\r
93e3992d 86VOID\r
87EncodePassword (\r
88 IN CHAR16 *Password,\r
5a829f56 89 IN UINTN MaxSize\r
93e3992d 90 )\r
91{\r
92 UINTN Index;\r
93 UINTN Loop;\r
94 CHAR16 *Buffer;\r
95 CHAR16 *Key;\r
96\r
97 Key = L"MAR10648567";\r
98 Buffer = AllocateZeroPool (MaxSize);\r
99 ASSERT (Buffer != NULL);\r
100\r
101 for (Index = 0; Key[Index] != 0; Index++) {\r
102 for (Loop = 0; Loop < (UINT8) (MaxSize / 2); Loop++) {\r
103 Buffer[Loop] = (CHAR16) (Password[Loop] ^ Key[Index]);\r
104 }\r
105 }\r
106\r
107 CopyMem (Password, Buffer, MaxSize);\r
108\r
53cf48e0 109 FreePool (Buffer);\r
93e3992d 110 return ;\r
111}\r
112\r
7064c0a5 113/**\r
114 Validate the user's password.\r
115 \r
116 @param PrivateData This driver's private context data.\r
117 @param StringId The user's input.\r
118 \r
119 @retval EFI_SUCCESS The user's input matches the password.\r
120 @retval EFI_NOT_READY The user's input does not match the password.\r
121**/\r
93e3992d 122EFI_STATUS\r
123ValidatePassword (\r
7064c0a5 124 IN DRIVER_SAMPLE_PRIVATE_DATA *PrivateData,\r
125 IN EFI_STRING_ID StringId\r
93e3992d 126 )\r
127{\r
128 EFI_STATUS Status;\r
129 UINTN Index;\r
130 UINTN BufferSize;\r
84f9a9ec 131 UINTN PasswordMaxSize;\r
93e3992d 132 CHAR16 *Password;\r
133 CHAR16 *EncodedPassword;\r
134 BOOLEAN OldPassword;\r
135\r
136 //\r
137 // Get encoded password first\r
138 //\r
139 BufferSize = sizeof (DRIVER_SAMPLE_CONFIGURATION);\r
140 Status = gRT->GetVariable (\r
141 VariableName,\r
142 &mFormSetGuid,\r
143 NULL,\r
144 &BufferSize,\r
145 &PrivateData->Configuration\r
146 );\r
147 if (EFI_ERROR (Status)) {\r
148 //\r
149 // Old password not exist, prompt for new password\r
150 //\r
151 return EFI_SUCCESS;\r
152 }\r
153\r
154 OldPassword = FALSE;\r
84f9a9ec 155 PasswordMaxSize = sizeof (PrivateData->Configuration.WhatIsThePassword2);\r
93e3992d 156 //\r
157 // Check whether we have any old password set\r
158 //\r
84f9a9ec 159 for (Index = 0; Index < PasswordMaxSize / sizeof (UINT16); Index++) {\r
93e3992d 160 if (PrivateData->Configuration.WhatIsThePassword2[Index] != 0) {\r
161 OldPassword = TRUE;\r
162 break;\r
163 }\r
164 }\r
165 if (!OldPassword) {\r
166 //\r
167 // Old password not exist, return EFI_SUCCESS to prompt for new password\r
168 //\r
169 return EFI_SUCCESS;\r
170 }\r
171\r
172 //\r
173 // Get user input password\r
174 //\r
cb7d01c0 175 Password = HiiGetString (PrivateData->HiiHandle[0], StringId, NULL);\r
176 if (Password == NULL) {\r
177 return EFI_NOT_READY;\r
178 }\r
84f9a9ec 179 if (StrSize (Password) > PasswordMaxSize) {\r
53cf48e0 180 FreePool (Password);\r
cb7d01c0 181 return EFI_NOT_READY;\r
93e3992d 182 }\r
183\r
184 //\r
185 // Validate old password\r
186 //\r
84f9a9ec 187 EncodedPassword = AllocateZeroPool (PasswordMaxSize);\r
93e3992d 188 ASSERT (EncodedPassword != NULL);\r
84f9a9ec
LG
189 StrnCpy (EncodedPassword, Password, StrLen (Password));\r
190 EncodePassword (EncodedPassword, StrLen (EncodedPassword) * sizeof (CHAR16));\r
191 if (CompareMem (EncodedPassword, PrivateData->Configuration.WhatIsThePassword2, StrLen (EncodedPassword) * sizeof (CHAR16)) != 0) {\r
93e3992d 192 //\r
193 // Old password mismatch, return EFI_NOT_READY to prompt for error message\r
194 //\r
195 Status = EFI_NOT_READY;\r
196 } else {\r
197 Status = EFI_SUCCESS;\r
198 }\r
199\r
53cf48e0 200 FreePool (Password);\r
201 FreePool (EncodedPassword);\r
93e3992d 202\r
203 return Status;\r
204}\r
205\r
7064c0a5 206/**\r
207 Encode the password using a simple algorithm.\r
208 \r
209 @param PrivateData This driver's private context data.\r
210 @param StringId The password from User.\r
211 \r
212 @retval EFI_SUCESS The operation is successful.\r
213 @return Other value if gRT->SetVariable () fails.\r
214 \r
215**/\r
93e3992d 216EFI_STATUS\r
217SetPassword (\r
7064c0a5 218 IN DRIVER_SAMPLE_PRIVATE_DATA *PrivateData,\r
219 IN EFI_STRING_ID StringId\r
93e3992d 220 )\r
221{\r
222 EFI_STATUS Status;\r
93e3992d 223 CHAR16 *Password;\r
cb7d01c0 224 CHAR16 *TempPassword;\r
5a829f56 225 UINTN PasswordSize;\r
93e3992d 226 DRIVER_SAMPLE_CONFIGURATION *Configuration;\r
7e3bcccb 227 UINTN BufferSize;\r
93e3992d 228\r
229 //\r
230 // Get Buffer Storage data from EFI variable\r
231 //\r
232 BufferSize = sizeof (DRIVER_SAMPLE_CONFIGURATION);\r
233 Status = gRT->GetVariable (\r
234 VariableName,\r
235 &mFormSetGuid,\r
236 NULL,\r
237 &BufferSize,\r
238 &PrivateData->Configuration\r
239 );\r
240 if (EFI_ERROR (Status)) {\r
241 return Status;\r
242 }\r
243\r
244 //\r
245 // Get user input password\r
246 //\r
247 Password = &PrivateData->Configuration.WhatIsThePassword2[0];\r
cb7d01c0 248 PasswordSize = sizeof (PrivateData->Configuration.WhatIsThePassword2); \r
5a829f56 249 ZeroMem (Password, PasswordSize);\r
cb7d01c0 250 \r
251 TempPassword = HiiGetString (PrivateData->HiiHandle[0], StringId, NULL);\r
252 if (TempPassword == NULL) {\r
253 return EFI_NOT_READY;\r
254 }\r
84f9a9ec 255 if (StrSize (TempPassword) > PasswordSize) {\r
cb7d01c0 256 FreePool (TempPassword);\r
257 return EFI_NOT_READY;\r
93e3992d 258 }\r
84f9a9ec 259 StrnCpy (Password, TempPassword, StrLen (TempPassword));\r
cb7d01c0 260 FreePool (TempPassword);\r
93e3992d 261\r
262 //\r
263 // Retrive uncommitted data from Browser\r
264 //\r
1d451ff9
LG
265 Configuration = AllocateZeroPool (sizeof (DRIVER_SAMPLE_CONFIGURATION));\r
266 ASSERT (Configuration != NULL);\r
267 if (HiiGetBrowserData (&mFormSetGuid, VariableName, sizeof (DRIVER_SAMPLE_CONFIGURATION), (UINT8 *) Configuration)) {\r
93e3992d 268 //\r
269 // Update password's clear text in the screen\r
270 //\r
84f9a9ec 271 CopyMem (Configuration->PasswordClearText, Password, StrSize (Password));\r
93e3992d 272\r
273 //\r
274 // Update uncommitted data of Browser\r
275 //\r
7e3bcccb
LG
276 HiiSetBrowserData (\r
277 &mFormSetGuid,\r
278 VariableName,\r
279 sizeof (DRIVER_SAMPLE_CONFIGURATION),\r
280 (UINT8 *) Configuration,\r
281 NULL\r
282 );\r
93e3992d 283 }\r
7e3bcccb 284\r
1d451ff9
LG
285 //\r
286 // Free Configuration Buffer\r
287 //\r
288 FreePool (Configuration);\r
289\r
93e3992d 290\r
291 //\r
292 // Set password\r
293 //\r
84f9a9ec 294 EncodePassword (Password, StrLen (Password) * 2);\r
93e3992d 295 Status = gRT->SetVariable(\r
296 VariableName,\r
297 &mFormSetGuid,\r
298 EFI_VARIABLE_NON_VOLATILE | EFI_VARIABLE_BOOTSERVICE_ACCESS,\r
299 sizeof (DRIVER_SAMPLE_CONFIGURATION),\r
300 &PrivateData->Configuration\r
301 );\r
302 return Status;\r
303}\r
304\r
305\r
306/**\r
307 This function allows a caller to extract the current configuration for one\r
308 or more named elements from the target driver.\r
309\r
310 @param This Points to the EFI_HII_CONFIG_ACCESS_PROTOCOL.\r
311 @param Request A null-terminated Unicode string in\r
312 <ConfigRequest> format.\r
313 @param Progress On return, points to a character in the Request\r
314 string. Points to the string's null terminator if\r
315 request was successful. Points to the most recent\r
316 '&' before the first failing name/value pair (or\r
317 the beginning of the string if the failure is in\r
318 the first name/value pair) if the request was not\r
319 successful.\r
320 @param Results A null-terminated Unicode string in\r
321 <ConfigAltResp> format which has all values filled\r
322 in for the names in the Request string. String to\r
323 be allocated by the called function.\r
324\r
325 @retval EFI_SUCCESS The Results is filled with the requested values.\r
326 @retval EFI_OUT_OF_RESOURCES Not enough memory to store the results.\r
327 @retval EFI_INVALID_PARAMETER Request is NULL, illegal syntax, or unknown name.\r
328 @retval EFI_NOT_FOUND Routing data doesn't match any storage in this\r
329 driver.\r
330\r
331**/\r
332EFI_STATUS\r
333EFIAPI\r
334ExtractConfig (\r
335 IN CONST EFI_HII_CONFIG_ACCESS_PROTOCOL *This,\r
336 IN CONST EFI_STRING Request,\r
337 OUT EFI_STRING *Progress,\r
338 OUT EFI_STRING *Results\r
339 )\r
340{\r
341 EFI_STATUS Status;\r
342 UINTN BufferSize;\r
343 DRIVER_SAMPLE_PRIVATE_DATA *PrivateData;\r
344 EFI_HII_CONFIG_ROUTING_PROTOCOL *HiiConfigRouting;\r
84f9a9ec
LG
345 EFI_STRING ConfigRequestHdr;\r
346 EFI_STRING ConfigRequest;\r
347 UINTN Size;\r
348\r
349 //\r
350 // Initialize the local variables.\r
351 //\r
352 ConfigRequestHdr = NULL;\r
353 ConfigRequest = NULL;\r
354 Size = 0;\r
355 *Progress = Request;\r
93e3992d 356\r
357 PrivateData = DRIVER_SAMPLE_PRIVATE_FROM_THIS (This);\r
358 HiiConfigRouting = PrivateData->HiiConfigRouting;\r
359\r
8d00a0f1 360 //\r
84f9a9ec
LG
361 // Get Buffer Storage data from EFI variable.\r
362 // Try to get the current setting from variable.\r
93e3992d 363 //\r
364 BufferSize = sizeof (DRIVER_SAMPLE_CONFIGURATION);\r
84f9a9ec
LG
365 gRT->GetVariable (\r
366 VariableName,\r
367 &mFormSetGuid,\r
368 NULL,\r
369 &BufferSize,\r
370 &PrivateData->Configuration\r
371 );\r
372 \r
8d00a0f1 373 if (Request == NULL) {\r
374 //\r
84f9a9ec 375 // Request is set to NULL, construct full request string.\r
7e3bcccb 376 //\r
84f9a9ec 377\r
7e3bcccb 378 //\r
84f9a9ec
LG
379 // First Set ConfigRequestHdr string.\r
380 //\r
381 ConfigRequestHdr = HiiConstructConfigHdr (&mFormSetGuid, VariableName, PrivateData->DriverHandle[0]);\r
382 ASSERT (ConfigRequestHdr != NULL);\r
8d00a0f1 383\r
84f9a9ec
LG
384 //\r
385 // Allocate and fill a buffer large enough to hold the <ConfigHdr> template \r
386 // followed by "&OFFSET=0&WIDTH=WWWWWWWWWWWWWWWW" followed by a Null-terminator\r
387 //\r
388 Size = (StrLen (ConfigRequestHdr) + 32 + 1) * sizeof (CHAR16);\r
389 ConfigRequest = AllocateZeroPool (Size);\r
390 UnicodeSPrint (ConfigRequest, Size, L"%s&OFFSET=0&WIDTH=%016LX", ConfigRequestHdr, (UINT64)BufferSize);\r
391 FreePool (ConfigRequestHdr);\r
392 } else {\r
393 ConfigRequest = Request;\r
8d00a0f1 394 }\r
395\r
396 //\r
397 // Check routing data in <ConfigHdr>.\r
398 // Note: if only one Storage is used, then this checking could be skipped.\r
399 //\r
84f9a9ec
LG
400 if (!HiiIsConfigHdrMatch (ConfigRequest, &mFormSetGuid, VariableName)) {\r
401 if (Request == NULL) {\r
402 FreePool (ConfigRequest);\r
403 }\r
8d00a0f1 404 return EFI_NOT_FOUND;\r
405 }\r
406\r
93e3992d 407 //\r
408 // Convert buffer data to <ConfigResp> by helper function BlockToConfig()\r
409 //\r
410 Status = HiiConfigRouting->BlockToConfig (\r
411 HiiConfigRouting,\r
84f9a9ec 412 ConfigRequest,\r
93e3992d 413 (UINT8 *) &PrivateData->Configuration,\r
414 BufferSize,\r
415 Results,\r
416 Progress\r
417 );\r
84f9a9ec
LG
418 \r
419 if (Request == NULL) {\r
420 FreePool (ConfigRequest);\r
421 }\r
422\r
93e3992d 423 return Status;\r
424}\r
425\r
426\r
427/**\r
428 This function processes the results of changes in configuration.\r
429\r
430 @param This Points to the EFI_HII_CONFIG_ACCESS_PROTOCOL.\r
431 @param Configuration A null-terminated Unicode string in <ConfigResp>\r
432 format.\r
433 @param Progress A pointer to a string filled in with the offset of\r
434 the most recent '&' before the first failing\r
435 name/value pair (or the beginning of the string if\r
436 the failure is in the first name/value pair) or\r
437 the terminating NULL if all was successful.\r
438\r
439 @retval EFI_SUCCESS The Results is processed successfully.\r
440 @retval EFI_INVALID_PARAMETER Configuration is NULL.\r
441 @retval EFI_NOT_FOUND Routing data doesn't match any storage in this\r
442 driver.\r
443\r
444**/\r
445EFI_STATUS\r
446EFIAPI\r
447RouteConfig (\r
448 IN CONST EFI_HII_CONFIG_ACCESS_PROTOCOL *This,\r
449 IN CONST EFI_STRING Configuration,\r
450 OUT EFI_STRING *Progress\r
451 )\r
452{\r
453 EFI_STATUS Status;\r
454 UINTN BufferSize;\r
455 DRIVER_SAMPLE_PRIVATE_DATA *PrivateData;\r
456 EFI_HII_CONFIG_ROUTING_PROTOCOL *HiiConfigRouting;\r
457\r
8d00a0f1 458 if (Configuration == NULL) {\r
459 return EFI_INVALID_PARAMETER;\r
460 }\r
461\r
93e3992d 462 PrivateData = DRIVER_SAMPLE_PRIVATE_FROM_THIS (This);\r
463 HiiConfigRouting = PrivateData->HiiConfigRouting;\r
464\r
84f9a9ec 465 //\r
8d00a0f1 466 // Check routing data in <ConfigHdr>.\r
467 // Note: if only one Storage is used, then this checking could be skipped.\r
468 //\r
7e3bcccb 469 if (!HiiIsConfigHdrMatch (Configuration, &mFormSetGuid, VariableName)) {\r
8d00a0f1 470 *Progress = Configuration;\r
471 return EFI_NOT_FOUND;\r
472 }\r
473\r
93e3992d 474 //\r
475 // Get Buffer Storage data from EFI variable\r
476 //\r
477 BufferSize = sizeof (DRIVER_SAMPLE_CONFIGURATION);\r
84f9a9ec
LG
478 gRT->GetVariable (\r
479 VariableName,\r
480 &mFormSetGuid,\r
481 NULL,\r
482 &BufferSize,\r
483 &PrivateData->Configuration\r
484 );\r
93e3992d 485\r
486 //\r
487 // Convert <ConfigResp> to buffer data by helper function ConfigToBlock()\r
488 //\r
489 BufferSize = sizeof (DRIVER_SAMPLE_CONFIGURATION);\r
490 Status = HiiConfigRouting->ConfigToBlock (\r
491 HiiConfigRouting,\r
492 Configuration,\r
493 (UINT8 *) &PrivateData->Configuration,\r
494 &BufferSize,\r
495 Progress\r
496 );\r
497 if (EFI_ERROR (Status)) {\r
498 return Status;\r
499 }\r
500\r
501 //\r
502 // Store Buffer Storage back to EFI variable\r
503 //\r
504 Status = gRT->SetVariable(\r
505 VariableName,\r
506 &mFormSetGuid,\r
507 EFI_VARIABLE_NON_VOLATILE | EFI_VARIABLE_BOOTSERVICE_ACCESS,\r
508 sizeof (DRIVER_SAMPLE_CONFIGURATION),\r
509 &PrivateData->Configuration\r
510 );\r
511\r
512 return Status;\r
513}\r
514\r
515\r
516/**\r
517 This function processes the results of changes in configuration.\r
518\r
519 @param This Points to the EFI_HII_CONFIG_ACCESS_PROTOCOL.\r
520 @param Action Specifies the type of action taken by the browser.\r
521 @param QuestionId A unique value which is sent to the original\r
522 exporting driver so that it can identify the type\r
523 of data to expect.\r
524 @param Type The type of value for the question.\r
525 @param Value A pointer to the data being sent to the original\r
526 exporting driver.\r
527 @param ActionRequest On return, points to the action requested by the\r
528 callback function.\r
529\r
530 @retval EFI_SUCCESS The callback successfully handled the action.\r
531 @retval EFI_OUT_OF_RESOURCES Not enough storage is available to hold the\r
532 variable and its data.\r
533 @retval EFI_DEVICE_ERROR The variable could not be saved.\r
534 @retval EFI_UNSUPPORTED The specified Action is not supported by the\r
535 callback.\r
536\r
537**/\r
538EFI_STATUS\r
539EFIAPI\r
540DriverCallback (\r
541 IN CONST EFI_HII_CONFIG_ACCESS_PROTOCOL *This,\r
542 IN EFI_BROWSER_ACTION Action,\r
543 IN EFI_QUESTION_ID QuestionId,\r
544 IN UINT8 Type,\r
545 IN EFI_IFR_TYPE_VALUE *Value,\r
546 OUT EFI_BROWSER_ACTION_REQUEST *ActionRequest\r
547 )\r
548{\r
549 DRIVER_SAMPLE_PRIVATE_DATA *PrivateData;\r
550 EFI_STATUS Status;\r
8d00a0f1 551 UINT8 MyVar;\r
7e3bcccb
LG
552 VOID *StartOpCodeHandle;\r
553 VOID *OptionsOpCodeHandle;\r
554 EFI_IFR_GUID_LABEL *StartLabel;\r
555 VOID *EndOpCodeHandle;\r
556 EFI_IFR_GUID_LABEL *EndLabel;\r
93e3992d 557\r
558 if ((Value == NULL) || (ActionRequest == NULL)) {\r
559 return EFI_INVALID_PARAMETER;\r
560 }\r
84f9a9ec
LG
561 \r
562 if ((Type == EFI_IFR_TYPE_STRING) && (Value->string == 0)) {\r
563 return EFI_INVALID_PARAMETER;\r
564 }\r
565 \r
93e3992d 566\r
567 Status = EFI_SUCCESS;\r
568 PrivateData = DRIVER_SAMPLE_PRIVATE_FROM_THIS (This);\r
569\r
570 switch (QuestionId) {\r
571 case 0x1234:\r
572 //\r
8d00a0f1 573 // Initialize the container for dynamic opcodes\r
93e3992d 574 //\r
7e3bcccb
LG
575 StartOpCodeHandle = HiiAllocateOpCodeHandle ();\r
576 ASSERT (StartOpCodeHandle != NULL);\r
577\r
578 EndOpCodeHandle = HiiAllocateOpCodeHandle ();\r
579 ASSERT (EndOpCodeHandle != NULL);\r
580\r
581 //\r
582 // Create Hii Extend Label OpCode as the start opcode\r
583 //\r
584 StartLabel = (EFI_IFR_GUID_LABEL *) HiiCreateGuidOpCode (StartOpCodeHandle, &gEfiIfrTianoGuid, NULL, sizeof (EFI_IFR_GUID_LABEL));\r
585 StartLabel->ExtendOpCode = EFI_IFR_EXTEND_OP_LABEL;\r
586 StartLabel->Number = LABEL_UPDATE1;\r
587\r
588 //\r
589 // Create Hii Extend Label OpCode as the end opcode\r
590 //\r
591 EndLabel = (EFI_IFR_GUID_LABEL *) HiiCreateGuidOpCode (EndOpCodeHandle, &gEfiIfrTianoGuid, NULL, sizeof (EFI_IFR_GUID_LABEL));\r
592 EndLabel->ExtendOpCode = EFI_IFR_EXTEND_OP_LABEL;\r
593 EndLabel->Number = LABEL_END;\r
594\r
595 HiiCreateActionOpCode (\r
596 StartOpCodeHandle, // Container for dynamic created opcodes\r
597 0x1237, // Question ID\r
598 STRING_TOKEN(STR_EXIT_TEXT), // Prompt text\r
599 STRING_TOKEN(STR_EXIT_TEXT), // Help text\r
600 EFI_IFR_FLAG_CALLBACK, // Question flag\r
601 0 // Action String ID\r
602 );\r
603\r
604 //\r
605 // Create Option OpCode\r
606 //\r
607 OptionsOpCodeHandle = HiiAllocateOpCodeHandle ();\r
608 ASSERT (OptionsOpCodeHandle != NULL);\r
609\r
610 HiiCreateOneOfOptionOpCode (\r
611 OptionsOpCodeHandle,\r
612 STRING_TOKEN (STR_BOOT_OPTION1),\r
613 0,\r
614 EFI_IFR_NUMERIC_SIZE_1,\r
615 1\r
616 );\r
617\r
618 HiiCreateOneOfOptionOpCode (\r
619 OptionsOpCodeHandle,\r
620 STRING_TOKEN (STR_BOOT_OPTION2),\r
621 0,\r
622 EFI_IFR_NUMERIC_SIZE_1,\r
623 2\r
624 );\r
625\r
626 //\r
627 // Prepare initial value for the dynamic created oneof Question\r
628 //\r
629 PrivateData->Configuration.DynamicOneof = 2;\r
630 Status = gRT->SetVariable(\r
631 VariableName,\r
632 &mFormSetGuid,\r
633 EFI_VARIABLE_NON_VOLATILE | EFI_VARIABLE_BOOTSERVICE_ACCESS,\r
634 sizeof (DRIVER_SAMPLE_CONFIGURATION),\r
635 &PrivateData->Configuration\r
636 );\r
637\r
638 HiiCreateOneOfOpCode (\r
639 StartOpCodeHandle, // Container for dynamic created opcodes\r
640 0x8001, // Question ID (or call it "key")\r
641 CONFIGURATION_VARSTORE_ID, // VarStore ID\r
642 (UINT16) DYNAMIC_ONE_OF_VAR_OFFSET, // Offset in Buffer Storage\r
643 STRING_TOKEN (STR_ONE_OF_PROMPT), // Question prompt text\r
644 STRING_TOKEN (STR_ONE_OF_HELP), // Question help text\r
645 EFI_IFR_FLAG_CALLBACK, // Question flag\r
646 EFI_IFR_NUMERIC_SIZE_1, // Data type of Question Value\r
647 OptionsOpCodeHandle, // Option Opcode list\r
648 NULL // Default Opcode is NULl\r
649 );\r
650\r
651 HiiCreateOrderedListOpCode (\r
652 StartOpCodeHandle, // Container for dynamic created opcodes\r
653 0x8002, // Question ID\r
654 CONFIGURATION_VARSTORE_ID, // VarStore ID\r
655 (UINT16) DYNAMIC_ORDERED_LIST_VAR_OFFSET, // Offset in Buffer Storage\r
656 STRING_TOKEN (STR_BOOT_OPTIONS), // Question prompt text\r
657 STRING_TOKEN (STR_BOOT_OPTIONS), // Question help text\r
658 EFI_IFR_FLAG_RESET_REQUIRED, // Question flag\r
659 0, // Ordered list flag, e.g. EFI_IFR_UNIQUE_SET\r
660 EFI_IFR_NUMERIC_SIZE_1, // Data type of Question value\r
661 5, // Maximum container\r
662 OptionsOpCodeHandle, // Option Opcode list\r
663 NULL // Default Opcode is NULl\r
664 );\r
665\r
666 HiiCreateGotoOpCode (\r
667 StartOpCodeHandle, // Container for dynamic created opcodes\r
668 1, // Target Form ID\r
669 STRING_TOKEN (STR_GOTO_FORM1), // Prompt text\r
670 STRING_TOKEN (STR_GOTO_HELP), // Help text\r
671 0, // Question flag\r
672 0x8003 // Question ID\r
673 );\r
674\r
675 HiiUpdateForm (\r
676 PrivateData->HiiHandle[0], // HII handle\r
677 &mFormSetGuid, // Formset GUID\r
678 0x1234, // Form ID\r
679 StartOpCodeHandle, // Label for where to insert opcodes\r
680 EndOpCodeHandle // Replace data\r
681 );\r
682\r
683 HiiFreeOpCodeHandle (StartOpCodeHandle);\r
684 HiiFreeOpCodeHandle (OptionsOpCodeHandle);\r
685 break;\r
8d00a0f1 686 \r
7e3bcccb
LG
687 case 0x5678:\r
688 //\r
689 // We will reach here once the Question is refreshed\r
690 //\r
691\r
692 //\r
693 // Initialize the container for dynamic opcodes\r
694 //\r
695 StartOpCodeHandle = HiiAllocateOpCodeHandle ();\r
696 ASSERT (StartOpCodeHandle != NULL);\r
697\r
698 //\r
699 // Create Hii Extend Label OpCode as the start opcode\r
700 //\r
701 StartLabel = (EFI_IFR_GUID_LABEL *) HiiCreateGuidOpCode (StartOpCodeHandle, &gEfiIfrTianoGuid, NULL, sizeof (EFI_IFR_GUID_LABEL));\r
702 StartLabel->ExtendOpCode = EFI_IFR_EXTEND_OP_LABEL;\r
703 StartLabel->Number = LABEL_UPDATE2;\r
704\r
705 HiiCreateActionOpCode (\r
706 StartOpCodeHandle, // Container for dynamic created opcodes\r
707 0x1237, // Question ID\r
708 STRING_TOKEN(STR_EXIT_TEXT), // Prompt text\r
709 STRING_TOKEN(STR_EXIT_TEXT), // Help text\r
710 EFI_IFR_FLAG_CALLBACK, // Question flag\r
711 0 // Action String ID\r
712 );\r
8d00a0f1 713 \r
7e3bcccb
LG
714 HiiUpdateForm (\r
715 PrivateData->HiiHandle[0], // HII handle\r
716 &mFormSetGuid, // Formset GUID\r
717 0x3, // Form ID\r
718 StartOpCodeHandle, // Label for where to insert opcodes\r
719 NULL // Insert data\r
720 );\r
721\r
722 HiiFreeOpCodeHandle (StartOpCodeHandle); \r
723 \r
724 //\r
725 // Refresh the Question value\r
726 //\r
727 PrivateData->Configuration.DynamicRefresh++;\r
728 Status = gRT->SetVariable(\r
729 VariableName,\r
730 &mFormSetGuid,\r
731 EFI_VARIABLE_NON_VOLATILE | EFI_VARIABLE_BOOTSERVICE_ACCESS,\r
732 sizeof (DRIVER_SAMPLE_CONFIGURATION),\r
733 &PrivateData->Configuration\r
734 );\r
735 \r
736 //\r
737 // Change an EFI Variable storage (MyEfiVar) asynchronous, this will cause\r
738 // the first statement in Form 3 be suppressed\r
739 //\r
740 MyVar = 111;\r
741 Status = gRT->SetVariable(\r
742 L"MyVar",\r
743 &mFormSetGuid,\r
744 EFI_VARIABLE_NON_VOLATILE | EFI_VARIABLE_BOOTSERVICE_ACCESS,\r
745 1,\r
746 &MyVar\r
747 );\r
93e3992d 748 break;\r
749\r
750 case 0x1237:\r
751 //\r
752 // User press "Exit now", request Browser to exit\r
753 //\r
754 *ActionRequest = EFI_BROWSER_ACTION_REQUEST_EXIT;\r
755 break;\r
756\r
757 case 0x1238:\r
758 //\r
759 // User press "Save now", request Browser to save the uncommitted data.\r
760 //\r
761 *ActionRequest = EFI_BROWSER_ACTION_REQUEST_SUBMIT;\r
762 break;\r
763\r
764 case 0x2000:\r
765 //\r
766 // When try to set a new password, user will be chanlleged with old password.\r
767 // The Callback is responsible for validating old password input by user,\r
768 // If Callback return EFI_SUCCESS, it indicates validation pass.\r
769 //\r
770 switch (PrivateData->PasswordState) {\r
771 case BROWSER_STATE_VALIDATE_PASSWORD:\r
772 Status = ValidatePassword (PrivateData, Value->string);\r
773 if (Status == EFI_SUCCESS) {\r
774 PrivateData->PasswordState = BROWSER_STATE_SET_PASSWORD;\r
775 }\r
776 break;\r
777\r
778 case BROWSER_STATE_SET_PASSWORD:\r
779 Status = SetPassword (PrivateData, Value->string);\r
780 PrivateData->PasswordState = BROWSER_STATE_VALIDATE_PASSWORD;\r
781 break;\r
782\r
783 default:\r
784 break;\r
785 }\r
786\r
787 break;\r
788\r
789 default:\r
790 break;\r
791 }\r
792\r
793 return Status;\r
794}\r
795\r
7064c0a5 796/**\r
797 Main entry for this driver.\r
798 \r
799 @param ImageHandle Image handle this driver.\r
800 @param SystemTable Pointer to SystemTable.\r
801\r
802 @retval EFI_SUCESS This function always complete successfully.\r
803\r
804**/\r
93e3992d 805EFI_STATUS\r
806EFIAPI\r
807DriverSampleInit (\r
808 IN EFI_HANDLE ImageHandle,\r
809 IN EFI_SYSTEM_TABLE *SystemTable\r
810 )\r
811{\r
812 EFI_STATUS Status;\r
93e3992d 813 EFI_HII_HANDLE HiiHandle[2];\r
93e3992d 814 EFI_SCREEN_DESCRIPTOR Screen;\r
815 EFI_HII_DATABASE_PROTOCOL *HiiDatabase;\r
816 EFI_HII_STRING_PROTOCOL *HiiString;\r
817 EFI_FORM_BROWSER2_PROTOCOL *FormBrowser2;\r
818 EFI_HII_CONFIG_ROUTING_PROTOCOL *HiiConfigRouting;\r
819 CHAR16 *NewString;\r
820 UINTN BufferSize;\r
821 DRIVER_SAMPLE_CONFIGURATION *Configuration;\r
84f9a9ec
LG
822 BOOLEAN ActionFlag;\r
823 EFI_STRING ConfigRequestHdr; \r
93e3992d 824\r
825 //\r
84f9a9ec 826 // Initialize the local variables.\r
93e3992d 827 //\r
84f9a9ec 828 ConfigRequestHdr = NULL;\r
93e3992d 829\r
830 //\r
831 // Initialize screen dimensions for SendForm().\r
832 // Remove 3 characters from top and bottom\r
833 //\r
834 ZeroMem (&Screen, sizeof (EFI_SCREEN_DESCRIPTOR));\r
835 gST->ConOut->QueryMode (gST->ConOut, gST->ConOut->Mode->Mode, &Screen.RightColumn, &Screen.BottomRow);\r
836\r
837 Screen.TopRow = 3;\r
838 Screen.BottomRow = Screen.BottomRow - 3;\r
839\r
840 //\r
841 // Initialize driver private data\r
842 //\r
843 PrivateData = AllocatePool (sizeof (DRIVER_SAMPLE_PRIVATE_DATA));\r
844 if (PrivateData == NULL) {\r
845 return EFI_OUT_OF_RESOURCES;\r
846 }\r
847\r
848 PrivateData->Signature = DRIVER_SAMPLE_PRIVATE_SIGNATURE;\r
849\r
850 PrivateData->ConfigAccess.ExtractConfig = ExtractConfig;\r
851 PrivateData->ConfigAccess.RouteConfig = RouteConfig;\r
852 PrivateData->ConfigAccess.Callback = DriverCallback;\r
853 PrivateData->PasswordState = BROWSER_STATE_VALIDATE_PASSWORD;\r
854\r
855 //\r
856 // Locate Hii Database protocol\r
857 //\r
858 Status = gBS->LocateProtocol (&gEfiHiiDatabaseProtocolGuid, NULL, (VOID **) &HiiDatabase);\r
859 if (EFI_ERROR (Status)) {\r
860 return Status;\r
861 }\r
862 PrivateData->HiiDatabase = HiiDatabase;\r
863\r
864 //\r
865 // Locate HiiString protocol\r
866 //\r
867 Status = gBS->LocateProtocol (&gEfiHiiStringProtocolGuid, NULL, (VOID **) &HiiString);\r
868 if (EFI_ERROR (Status)) {\r
869 return Status;\r
870 }\r
871 PrivateData->HiiString = HiiString;\r
872\r
873 //\r
874 // Locate Formbrowser2 protocol\r
875 //\r
876 Status = gBS->LocateProtocol (&gEfiFormBrowser2ProtocolGuid, NULL, (VOID **) &FormBrowser2);\r
877 if (EFI_ERROR (Status)) {\r
878 return Status;\r
879 }\r
880 PrivateData->FormBrowser2 = FormBrowser2;\r
881\r
882 //\r
883 // Locate ConfigRouting protocol\r
884 //\r
885 Status = gBS->LocateProtocol (&gEfiHiiConfigRoutingProtocolGuid, NULL, (VOID **) &HiiConfigRouting);\r
886 if (EFI_ERROR (Status)) {\r
887 return Status;\r
888 }\r
889 PrivateData->HiiConfigRouting = HiiConfigRouting;\r
890\r
f6f910dd 891 Status = gBS->InstallMultipleProtocolInterfaces (\r
93e3992d 892 &DriverHandle[0],\r
f6f910dd 893 &gEfiDevicePathProtocolGuid,\r
2f3065c0 894 &mHiiVendorDevicePath0,\r
93e3992d 895 &gEfiHiiConfigAccessProtocolGuid,\r
f6f910dd 896 &PrivateData->ConfigAccess,\r
897 NULL\r
93e3992d 898 );\r
899 ASSERT_EFI_ERROR (Status);\r
900\r
f6f910dd 901 PrivateData->DriverHandle[0] = DriverHandle[0];\r
902\r
93e3992d 903 //\r
904 // Publish our HII data\r
905 //\r
cb7d01c0 906 HiiHandle[0] = HiiAddPackages (\r
907 &mFormSetGuid,\r
908 DriverHandle[0],\r
909 DriverSampleStrings,\r
910 VfrBin,\r
911 NULL\r
912 );\r
913 if (HiiHandle[0] == NULL) {\r
93e3992d 914 return EFI_OUT_OF_RESOURCES;\r
915 }\r
916\r
93e3992d 917 PrivateData->HiiHandle[0] = HiiHandle[0];\r
918\r
919 //\r
920 // Publish another Fromset\r
921 //\r
f6f910dd 922 Status = gBS->InstallMultipleProtocolInterfaces (\r
923 &DriverHandle[1],\r
924 &gEfiDevicePathProtocolGuid,\r
2f3065c0 925 &mHiiVendorDevicePath1,\r
f6f910dd 926 NULL\r
927 );\r
928 ASSERT_EFI_ERROR (Status);\r
929\r
93e3992d 930 PrivateData->DriverHandle[1] = DriverHandle[1];\r
931\r
cb7d01c0 932 HiiHandle[1] = HiiAddPackages (\r
933 &mInventoryGuid,\r
934 DriverHandle[1],\r
935 DriverSampleStrings,\r
936 InventoryBin,\r
937 NULL\r
938 );\r
939 if (HiiHandle[1] == NULL) {\r
93e3992d 940 return EFI_OUT_OF_RESOURCES;\r
941 }\r
942\r
93e3992d 943 PrivateData->HiiHandle[1] = HiiHandle[1];\r
944\r
945 //\r
946 // Very simple example of how one would update a string that is already\r
947 // in the HII database\r
948 //\r
949 NewString = L"700 Mhz";\r
950\r
cb7d01c0 951 if (HiiSetString (HiiHandle[0], STRING_TOKEN (STR_CPU_STRING2), NewString, NULL) == 0) {\r
952 return EFI_OUT_OF_RESOURCES;\r
93e3992d 953 }\r
954\r
955 //\r
956 // Initialize configuration data\r
957 //\r
958 Configuration = &PrivateData->Configuration;\r
959 ZeroMem (Configuration, sizeof (DRIVER_SAMPLE_CONFIGURATION));\r
960\r
961 //\r
962 // Try to read NV config EFI variable first\r
963 //\r
84f9a9ec
LG
964 ConfigRequestHdr = HiiConstructConfigHdr (&mFormSetGuid, VariableName, DriverHandle[0]);\r
965 ASSERT (ConfigRequestHdr != NULL);\r
966\r
93e3992d 967 BufferSize = sizeof (DRIVER_SAMPLE_CONFIGURATION);\r
968 Status = gRT->GetVariable (VariableName, &mFormSetGuid, NULL, &BufferSize, Configuration);\r
84f9a9ec 969 if (EFI_ERROR (Status)) {\r
93e3992d 970 //\r
971 // EFI variable for NV config doesn't exit, we should build this variable\r
972 // based on default values stored in IFR\r
973 //\r
84f9a9ec
LG
974 ActionFlag = HiiSetToDefaults (ConfigRequestHdr, EFI_HII_DEFAULT_CLASS_STANDARD);\r
975 ASSERT (ActionFlag);\r
976 } else {\r
977 //\r
978 // EFI variable does exist and Validate Current Setting\r
979 //\r
980 ActionFlag = HiiValidateSettings (ConfigRequestHdr);\r
981 ASSERT (ActionFlag);\r
93e3992d 982 }\r
84f9a9ec
LG
983 \r
984 FreePool (ConfigRequestHdr);\r
985\r
93e3992d 986\r
987 //\r
84f9a9ec 988 // In default, this driver is built into Flash device image, \r
7e3bcccb 989 // the following code doesn't run.\r
93e3992d 990 //\r
7e3bcccb 991\r
2f3065c0 992 //\r
7e3bcccb
LG
993 // Example of how to display only the item we sent to HII\r
994 // When this driver is not built into Flash device image, \r
995 // it need to call SendForm to show front page by itself.\r
2f3065c0 996 //\r
7e3bcccb
LG
997 if (DISPLAY_ONLY_MY_ITEM <= 1) {\r
998 //\r
999 // Have the browser pull out our copy of the data, and only display our data\r
1000 //\r
1001 Status = FormBrowser2->SendForm (\r
1002 FormBrowser2,\r
1003 &(HiiHandle[DISPLAY_ONLY_MY_ITEM]),\r
1004 1,\r
1005 NULL,\r
1006 0,\r
1007 NULL,\r
1008 NULL\r
1009 );\r
7e3bcccb 1010 \r
14d59fa1 1011 HiiRemovePackages (HiiHandle[0]);\r
7e3bcccb 1012 \r
14d59fa1 1013 HiiRemovePackages (HiiHandle[1]);\r
2f3065c0
LG
1014 }\r
1015\r
84f9a9ec 1016 return EFI_SUCCESS;\r
2f3065c0
LG
1017}\r
1018\r
1019/**\r
1020 Unloads the application and its installed protocol.\r
1021\r
1022 @param[in] ImageHandle Handle that identifies the image to be unloaded.\r
1023\r
1024 @retval EFI_SUCCESS The image has been unloaded.\r
1025**/\r
1026EFI_STATUS\r
1027EFIAPI\r
1028DriverSampleUnload (\r
1029 IN EFI_HANDLE ImageHandle\r
1030 )\r
1031{\r
1032 if (DriverHandle[0] != NULL) {\r
1033 gBS->UninstallMultipleProtocolInterfaces (\r
1034 DriverHandle[0],\r
1035 &gEfiDevicePathProtocolGuid,\r
1036 &mHiiVendorDevicePath0,\r
1037 &gEfiHiiConfigAccessProtocolGuid,\r
1038 &PrivateData->ConfigAccess,\r
1039 NULL\r
1040 );\r
1041 }\r
1042\r
1043 if (DriverHandle[1] != NULL) {\r
1044 gBS->UninstallMultipleProtocolInterfaces (\r
1045 DriverHandle[1],\r
1046 &gEfiDevicePathProtocolGuid,\r
1047 &mHiiVendorDevicePath1,\r
1048 NULL\r
1049 );\r
1050 }\r
1051\r
14d59fa1
LG
1052 if (PrivateData->HiiHandle[0] != NULL) {\r
1053 HiiRemovePackages (PrivateData->HiiHandle[0]);\r
1054 }\r
1055\r
1056 if (PrivateData->HiiHandle[1] != NULL) {\r
1057 HiiRemovePackages (PrivateData->HiiHandle[1]);\r
1058 }\r
1059\r
2f3065c0
LG
1060 if (PrivateData != NULL) {\r
1061 FreePool (PrivateData);\r
93e3992d 1062 }\r
1063\r
1064 return EFI_SUCCESS;\r
1065}\r