/** @file DiskIo driver that lays on every BlockIo protocol in the system. DiskIo converts a block oriented device to a byte oriented device. Disk access may have to handle unaligned request about sector boundaries. There are three cases: UnderRun - The first byte is not on a sector boundary or the read request is less than a sector in length. Aligned - A read of N contiguous sectors. OverRun - The last byte is not on a sector boundary. Copyright (c) 2006 - 2009, Intel Corporation.
All rights reserved. This program and the accompanying materials are licensed and made available under the terms and conditions of the BSD License which accompanies this distribution. The full text of the license may be found at http://opensource.org/licenses/bsd-license.php THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED. **/ #include "DriverHealthDxe.h" #undef STRING_TOKEN #define STRING_TOKEN(x) 0 extern EFI_GUID gEfiCallerIdGuid; CHAR16 VariableName[] = L"Config"; UINTN mNumNotHealthy = 0; UINT8 ControllerIndex = 0; // // Link used to store the controller health status // LIST_ENTRY mControllerList = {NULL, NULL}; // // 0 - Healthy -> {0} // 1 - Health with warning messages -> {1} // 2 - Failed -> {2} // 3 - Failed with error messages -> {3} // 4 - RebootRequired -> {4} // 5 - RebootRequired with messages -> {5} // 6 - ReconnectRequired -> {6} // 7 - ReconnectRequired with messages -> {7} // 100..103 - RepairRequired -> {0..3} // 104..107 - RepairRequired with error messages -> {0..3} // 108..111 - RepairRequired with progress notifications -> {0..3} // 112..115 - RepairRequired with error messages and progress notifications -> {0..3} // 132..163 - RepairRequired -> {300..331} // 164..195 - RepairRequired with error messages -> {300..331} // 196..227 - RepairRequired with progress notifications -> {300..331} // 228..259 - RepairRequired with error messages and progress notifications -> {300..331} // 300..307 - ConfigRequired -> {0..7} // 308..315 - ConfigRequired with error messages -> {0..7} // 316..323 - ConfigRequired with forms -> {0..7} // 324..331 - ConfigRequired with forms and error messages -> {0..7} // 332..347 - ConfigRequired -> {100..115} // 348..363 - ConfigRequired with error messages -> {100..115} // 364..379 - ConfigRequired with forms -> {100.115} // 380..395 - ConfigRequired with forms and error messages -> {100..115} DEVICE_STATE mDeviceState[] = { { TRUE, 308, 000, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 309, 001, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 310, 002, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 311, 003, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 312, 004, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 313, 005, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 314, 006, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 315, 007, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 000, 000, 0, FALSE, EfiDriverHealthStatusHealthy }, { TRUE, 001, 001, STRING_TOKEN (STR_HEALTHY_WARNING), FALSE, EfiDriverHealthStatusHealthy }, { TRUE, 002, 002, 0, FALSE, EfiDriverHealthStatusFailed }, { TRUE, 003, 003, STRING_TOKEN (STR_FAILED_ERROR), FALSE, EfiDriverHealthStatusFailed }, { FALSE, 004, 004, 0, FALSE, EfiDriverHealthStatusRebootRequired }, { FALSE, 005, 005, STRING_TOKEN (STR_REBOOT_REQUIRED), FALSE, EfiDriverHealthStatusRebootRequired }, { FALSE, 006, 006, 0, FALSE, EfiDriverHealthStatusReconnectRequired }, { FALSE, 007, 007, STRING_TOKEN (STR_RECONNECT_REQUIRED), FALSE, EfiDriverHealthStatusReconnectRequired }, { TRUE, 100, 000, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 101, 001, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 102, 002, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 103, 003, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 104, 000, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 105, 001, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 106, 002, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 107, 003, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 108, 000, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 109, 001, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 110, 002, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 111, 003, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 112, 000, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 113, 001, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 114, 002, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 115, 003, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 132, 300, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 133, 301, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 134, 302, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 135, 303, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 136, 304, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 137, 305, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 138, 306, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 139, 307, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 140, 308, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 141, 309, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 142, 310, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 143, 311, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 144, 312, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 145, 313, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 146, 314, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 147, 315, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 148, 316, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 149, 317, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 150, 318, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 151, 319, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 152, 320, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 153, 321, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 154, 322, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 155, 323, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 156, 324, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 157, 325, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 158, 326, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 159, 327, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 160, 328, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 161, 329, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 162, 330, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 163, 331, 0, FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 164, 300, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 165, 301, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 166, 302, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 167, 303, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 168, 304, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 169, 305, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 170, 306, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 171, 307, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 172, 308, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 173, 309, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 174, 310, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 175, 311, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 176, 312, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 177, 313, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 178, 314, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 179, 315, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 180, 316, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 181, 317, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 182, 318, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 183, 319, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 184, 320, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 185, 321, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 186, 322, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 187, 323, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 188, 324, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 189, 325, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 190, 326, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 191, 327, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 192, 328, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 193, 329, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 194, 330, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 195, 331, STRING_TOKEN (STR_REPAIR_REQUIRED), FALSE, EfiDriverHealthStatusRepairRequired }, { TRUE, 196, 300, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 197, 301, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 198, 302, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 199, 303, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 200, 304, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 201, 305, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 202, 306, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 203, 307, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 204, 308, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 205, 309, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 206, 310, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 207, 311, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 208, 312, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 209, 313, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 210, 314, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 211, 315, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 212, 316, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 213, 317, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 214, 318, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 215, 319, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 216, 320, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 217, 321, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 218, 322, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 219, 323, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 220, 324, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 221, 325, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 222, 326, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 223, 327, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 224, 328, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 225, 329, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 226, 330, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 227, 331, 0, TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 228, 300, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 229, 301, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 230, 302, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 231, 303, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 232, 304, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 233, 305, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 234, 306, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 235, 307, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 236, 308, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 237, 309, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 238, 310, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 239, 311, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 240, 312, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 241, 313, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 242, 314, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 243, 315, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 244, 316, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 245, 317, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 246, 318, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 247, 319, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 248, 320, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 249, 321, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 250, 322, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 251, 323, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 252, 324, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 253, 325, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 254, 326, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 255, 327, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 256, 328, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 257, 329, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 258, 330, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 259, 331, STRING_TOKEN (STR_REPAIR_REQUIRED), TRUE, EfiDriverHealthStatusRepairRequired }, { TRUE, 300, 000, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 301, 001, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 302, 002, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 303, 003, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 304, 004, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 305, 005, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 306, 006, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 307, 007, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 308, 000, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 309, 001, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 310, 002, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 311, 003, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 312, 004, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 313, 005, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 314, 006, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 315, 007, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 316, 000, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 317, 001, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 318, 002, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 319, 003, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 320, 004, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 321, 005, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 322, 006, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 323, 007, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 324, 000, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 325, 001, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 326, 002, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 327, 003, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 328, 004, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 329, 005, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 330, 006, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 331, 007, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 332, 100, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 333, 101, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 334, 102, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 335, 103, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 336, 104, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 337, 105, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 338, 106, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 339, 107, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 340, 108, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 341, 109, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 342, 110, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 343, 111, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 344, 112, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 345, 113, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 346, 114, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 347, 115, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 348, 100, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 349, 101, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 350, 102, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 351, 103, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 352, 104, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 353, 105, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 354, 106, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 355, 107, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 356, 108, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 357, 109, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 358, 110, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 359, 111, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 360, 112, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 361, 113, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 362, 114, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 363, 115, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 364, 100, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 365, 101, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 366, 102, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 367, 103, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 368, 104, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 369, 105, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 370, 106, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 371, 107, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 372, 108, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 373, 109, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 374, 110, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 375, 111, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 376, 112, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 377, 113, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 378, 114, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 379, 115, 0, FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 380, 100, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 381, 101, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 382, 102, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 383, 103, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 384, 104, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 385, 105, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 386, 106, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 387, 107, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 388, 108, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 389, 109, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 390, 110, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 391, 111, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 392, 112, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 393, 113, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 394, 114, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 395, 115, STRING_TOKEN (STR_CONFIG_WARNING), FALSE, EfiDriverHealthStatusConfigurationRequired }, { TRUE, 999, 999, 0, FALSE } }; HII_VENDOR_DEVICE_PATH mHiiVendorDevicePathDiskIoDummy = { { { HARDWARE_DEVICE_PATH, HW_VENDOR_DP, { (UINT8) (sizeof (VENDOR_DEVICE_PATH)), (UINT8) ((sizeof (VENDOR_DEVICE_PATH)) >> 8) } }, // // {C153B68E-EBFC-488e-B110-662867745BBE} // { 0xc153b68e, 0xebfc, 0x488e, { 0xb1, 0x10, 0x66, 0x28, 0x67, 0x74, 0x5b, 0xbe} } }, { END_DEVICE_PATH_TYPE, END_ENTIRE_DEVICE_PATH_SUBTYPE, { (UINT8) (END_DEVICE_PATH_LENGTH), (UINT8) ((END_DEVICE_PATH_LENGTH) >> 8) } } }; EFI_HII_HANDLE mHiiHandle = NULL; EFI_DRIVER_BINDING_PROTOCOL gDiskIoDriverBinding = { DiskIoDriverBindingSupported, DiskIoDriverBindingStart, DiskIoDriverBindingStop, 0xaa, NULL, NULL }; EFI_DRIVER_HEALTH_PROTOCOL gDiskIoDriverHealth = { DiskIoDriverHealthGetHealthStatus, DiskIoDriverHealthRepair }; // // Template for DiskIo private data structure. // The pointer to BlockIo protocol interface is assigned dynamically. // DISK_IO_PRIVATE_DATA gDiskIoPrivateDataTemplate = { DISK_IO_PRIVATE_DATA_SIGNATURE, { EFI_DISK_IO_PROTOCOL_REVISION, DiskIoReadDisk, DiskIoWriteDisk }, NULL, NULL, // Handle NULL, // Consumed Protocol NULL, // // Produced Protocol // { DummyExtractConfig, DummyRouteConfig, DummyDriverCallback }, // // NVdata // { 0x0 }, // // Controller Name // NULL, // // Controller Index // 0 }; DEVICE_STATE * GetDeviceState ( UINTN DeviceStateNumber ) { UINTN Index; for (Index = 0; mDeviceState[Index].CurrentState != 999 && mDeviceState[Index].CurrentState != DeviceStateNumber; Index++); ASSERT (mDeviceState[Index].CurrentState != 999); return &mDeviceState[Index]; } /** Test to see if this driver supports ControllerHandle. @param This Protocol instance pointer. @param ControllerHandle Handle of device to test @param RemainingDevicePath Optional parameter use to pick a specific child device to start. @retval EFI_SUCCESS This driver supports this device @retval EFI_ALREADY_STARTED This driver is already running on this device @retval other This driver does not support this device **/ EFI_STATUS EFIAPI DiskIoDriverBindingSupported ( IN EFI_DRIVER_BINDING_PROTOCOL *This, IN EFI_HANDLE ControllerHandle, IN EFI_DEVICE_PATH_PROTOCOL *RemainingDevicePath OPTIONAL ) { EFI_STATUS Status; EFI_BLOCK_IO_PROTOCOL *BlockIo; // // Open the IO Abstraction(s) needed to perform the supported test. // Status = gBS->OpenProtocol ( ControllerHandle, &gEfiBlockIoProtocolGuid, (VOID **) &BlockIo, This->DriverBindingHandle, ControllerHandle, EFI_OPEN_PROTOCOL_BY_DRIVER ); if (EFI_ERROR (Status)) { return Status; } // // Close the I/O Abstraction(s) used to perform the supported test. // gBS->CloseProtocol ( ControllerHandle, &gEfiBlockIoProtocolGuid, This->DriverBindingHandle, ControllerHandle ); return EFI_SUCCESS; } /** Start this driver on ControllerHandle by opening a Block IO protocol and installing a Disk IO protocol on ControllerHandle. @param This Protocol instance pointer. @param ControllerHandle Handle of device to bind driver to @param RemainingDevicePath Optional parameter use to pick a specific child device to start. @retval EFI_SUCCESS This driver is added to ControllerHandle @retval EFI_ALREADY_STARTED This driver is already running on ControllerHandle @retval other This driver does not support this device **/ EFI_STATUS EFIAPI DiskIoDriverBindingStart ( IN EFI_DRIVER_BINDING_PROTOCOL *This, IN EFI_HANDLE ControllerHandle, IN EFI_DEVICE_PATH_PROTOCOL *RemainingDevicePath OPTIONAL ) { EFI_STATUS Status; DISK_IO_PRIVATE_DATA *Private; DEVICE_STATE *DeviceState; UINTN DataSize; UINT32 StartCount; CONTROLLER_STATE *ControllerState; Private = NULL; ControllerState = NULL; // // Connect to the Block IO interface on ControllerHandle. // Status = gBS->OpenProtocol ( ControllerHandle, &gEfiBlockIoProtocolGuid, (VOID **) &gDiskIoPrivateDataTemplate.BlockIo, This->DriverBindingHandle, ControllerHandle, EFI_OPEN_PROTOCOL_BY_DRIVER ); if (EFI_ERROR (Status)) { return Status; } // // Initialize the Disk IO device instance. // Private = AllocateCopyPool (sizeof (DISK_IO_PRIVATE_DATA), &gDiskIoPrivateDataTemplate); if (Private == NULL) { Status = EFI_OUT_OF_RESOURCES; goto ErrorExit; } // // Begin Driver Health Protocol Support // DataSize = sizeof (StartCount); Status = gRT->GetVariable ( L"StartCount", &gEfiCallerIdGuid, NULL, &DataSize, &StartCount ); if (EFI_ERROR (Status)) { // // If the configuration can not be read, then set the default config value of 0 // StartCount = 0; } ControllerIndex++; DeviceState = GetDeviceState (mDeviceState[StartCount].CurrentState); ASSERT (DeviceState != NULL); ControllerState = AllocateZeroPool (sizeof (CONTROLLER_STATE)); if (ControllerState == NULL) { return EFI_OUT_OF_RESOURCES; } ControllerState->ControllerHandle = ControllerHandle; ControllerState->Signature = DISK_IO_CONTROLLER_STATE_SIGNATURE; ControllerState->DeviceStateNum = DeviceState->CurrentState; ControllerState->ChildHandle = NULL; ControllerState->ControllerIndex = ControllerIndex; InsertTailList (&mControllerList, &ControllerState->Link); if (DeviceState->HealthStatus != EfiDriverHealthStatusHealthy || DeviceState->StringId != 0) { mNumNotHealthy++; } StartCount++; while (!mDeviceState[StartCount].StartState) { if (mDeviceState[StartCount].CurrentState == 999) { StartCount = 0; } else { StartCount++; } } if (mDeviceState[StartCount].CurrentState == 999) { StartCount = 0; } Status = gRT->SetVariable ( L"StartCount", &gEfiCallerIdGuid, EFI_VARIABLE_NON_VOLATILE | EFI_VARIABLE_BOOTSERVICE_ACCESS, sizeof (StartCount), &StartCount ); ASSERT_EFI_ERROR (Status); if (DeviceState->HealthStatus == EfiDriverHealthStatusConfigurationRequired) { Private->NVdata.ConfigGood = 0; } else { Private->NVdata.ConfigGood = 1; } Status = gRT->SetVariable ( L"Config", &gEfiCallerIdGuid, EFI_VARIABLE_NON_VOLATILE | EFI_VARIABLE_BOOTSERVICE_ACCESS, sizeof (Private->NVdata.ConfigGood), &Private->NVdata.ConfigGood ); ASSERT_EFI_ERROR (Status); // // End Driver Health Protocol Support // // // Install protocol interfaces for the Disk IO device. // Status = gBS->InstallProtocolInterface ( &ControllerHandle, &gEfiDiskIoProtocolGuid, EFI_NATIVE_INTERFACE, &Private->DiskIo ); Private->ControllerIndex = ControllerIndex; AddName (Private); ErrorExit: if (EFI_ERROR (Status)) { if (Private != NULL) { FreeUnicodeStringTable (Private->ControllerNameTable); FreePool (Private); } gBS->CloseProtocol ( ControllerHandle, &gEfiBlockIoProtocolGuid, This->DriverBindingHandle, ControllerHandle ); } return Status; } /** Stop this driver on ControllerHandle by removing Disk IO protocol and closing the Block IO protocol on ControllerHandle. @param This Protocol instance pointer. @param ControllerHandle Handle of device to stop driver on @param NumberOfChildren Number of Handles in ChildHandleBuffer. If number of children is zero stop the entire bus driver. @param ChildHandleBuffer List of Child Handles to Stop. @retval EFI_SUCCESS This driver is removed ControllerHandle @retval other This driver was not removed from this device **/ EFI_STATUS EFIAPI DiskIoDriverBindingStop ( IN EFI_DRIVER_BINDING_PROTOCOL *This, IN EFI_HANDLE ControllerHandle, IN UINTN NumberOfChildren, IN EFI_HANDLE *ChildHandleBuffer ) { EFI_STATUS Status; EFI_DISK_IO_PROTOCOL *DiskIo; DISK_IO_PRIVATE_DATA *Private; DEVICE_STATE *DeviceState; CONTROLLER_STATE *ControllerState; LIST_ENTRY *Link; ControllerState = NULL; DeviceState = NULL; // // Get our context back. // Status = gBS->OpenProtocol ( ControllerHandle, &gEfiDiskIoProtocolGuid, (VOID **) &DiskIo, This->DriverBindingHandle, ControllerHandle, EFI_OPEN_PROTOCOL_GET_PROTOCOL ); if (EFI_ERROR (Status)) { return EFI_UNSUPPORTED; } Private = DISK_IO_PRIVATE_DATA_FROM_THIS (DiskIo); Status = gBS->UninstallProtocolInterface ( ControllerHandle, &gEfiDiskIoProtocolGuid, &Private->DiskIo ); if (!EFI_ERROR (Status)) { Status = gBS->CloseProtocol ( ControllerHandle, &gEfiBlockIoProtocolGuid, This->DriverBindingHandle, ControllerHandle ); // // Get the Controller State from global list // Link = GetFirstNode (&mControllerList); while (!IsNull (&mControllerList, Link)) { ControllerState = DISK_IO_CONTROLLER_STATE_FROM_LINK (Link); if (ControllerState->ControllerHandle == ControllerHandle) { DeviceState = GetDeviceState (ControllerState->DeviceStateNum); break; } Link = GetNextNode (&mControllerList, Link); } ASSERT (DeviceState != NULL); if (DeviceState->HealthStatus != EfiDriverHealthStatusHealthy || DeviceState->StringId != 0) { mNumNotHealthy--; } RemoveEntryList (Link); if (ControllerState != NULL) { FreePool (ControllerState); } } if (!EFI_ERROR (Status)) { FreeUnicodeStringTable (Private->ControllerNameTable); FreePool (Private); } ControllerIndex = 0; return Status; } /** Read BufferSize bytes from Offset into Buffer. Reads may support reads that are not aligned on sector boundaries. There are three cases: UnderRun - The first byte is not on a sector boundary or the read request is less than a sector in length. Aligned - A read of N contiguous sectors. OverRun - The last byte is not on a sector boundary. @param This Protocol instance pointer. @param MediaId Id of the media, changes every time the media is replaced. @param Offset The starting byte offset to read from @param BufferSize Size of Buffer @param Buffer Buffer containing read data @retval EFI_SUCCESS The data was read correctly from the device. @retval EFI_DEVICE_ERROR The device reported an error while performing the read. @retval EFI_NO_MEDIA There is no media in the device. @retval EFI_MEDIA_CHNAGED The MediaId does not matched the current device. @retval EFI_INVALID_PARAMETER The read request contains device addresses that are not valid for the device. **/ EFI_STATUS EFIAPI DiskIoReadDisk ( IN EFI_DISK_IO_PROTOCOL *This, IN UINT32 MediaId, IN UINT64 Offset, IN UINTN BufferSize, OUT VOID *Buffer ) { EFI_STATUS Status; DISK_IO_PRIVATE_DATA *Private; EFI_BLOCK_IO_PROTOCOL *BlockIo; EFI_BLOCK_IO_MEDIA *Media; UINT32 BlockSize; UINT64 Lba; UINT64 OverRunLba; UINT32 UnderRun; UINT32 OverRun; BOOLEAN TransactionComplete; UINTN WorkingBufferSize; UINT8 *WorkingBuffer; UINTN Length; UINT8 *Data; UINT8 *PreData; UINTN IsBufferAligned; UINTN DataBufferSize; BOOLEAN LastRead; Private = DISK_IO_PRIVATE_DATA_FROM_THIS (This); BlockIo = Private->BlockIo; Media = BlockIo->Media; BlockSize = Media->BlockSize; if (Media->MediaId != MediaId) { return EFI_MEDIA_CHANGED; } WorkingBuffer = Buffer; WorkingBufferSize = BufferSize; // // Allocate a temporary buffer for operation // DataBufferSize = BlockSize * DATA_BUFFER_BLOCK_NUM; if (Media->IoAlign > 1) { PreData = AllocatePool (DataBufferSize + Media->IoAlign); Data = PreData - ((UINTN) PreData & (Media->IoAlign - 1)) + Media->IoAlign; } else { PreData = AllocatePool (DataBufferSize); Data = PreData; } if (PreData == NULL) { return EFI_OUT_OF_RESOURCES; } Lba = DivU64x32Remainder (Offset, BlockSize, &UnderRun); Length = BlockSize - UnderRun; TransactionComplete = FALSE; Status = EFI_SUCCESS; if (UnderRun != 0) { // // Offset starts in the middle of an Lba, so read the entire block. // Status = BlockIo->ReadBlocks ( BlockIo, MediaId, Lba, BlockSize, Data ); if (EFI_ERROR (Status)) { goto Done; } if (Length > BufferSize) { Length = BufferSize; TransactionComplete = TRUE; } CopyMem (WorkingBuffer, Data + UnderRun, Length); WorkingBuffer += Length; WorkingBufferSize -= Length; if (WorkingBufferSize == 0) { goto Done; } Lba += 1; } OverRunLba = Lba + DivU64x32Remainder (WorkingBufferSize, BlockSize, &OverRun); if (!TransactionComplete && WorkingBufferSize >= BlockSize) { // // If the DiskIo maps directly to a BlockIo device do the read. // if (OverRun != 0) { WorkingBufferSize -= OverRun; } // // Check buffer alignment // IsBufferAligned = (UINTN) WorkingBuffer & (UINTN) (Media->IoAlign - 1); if (Media->IoAlign <= 1 || IsBufferAligned == 0) { // // Alignment is satisfied, so read them together // Status = BlockIo->ReadBlocks ( BlockIo, MediaId, Lba, WorkingBufferSize, WorkingBuffer ); if (EFI_ERROR (Status)) { goto Done; } WorkingBuffer += WorkingBufferSize; } else { // // Use the allocated buffer instead of the original buffer // to avoid alignment issue. // Here, the allocated buffer (8-byte align) can satisfy the alignment // LastRead = FALSE; do { if (WorkingBufferSize <= DataBufferSize) { // // It is the last calling to readblocks in this loop // DataBufferSize = WorkingBufferSize; LastRead = TRUE; } Status = BlockIo->ReadBlocks ( BlockIo, MediaId, Lba, DataBufferSize, Data ); if (EFI_ERROR (Status)) { goto Done; } CopyMem (WorkingBuffer, Data, DataBufferSize); WorkingBufferSize -= DataBufferSize; WorkingBuffer += DataBufferSize; Lba += DATA_BUFFER_BLOCK_NUM; } while (!LastRead); } } if (!TransactionComplete && OverRun != 0) { // // Last read is not a complete block. // Status = BlockIo->ReadBlocks ( BlockIo, MediaId, OverRunLba, BlockSize, Data ); if (EFI_ERROR (Status)) { goto Done; } CopyMem (WorkingBuffer, Data, OverRun); } Done: if (PreData != NULL) { FreePool (PreData); } return Status; } /** Writes BufferSize bytes from Buffer into Offset. Writes may require a read modify write to support writes that are not aligned on sector boundaries. There are three cases: UnderRun - The first byte is not on a sector boundary or the write request is less than a sector in length. Read modify write is required. Aligned - A write of N contiguous sectors. OverRun - The last byte is not on a sector boundary. Read modified write required. @param This Protocol instance pointer. @param MediaId Id of the media, changes every time the media is replaced. @param Offset The starting byte offset to read from @param BufferSize Size of Buffer @param Buffer Buffer containing read data @retval EFI_SUCCESS The data was written correctly to the device. @retval EFI_WRITE_PROTECTED The device can not be written to. @retval EFI_DEVICE_ERROR The device reported an error while performing the write. @retval EFI_NO_MEDIA There is no media in the device. @retval EFI_MEDIA_CHNAGED The MediaId does not matched the current device. @retval EFI_INVALID_PARAMETER The write request contains device addresses that are not valid for the device. **/ EFI_STATUS EFIAPI DiskIoWriteDisk ( IN EFI_DISK_IO_PROTOCOL *This, IN UINT32 MediaId, IN UINT64 Offset, IN UINTN BufferSize, IN VOID *Buffer ) { EFI_STATUS Status; DISK_IO_PRIVATE_DATA *Private; EFI_BLOCK_IO_PROTOCOL *BlockIo; EFI_BLOCK_IO_MEDIA *Media; UINT32 BlockSize; UINT64 Lba; UINT64 OverRunLba; UINT32 UnderRun; UINT32 OverRun; BOOLEAN TransactionComplete; UINTN WorkingBufferSize; UINT8 *WorkingBuffer; UINTN Length; UINT8 *Data; UINT8 *PreData; UINTN IsBufferAligned; UINTN DataBufferSize; BOOLEAN LastWrite; Private = DISK_IO_PRIVATE_DATA_FROM_THIS (This); BlockIo = Private->BlockIo; Media = BlockIo->Media; BlockSize = Media->BlockSize; if (Media->ReadOnly) { return EFI_WRITE_PROTECTED; } if (Media->MediaId != MediaId) { return EFI_MEDIA_CHANGED; } DataBufferSize = BlockSize * DATA_BUFFER_BLOCK_NUM; if (Media->IoAlign > 1) { PreData = AllocatePool (DataBufferSize + Media->IoAlign); Data = PreData - ((UINTN) PreData & (Media->IoAlign - 1)) + Media->IoAlign; } else { PreData = AllocatePool (DataBufferSize); Data = PreData; } if (PreData == NULL) { return EFI_OUT_OF_RESOURCES; } WorkingBuffer = Buffer; WorkingBufferSize = BufferSize; Lba = DivU64x32Remainder (Offset, BlockSize, &UnderRun); Length = BlockSize - UnderRun; TransactionComplete = FALSE; Status = EFI_SUCCESS; if (UnderRun != 0) { // // Offset starts in the middle of an Lba, so do read modify write. // Status = BlockIo->ReadBlocks ( BlockIo, MediaId, Lba, BlockSize, Data ); if (EFI_ERROR (Status)) { goto Done; } if (Length > BufferSize) { Length = BufferSize; TransactionComplete = TRUE; } CopyMem (Data + UnderRun, WorkingBuffer, Length); Status = BlockIo->WriteBlocks ( BlockIo, MediaId, Lba, BlockSize, Data ); if (EFI_ERROR (Status)) { goto Done; } WorkingBuffer += Length; WorkingBufferSize -= Length; if (WorkingBufferSize == 0) { goto Done; } Lba += 1; } OverRunLba = Lba + DivU64x32Remainder (WorkingBufferSize, BlockSize, &OverRun); if (!TransactionComplete && WorkingBufferSize >= BlockSize) { // // If the DiskIo maps directly to a BlockIo device do the write. // if (OverRun != 0) { WorkingBufferSize -= OverRun; } // // Check buffer alignment // IsBufferAligned = (UINTN) WorkingBuffer & (UINTN) (Media->IoAlign - 1); if (Media->IoAlign <= 1 || IsBufferAligned == 0) { // // Alignment is satisfied, so write them together // Status = BlockIo->WriteBlocks ( BlockIo, MediaId, Lba, WorkingBufferSize, WorkingBuffer ); if (EFI_ERROR (Status)) { goto Done; } WorkingBuffer += WorkingBufferSize; } else { // // The buffer parameter is not aligned with the request // So use the allocated instead. // It can fit almost all the cases. // LastWrite = FALSE; do { if (WorkingBufferSize <= DataBufferSize) { // // It is the last calling to writeblocks in this loop // DataBufferSize = WorkingBufferSize; LastWrite = TRUE; } CopyMem (Data, WorkingBuffer, DataBufferSize); Status = BlockIo->WriteBlocks ( BlockIo, MediaId, Lba, DataBufferSize, Data ); if (EFI_ERROR (Status)) { goto Done; } WorkingBufferSize -= DataBufferSize; WorkingBuffer += DataBufferSize; Lba += DATA_BUFFER_BLOCK_NUM; } while (!LastWrite); } } if (!TransactionComplete && OverRun != 0) { // // Last bit is not a complete block, so do a read modify write. // Status = BlockIo->ReadBlocks ( BlockIo, MediaId, OverRunLba, BlockSize, Data ); if (EFI_ERROR (Status)) { goto Done; } CopyMem (Data, WorkingBuffer, OverRun); Status = BlockIo->WriteBlocks ( BlockIo, MediaId, OverRunLba, BlockSize, Data ); if (EFI_ERROR (Status)) { goto Done; } } Done: if (PreData != NULL) { FreePool (PreData); } return Status; } /** Retrieves the health status of a controller in the platform. This function can also optionally return warning messages, error messages, and a set of HII Forms that may be repair a controller that is not proper configured. @param This A pointer to the EFI_DRIVER_HEALTH_PROTOCOL instance. @param ControllerHandle The handle of the controller to retrieve the health status on. This is an optional parameter that may be NULL. If this parameter is NULL, then the value of ChildHandle is ignored, and the combined health status of all the devices that the driver is managing is returned. @param ChildHandle The handle of the child controller to retrieve the health status on. This is an optional parameter that may be NULL. This parameter is ignored of ControllerHandle is NULL. It will be NULL for device drivers. It will also be NULL for bus drivers when an attempt is made to collect the health status of the bus controller. If will not be NULL when an attempt is made to collect the health status for a child controller produced by the driver. @param HealthStatus A pointer to the health status that is returned by this function. This is an optional parameter that may be NULL. This parameter is ignored of ControllerHandle is NULL. The health status for the controller specified by ControllerHandle and ChildHandle is returned. @param MessageList A pointer to an array of warning or error messages associated with the controller specified by ControllerHandle and ChildHandle. This is an optional parameter that may be NULL. MessageList is allocated by this function with the EFI Boot Service AllocatePool(), and it is the caller's responsibility to free MessageList with the EFI Boot Service FreePool(). Each message is specified by tuple of an EFI_HII_HANDLE and an EFI_STRING_ID. The array of messages is terminated by tuple containing a EFI_HII_HANDLE with a value of NULL. The EFI_HII_STRING_PROTOCOL.GetString() function can be used to retrieve the warning or error message as a Null-terminated Unicode string in a specific language. Messages may be returned for any of the HealthStatus values except EfiDriverHealthStatusReconnectRequired and EfiDriverHealthStatusRebootRequired. @param FormHiiHandle A pointer to the HII handle for an HII form associated with the controller specified by ControllerHandle and ChildHandle. This is an optional parameter that may be NULL. An HII form is specified by a combination of an EFI_HII_HANDLE and an EFI_GUID that identifies the Form Set GUID. The EFI_FORM_BROWSER2_PROTOCOL.SendForm() function can be used to display and allow the user to make configuration changes to the HII Form. An HII form may only be returned with a HealthStatus value of EfiDriverHealthStatusConfigurationRequired. @retval EFI_SUCCESS ControllerHandle is NULL, and all the controllers managed by this driver specified by This have a health status of EfiDriverHealthStatusHealthy with no warning messages to be returned. The ChildHandle, HealthStatus, MessageList, and FormList parameters are ignored. @retval EFI_DEVICE_ERROR ControllerHandle is NULL, and one or more of the controllers managed by this driver specified by This do not have a health status of EfiDriverHealthStatusHealthy. The ChildHandle, HealthStatus, MessageList, and FormList parameters are ignored. @retval EFI_DEVICE_ERROR ControllerHandle is NULL, and one or more of the controllers managed by this driver specified by This have one or more warning and/or error messages. The ChildHandle, HealthStatus, MessageList, and FormList parameters are ignored. @retval EFI_SUCCESS ControllerHandle is not NULL and the health status of the controller specified by ControllerHandle and ChildHandle was returned in HealthStatus. A list of warning and error messages may be optionally returned in MessageList, and a list of HII Forms may be optionally returned in FormList. @retval EFI_UNSUPPORTED ControllerHandle is not NULL, and the controller specified by ControllerHandle and ChildHandle is not currently being managed by the driver specified by This. @retval EFI_INVALID_PARAMETER HealthStatus is NULL. @retval EFI_OUT_OF_RESOURCES MessageList is not NULL, and there are not enough resource available to allocate memory for MessageList. **/ EFI_STATUS DiskIoDriverHealthGetHealthStatus ( IN EFI_DRIVER_HEALTH_PROTOCOL *This, IN EFI_HANDLE ControllerHandle OPTIONAL, IN EFI_HANDLE ChildHandle OPTIONAL, OUT EFI_DRIVER_HEALTH_STATUS *HealthStatus, OUT EFI_DRIVER_HEALTH_HII_MESSAGE **MessageList OPTIONAL, OUT EFI_HII_HANDLE *FormHiiHandle OPTIONAL ) { EFI_STATUS Status; EFI_DISK_IO_PROTOCOL *DiskIo; DISK_IO_PRIVATE_DATA *Private; DEVICE_STATE *DeviceState; CONTROLLER_STATE *ControllerState; LIST_ENTRY *Link; UINTN BufferSize; ControllerState = NULL; DeviceState = NULL; if (HealthStatus == NULL) { return EFI_INVALID_PARAMETER; } if (ControllerHandle == NULL) { *HealthStatus = EfiDriverHealthStatusHealthy; if (mNumNotHealthy != 0) { *HealthStatus = EfiDriverHealthStatusFailed; } return EFI_SUCCESS; } // // This is a device driver, so ChildHandle must be NULL. // if (ChildHandle != NULL) { return EFI_UNSUPPORTED; } // // Make sure this driver is currently managing ControllerHandle // Status = EfiTestManagedDevice ( ControllerHandle, gDiskIoDriverBinding.DriverBindingHandle, &gEfiBlockIoProtocolGuid ); if (EFI_ERROR (Status)) { return Status; } if (HealthStatus == NULL) { return EFI_INVALID_PARAMETER; } Status = gBS->HandleProtocol (ControllerHandle, &gEfiDiskIoProtocolGuid, (VOID **) &DiskIo); if (EFI_ERROR (Status)) { return EFI_UNSUPPORTED; } if (HealthStatus == NULL) { return EFI_INVALID_PARAMETER; } Private = DISK_IO_PRIVATE_DATA_FROM_THIS (DiskIo); // // Get the Controller State from global list // Link = GetFirstNode (&mControllerList); while (!IsNull (&mControllerList, Link)) { ControllerState = DISK_IO_CONTROLLER_STATE_FROM_LINK (Link); if (ControllerState->ControllerHandle == ControllerHandle) { DeviceState = GetDeviceState (ControllerState->DeviceStateNum); break; } Link = GetNextNode (&mControllerList, Link); } ASSERT (DeviceState != NULL); if (DeviceState->HealthStatus == EfiDriverHealthStatusConfigurationRequired) { // // Read the configuration for this device // BufferSize = sizeof (Private->NVdata.ConfigGood); Status = gRT->GetVariable ( L"Config", &gEfiCallerIdGuid, NULL, &BufferSize, &Private->NVdata.ConfigGood ); ASSERT_EFI_ERROR (Status); // // If the config value is 1, then the configuration is valid and the state machine can move to the next state // Otherwise, the state machine returns ConfigurationRequired again // if (Private->NVdata.ConfigGood == ControllerState->ControllerIndex) { if (DeviceState->HealthStatus != EfiDriverHealthStatusHealthy || DeviceState->StringId != 0) { mNumNotHealthy--; } ControllerState->DeviceStateNum = DeviceState->NextState; DeviceState = GetDeviceState (ControllerState->DeviceStateNum); ASSERT (DeviceState != NULL); if (DeviceState->HealthStatus != EfiDriverHealthStatusHealthy) { mNumNotHealthy++; } } } *HealthStatus = DeviceState->HealthStatus; if (MessageList != NULL) { *MessageList = NULL; if (DeviceState->StringId != 0) { *MessageList = AllocateZeroPool (sizeof(EFI_DRIVER_HEALTH_HII_MESSAGE) * 2); if (*MessageList == NULL) { return EFI_UNSUPPORTED; } (*MessageList)[0].HiiHandle = mHiiHandle; (*MessageList)[0].StringId = DeviceState->StringId; } else { *MessageList = AllocateZeroPool (sizeof(EFI_DRIVER_HEALTH_HII_MESSAGE) * 1); if (*MessageList == NULL) { return EFI_UNSUPPORTED; } } } if (FormHiiHandle != NULL) { *FormHiiHandle = mHiiHandle; } if (DeviceState->HealthStatus == EfiDriverHealthStatusConfigurationRequired) { Private->NVdata.ConfigGood = 0; Status = gRT->SetVariable ( L"Config", &gEfiCallerIdGuid, EFI_VARIABLE_NON_VOLATILE | EFI_VARIABLE_BOOTSERVICE_ACCESS, sizeof (Private->NVdata.ConfigGood), &Private->NVdata.ConfigGood ); ASSERT_EFI_ERROR (Status); } return EFI_SUCCESS; } /** Performs a repair operation on a controller in the platform. This function can optionally report repair progress information back to the platform. @param This A pointer to the EFI_DRIVER_HEALTH_PROTOCOL instance. @param ControllerHandle The handle of the controller to repair. @param ChildHandle The handle of the child controller to repair. This is an optional parameter that may be NULL. It will be NULL for device drivers. It will also be NULL for bus drivers when an attempt is made to repair a bus controller. If will not be NULL when an attempt is made to repair a child controller produced by the driver. @param RepairNotify A notification function that may be used by a driver to report the progress of the repair operation. This is an optional parameter that may be NULL. @retval EFI_SUCCESS An attempt to repair the controller specified by ControllerHandle and ChildHandle was performed. The result of the repair operation can be determined by calling GetHealthStatus(). @retval EFI_UNSUPPORTED The driver specified by This is not currently managing the controller specified by ControllerHandle and ChildHandle. @retval EFI_OUT_OF_RESOURCES There are not enough resources to perform the repair operation. */ EFI_STATUS DiskIoDriverHealthRepair ( IN EFI_DRIVER_HEALTH_PROTOCOL *This, IN EFI_HANDLE ControllerHandle, IN EFI_HANDLE ChildHandle OPTIONAL, IN EFI_DRIVER_HEALTH_REPAIR_PROGRESS_NOTIFY RepairNotify OPTIONAL ) { EFI_STATUS Status; UINTN Index; DEVICE_STATE *DeviceState; CONTROLLER_STATE *ControllerState; LIST_ENTRY *Link; EFI_DISK_IO_PROTOCOL *DiskIo; DISK_IO_PRIVATE_DATA *Private; Index = 0; ControllerState = NULL; DeviceState = NULL; // // This is a device driver, so ChildHandle must be NULL. // if (ChildHandle != NULL) { return EFI_UNSUPPORTED; } // // Make sure this driver is currently managing ControllerHandle // Status = EfiTestManagedDevice ( ControllerHandle, gDiskIoDriverBinding.DriverBindingHandle, &gEfiBlockIoProtocolGuid ); if (EFI_ERROR (Status)) { return Status; } Status = gBS->HandleProtocol (ControllerHandle, &gEfiDiskIoProtocolGuid, (VOID **) &DiskIo); if (EFI_ERROR (Status)) { return EFI_UNSUPPORTED; } Private = DISK_IO_PRIVATE_DATA_FROM_THIS (DiskIo); Link = GetFirstNode (&mControllerList); while (!IsNull (&mControllerList, Link)) { ControllerState = DISK_IO_CONTROLLER_STATE_FROM_LINK (Link); if (ControllerState->ControllerHandle == ControllerHandle) { DeviceState = GetDeviceState (ControllerState->DeviceStateNum); break; } Link = GetNextNode (&mControllerList, Link); } ASSERT (DeviceState != NULL); // // Check to see if the controller has already been repaired // if (DeviceState->HealthStatus != EfiDriverHealthStatusRepairRequired) { return EFI_SUCCESS; } if (DeviceState->RepairNotify) { do { RepairNotify(Index, 10); Index++; } while ((gBS->Stall(100000) == EFI_SUCCESS) && (Index < 10)); } if (DeviceState->HealthStatus != EfiDriverHealthStatusHealthy || DeviceState->StringId != 0) { mNumNotHealthy--; } // // Repair success, go to next state // ControllerState->DeviceStateNum = DeviceState->NextState; DeviceState = GetDeviceState (ControllerState->DeviceStateNum); ASSERT (DeviceState != NULL); if (DeviceState->HealthStatus != EfiDriverHealthStatusHealthy || DeviceState->StringId != 0) { mNumNotHealthy++; } if (DeviceState->HealthStatus == EfiDriverHealthStatusConfigurationRequired) { Private->NVdata.ConfigGood = 0; Status = gRT->SetVariable ( L"Config", &gEfiCallerIdGuid, EFI_VARIABLE_NON_VOLATILE | EFI_VARIABLE_BOOTSERVICE_ACCESS, sizeof (Private->NVdata.ConfigGood), &Private->NVdata.ConfigGood ); ASSERT_EFI_ERROR (Status); } return EFI_SUCCESS; } /** The user Entry Point for module DiskIo. The user code starts with this function. @param[in] ImageHandle The firmware allocated handle for the EFI image. @param[in] SystemTable A pointer to the EFI System Table. @retval EFI_SUCCESS The entry point is executed successfully. @retval other Some error occurs when executing this entry point. **/ EFI_STATUS EFIAPI InitializeDiskIo ( IN EFI_HANDLE ImageHandle, IN EFI_SYSTEM_TABLE *SystemTable ) { EFI_STATUS Status; // // Install driver model protocol(s). // Status = EfiLibInstallDriverBindingComponentName2 ( ImageHandle, SystemTable, &gDiskIoDriverBinding, ImageHandle, &gDiskIoComponentName, &gDiskIoComponentName2 ); ASSERT_EFI_ERROR (Status); Status = gBS->InstallProtocolInterface ( &ImageHandle, &gEfiDriverHealthProtocolGuid, EFI_NATIVE_INTERFACE, &gDiskIoDriverHealth ); ASSERT_EFI_ERROR (Status); InitializeListHead (&mControllerList); gDiskIoPrivateDataTemplate.Handle = ImageHandle; Status = DiskIoConfigFormInit (); ASSERT_EFI_ERROR (Status); return Status; } /** Initialize the serial configuration form. @retval EFI_SUCCESS The serial configuration form is initialized. @retval EFI_OUT_OF_RESOURCES Failed to allocate memory. @retval Others Other errors as indicated. **/ EFI_STATUS DiskIoConfigFormInit ( VOID ) { EFI_STATUS Status; // // Locate Hii Database protocol // Status = gBS->LocateProtocol (&gEfiHiiDatabaseProtocolGuid, NULL, (VOID **)&gDiskIoPrivateDataTemplate.HiiDatabase); if (EFI_ERROR (Status)) { return Status; } Status = gBS->LocateProtocol (&gEfiHiiConfigRoutingProtocolGuid, NULL, (VOID **)&gDiskIoPrivateDataTemplate.HiiConfigRouting); if (EFI_ERROR (Status)) { return Status; } // // Install Device Path Protocol and Config Access protocol to driver handle // Status = gBS->InstallMultipleProtocolInterfaces ( &gDiskIoPrivateDataTemplate.Handle, &gEfiDevicePathProtocolGuid, &mHiiVendorDevicePathDiskIoDummy, &gEfiHiiConfigAccessProtocolGuid, &gDiskIoPrivateDataTemplate.ConfigAccess, NULL ); if (EFI_ERROR (Status)) { return Status; } // // Publish our HII data // mHiiHandle = HiiAddPackages ( &gEfiCallerIdGuid, gDiskIoPrivateDataTemplate.Handle, DriverHealthDxeStrings, DriverHealthVfrBin, NULL ); if (mHiiHandle == NULL) { return EFI_OUT_OF_RESOURCES; } return EFI_SUCCESS; } /** This function processes the results of changes in configuration. @param This Points to the EFI_HII_CONFIG_ACCESS_PROTOCOL. @param Action Specifies the type of action taken by the browser. @param QuestionId A unique value which is sent to the original exporting driver so that it can identify the type of data to expect. @param Type The type of value for the question. @param Value A pointer to the data being sent to the original exporting driver. @param ActionRequest On return, points to the action requested by the callback function. @retval EFI_SUCCESS The callback successfully handled the action. @retval EFI_OUT_OF_RESOURCES Not enough storage is available to hold the variable and its data. @retval EFI_DEVICE_ERROR The variable could not be saved. @retval EFI_UNSUPPORTED The specified Action is not supported by the callback. **/ EFI_STATUS EFIAPI DummyDriverCallback ( IN CONST EFI_HII_CONFIG_ACCESS_PROTOCOL *This, IN EFI_BROWSER_ACTION Action, IN EFI_QUESTION_ID QuestionId, IN UINT8 Type, IN EFI_IFR_TYPE_VALUE *Value, OUT EFI_BROWSER_ACTION_REQUEST *ActionRequest ) { DISK_IO_NV_DATA *IfrNvData; // // Retrieve uncommitted data from Browser // IfrNvData = AllocateZeroPool (sizeof (DISK_IO_NV_DATA)); ASSERT (IfrNvData != NULL); if (!HiiGetBrowserData (&gEfiCallerIdGuid, VariableName, sizeof (DISK_IO_NV_DATA), (UINT8 *) IfrNvData)) { FreePool (IfrNvData); return EFI_NOT_FOUND; } if ((Value == NULL) || (ActionRequest == NULL)) { return EFI_INVALID_PARAMETER; } *ActionRequest = EFI_BROWSER_ACTION_REQUEST_EXIT; return EFI_SUCCESS; } /** This function allows a caller to extract the current configuration for one or more named elements from the target driver. @param This Points to the EFI_HII_CONFIG_ACCESS_PROTOCOL. @param Request A null-terminated Unicode string in format. @param Progress On return, points to a character in the Request string. Points to the string's null terminator if request was successful. Points to the most recent '&' before the first failing name/value pair (or the beginning of the string if the failure is in the first name/value pair) if the request was not successful. @param Results A null-terminated Unicode string in format which has all values filled in for the names in the Request string. String to be allocated by the called function. @retval EFI_SUCCESS The Results is filled with the requested values. @retval EFI_OUT_OF_RESOURCES Not enough memory to store the results. @retval EFI_INVALID_PARAMETER Request is NULL, illegal syntax, or unknown name. @retval EFI_NOT_FOUND Routing data doesn't match any storage in this driver. **/ EFI_STATUS EFIAPI DummyExtractConfig ( IN CONST EFI_HII_CONFIG_ACCESS_PROTOCOL *This, IN CONST EFI_STRING Request, OUT EFI_STRING *Progress, OUT EFI_STRING *Results ) { EFI_STATUS Status; UINTN BufferSize; DISK_IO_PRIVATE_DATA *PrivateData; EFI_HII_CONFIG_ROUTING_PROTOCOL *HiiConfigRouting; EFI_STRING ConfigRequest; EFI_STRING ConfigRequestHdr; UINTN Size; if (Progress == NULL || Results == NULL || Request == NULL) { return EFI_INVALID_PARAMETER; } // // Initialize the local variables. // ConfigRequestHdr = NULL; ConfigRequest = NULL; Size = 0; *Progress = Request; PrivateData = DISK_IO_PRIVATE_DATA_FROM_CONFIG_ACCESS(This); HiiConfigRouting = PrivateData->HiiConfigRouting; // // Get Buffer Storage data from EFI variable. // Try to get the current setting from variable. // BufferSize = sizeof (DISK_IO_NV_DATA); Status = gRT->GetVariable ( VariableName, &gEfiCallerIdGuid, NULL, &BufferSize, &PrivateData->NVdata ); if (EFI_ERROR (Status)) { return EFI_NOT_FOUND; } if (Request == NULL) { // // Request is set to NULL, construct full request string. // // // Allocate and fill a buffer large enough to hold the template // followed by "&OFFSET=0&WIDTH=WWWWWWWWWWWWWWWW" followed by a Null-terminator // ConfigRequestHdr = HiiConstructConfigHdr (&gEfiCallerIdGuid, VariableName, PrivateData->Handle); Size = (StrLen (ConfigRequest) + 32 + 1) * sizeof (CHAR16); ConfigRequest = AllocateZeroPool (Size); UnicodeSPrint (ConfigRequest, Size, L"%s&OFFSET=0&WIDTH=%016LX", ConfigRequestHdr, (UINT64)BufferSize); FreePool (ConfigRequestHdr); } else { // // Check routing data in . // Note: if only one Storage is used, then this checking could be skipped. // if (!HiiIsConfigHdrMatch (Request, &gEfiCallerIdGuid, VariableName)) { return EFI_NOT_FOUND; } ConfigRequest = Request; } // // Convert buffer data to by helper function BlockToConfig() // Status = HiiConfigRouting->BlockToConfig ( HiiConfigRouting, ConfigRequest, (UINT8 *) &PrivateData->NVdata, BufferSize, Results, Progress ); if (Request == NULL) { FreePool (ConfigRequest); *Progress = NULL; } return Status; } /** This function processes the results of changes in configuration. @param This Points to the EFI_HII_CONFIG_ACCESS_PROTOCOL. @param Configuration A null-terminated Unicode string in format. @param Progress A pointer to a string filled in with the offset of the most recent '&' before the first failing name/value pair (or the beginning of the string if the failure is in the first name/value pair) or the terminating NULL if all was successful. @retval EFI_SUCCESS The Results is processed successfully. @retval EFI_INVALID_PARAMETER Configuration is NULL. @retval EFI_NOT_FOUND Routing data doesn't match any storage in this driver. **/ EFI_STATUS EFIAPI DummyRouteConfig ( IN CONST EFI_HII_CONFIG_ACCESS_PROTOCOL *This, IN CONST EFI_STRING Configuration, OUT EFI_STRING *Progress ) { EFI_STATUS Status; UINTN BufferSize; DISK_IO_PRIVATE_DATA *PrivateData; EFI_HII_CONFIG_ROUTING_PROTOCOL *HiiConfigRouting; if (Configuration == NULL || Progress == NULL) { return EFI_INVALID_PARAMETER; } PrivateData = DISK_IO_PRIVATE_DATA_FROM_CONFIG_ACCESS (This); HiiConfigRouting = PrivateData->HiiConfigRouting; *Progress = Configuration; // // Check routing data in . // Note: if only one Storage is used, then this checking could be skipped. // if (!HiiIsConfigHdrMatch (Configuration, &gEfiCallerIdGuid, VariableName)) { return EFI_NOT_FOUND; } // // Get Buffer Storage data from EFI variable // BufferSize = sizeof (DISK_IO_NV_DATA); Status = gRT->GetVariable ( VariableName, &gEfiCallerIdGuid, NULL, &BufferSize, &PrivateData->NVdata ); if (EFI_ERROR (Status)) { return Status; } // // Convert to buffer data by helper function ConfigToBlock() // BufferSize = sizeof (DISK_IO_NV_DATA); Status = HiiConfigRouting->ConfigToBlock ( HiiConfigRouting, Configuration, (UINT8 *) &PrivateData->NVdata, &BufferSize, Progress ); if (EFI_ERROR (Status)) { return Status; } // // Store Buffer Storage back to EFI variable // Status = gRT->SetVariable( VariableName, &gEfiCallerIdGuid, EFI_VARIABLE_NON_VOLATILE | EFI_VARIABLE_BOOTSERVICE_ACCESS, sizeof (DISK_IO_NV_DATA), &PrivateData->NVdata ); return Status; }