forked from xt-sys/exectos
		
	Implement XTLDR modules support
This commit is contained in:
		| @@ -58,6 +58,11 @@ BlEfiPrint(IN PUINT16 Format, | |||||||
| EFI_STATUS | EFI_STATUS | ||||||
| BlEnumerateEfiBlockDevices(); | BlEnumerateEfiBlockDevices(); | ||||||
|  |  | ||||||
|  | EFI_STATUS | ||||||
|  | BlFindVolumeDevicePath(IN PEFI_DEVICE_PATH_PROTOCOL FsHandle, | ||||||
|  |                        IN CONST PWCHAR FileSystemPath, | ||||||
|  |                        OUT PEFI_DEVICE_PATH_PROTOCOL* DevicePath); | ||||||
|  |  | ||||||
| EFI_STATUS | EFI_STATUS | ||||||
| BlGetVolumeDevicePath(IN PUCHAR SystemPath, | BlGetVolumeDevicePath(IN PUCHAR SystemPath, | ||||||
|                       OUT PEFI_DEVICE_PATH_PROTOCOL *DevicePath, |                       OUT PEFI_DEVICE_PATH_PROTOCOL *DevicePath, | ||||||
|   | |||||||
| @@ -186,6 +186,89 @@ BlEnumerateEfiBlockDevices() | |||||||
|     return STATUS_EFI_SUCCESS; |     return STATUS_EFI_SUCCESS; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Finds an EFI device path for a specified path on a given file system. | ||||||
|  |  * | ||||||
|  |  * @param FsHandle | ||||||
|  |  *        The handle of the corresponding file system. | ||||||
|  |  * | ||||||
|  |  * @param FileSystemPath | ||||||
|  |  *        Specifies a path on the corresponding file system. | ||||||
|  |  * | ||||||
|  |  * @param DevicePath | ||||||
|  |  *        Specifies a pointer to the memory area, where found device path will be stored. | ||||||
|  |  * | ||||||
|  |  * @return This routine returns a status code. | ||||||
|  |  * | ||||||
|  |  * @since XT 1.0 | ||||||
|  |  */ | ||||||
|  | EFI_STATUS | ||||||
|  | BlFindVolumeDevicePath(IN PEFI_DEVICE_PATH_PROTOCOL FsHandle, | ||||||
|  |                        IN CONST PWCHAR FileSystemPath, | ||||||
|  |                        OUT PEFI_DEVICE_PATH_PROTOCOL* DevicePath) | ||||||
|  | { | ||||||
|  |     EFI_STATUS Status; | ||||||
|  |     SIZE_T FsPathLength, DevicePathLength = 0; | ||||||
|  |     PEFI_FILEPATH_DEVICE_PATH FilePath = NULL; | ||||||
|  |     PEFI_DEVICE_PATH_PROTOCOL EndDevicePath; | ||||||
|  |     PEFI_DEVICE_PATH_PROTOCOL DevicePathHandle; | ||||||
|  |  | ||||||
|  |     /* Set local device path handle */ | ||||||
|  |     DevicePathHandle = FsHandle; | ||||||
|  |  | ||||||
|  |     /* Find the end device path node */ | ||||||
|  |     while(TRUE) { | ||||||
|  |         /* Make sure there is a next node */ | ||||||
|  |         if(*(PUSHORT)DevicePathHandle->Length == 0) | ||||||
|  |         { | ||||||
|  |             /* End device path not found */ | ||||||
|  |             return STATUS_EFI_NOT_FOUND; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         /* Check if end device path node found */ | ||||||
|  |         if(DevicePathHandle->Type == EFI_END_DEVICE_PATH) | ||||||
|  |         { | ||||||
|  |             /* End device path node found */ | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         /* Get next node */ | ||||||
|  |         DevicePathLength += *(PUSHORT)DevicePathHandle->Length; | ||||||
|  |         DevicePathHandle = (PEFI_DEVICE_PATH_PROTOCOL)((PUCHAR)DevicePathHandle + *(PUSHORT)DevicePathHandle->Length); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* Check real path length */ | ||||||
|  |     FsPathLength = RtlWideStringLength(FileSystemPath, 0) * sizeof(WCHAR); | ||||||
|  |  | ||||||
|  |     /* Allocate memory pool for device path */ | ||||||
|  |     Status = BlEfiMemoryAllocatePool(FsPathLength + DevicePathLength + sizeof(EFI_DEVICE_PATH_PROTOCOL), | ||||||
|  |                                      (PVOID *)DevicePath); | ||||||
|  |     if(Status != STATUS_EFI_SUCCESS) | ||||||
|  |     { | ||||||
|  |         /* Memory allocation failure */ | ||||||
|  |         return Status; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* Set file path */ | ||||||
|  |     RtlCopyMemory(*DevicePath, FsHandle, DevicePathLength); | ||||||
|  |     FilePath = (PEFI_FILEPATH_DEVICE_PATH)((PUCHAR)*DevicePath + DevicePathLength); | ||||||
|  |     FilePath->Header.Type = EFI_MEDIA_DEVICE_PATH; | ||||||
|  |     FilePath->Header.SubType = EFI_MEDIA_FILEPATH_DP; | ||||||
|  |     FilePath->Header.Length[0] = (UCHAR)FsPathLength + FIELD_OFFSET(EFI_FILEPATH_DEVICE_PATH, PathName) + sizeof(WCHAR); | ||||||
|  |     FilePath->Header.Length[1] = FilePath->Header.Length[0] >> 8; | ||||||
|  |  | ||||||
|  |     /* Set device path end node */ | ||||||
|  |     RtlCopyMemory(FilePath->PathName, FileSystemPath, FsPathLength + sizeof(WCHAR)); | ||||||
|  |     EndDevicePath = (PEFI_DEVICE_PATH_PROTOCOL)&FilePath->PathName[(FsPathLength / sizeof(WCHAR)) + 1]; | ||||||
|  |     EndDevicePath->Type = EFI_END_DEVICE_PATH; | ||||||
|  |     EndDevicePath->SubType = EFI_END_ENTIRE_DP; | ||||||
|  |     EndDevicePath->Length[0] = sizeof(EFI_DEVICE_PATH_PROTOCOL); | ||||||
|  |     EndDevicePath->Length[1] = 0; | ||||||
|  |  | ||||||
|  |     /* Return success */ | ||||||
|  |     return STATUS_EFI_SUCCESS; | ||||||
|  | } | ||||||
|  |  | ||||||
| EFI_STATUS | EFI_STATUS | ||||||
| BlGetVolumeDevicePath(IN PUCHAR SystemPath, | BlGetVolumeDevicePath(IN PUCHAR SystemPath, | ||||||
|                       OUT PEFI_DEVICE_PATH_PROTOCOL *DevicePath, |                       OUT PEFI_DEVICE_PATH_PROTOCOL *DevicePath, | ||||||
|   | |||||||
							
								
								
									
										211
									
								
								xtldr/xtldr.c
									
									
									
									
									
								
							
							
						
						
									
										211
									
								
								xtldr/xtldr.c
									
									
									
									
									
								
							| @@ -15,6 +15,9 @@ EFI_HANDLE EfiImageHandle; | |||||||
| /* EFI System Table */ | /* EFI System Table */ | ||||||
| PEFI_SYSTEM_TABLE EfiSystemTable; | PEFI_SYSTEM_TABLE EfiSystemTable; | ||||||
|  |  | ||||||
|  | /* EFI Secure Boot status */ | ||||||
|  | INT_PTR EfiSecureBoot; | ||||||
|  |  | ||||||
| /* Serial port configuration */ | /* Serial port configuration */ | ||||||
| CPPORT EfiSerialPort; | CPPORT EfiSerialPort; | ||||||
|  |  | ||||||
| @@ -28,8 +31,215 @@ CPPORT EfiSerialPort; | |||||||
| EFI_STATUS | EFI_STATUS | ||||||
| BlLoadEfiModules() | BlLoadEfiModules() | ||||||
| { | { | ||||||
|  |     CONST PWCHAR ModulesDirPath = L"\\EFI\\BOOT\\XTLDR\\"; | ||||||
|  |     EFI_GUID DevicePathGuid = EFI_DEVICE_PATH_PROTOCOL_GUID; | ||||||
|  |     EFI_GUID LIPGuid = EFI_LOADED_IMAGE_PROTOCOL_GUID; | ||||||
|  |     PEFI_DEVICE_PATH_PROTOCOL VolumeDevicePath, DevicePath; | ||||||
|  |     PEFI_LOADED_IMAGE_PROTOCOL LoadedImage; | ||||||
|  |     PEFI_FILE_HANDLE FsHandle, ModulesDir; | ||||||
|  |     EFI_HANDLE DiskHandle, ModuleHandle; | ||||||
|  |     PEFI_HANDLE Handles; | ||||||
|  |     SIZE_T Length; | ||||||
|     EFI_STATUS Status; |     EFI_STATUS Status; | ||||||
|  |     UINT_PTR DirSize, ModulesCount; | ||||||
|  |     CHAR Buffer[1024]; | ||||||
|  |     WCHAR ModulePath[1024]; | ||||||
|  |     PWCHAR ModuleName; | ||||||
|  |     UINT Index; | ||||||
|  |  | ||||||
|  |     /* Open EFI volume */ | ||||||
|  |     Status = BlOpenVolume(NULL, &DiskHandle, &FsHandle); | ||||||
|  |     if(Status != STATUS_EFI_SUCCESS) | ||||||
|  |     { | ||||||
|  |         /* Failed to open a volume */ | ||||||
|  |         return Status; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* Open EFI/BOOT/XTLDR directory, which contains all the modules and close the FS immediately */ | ||||||
|  |     Status = FsHandle->Open(FsHandle, &ModulesDir, ModulesDirPath, EFI_FILE_MODE_READ, 0); | ||||||
|  |     FsHandle->Close(FsHandle); | ||||||
|  |  | ||||||
|  |     /* Check if modules directory opened successfully */ | ||||||
|  |     if(Status == STATUS_EFI_NOT_FOUND) | ||||||
|  |     { | ||||||
|  |         /* Directory not found, nothing to load */ | ||||||
|  |         BlDbgPrint(L"WARNING: Boot loader directory (EFI/BOOT/XTLDR) not found\n"); | ||||||
|  |  | ||||||
|  |         /* Close volume */ | ||||||
|  |         BlCloseVolume(DiskHandle); | ||||||
|  |         return STATUS_EFI_SUCCESS; | ||||||
|  |     } | ||||||
|  |     else if(Status != STATUS_EFI_SUCCESS) | ||||||
|  |     { | ||||||
|  |         /* Failed to open directory */ | ||||||
|  |         BlDbgPrint(L"ERROR: Unable to open XTLDR directory (EFI/BOOT/XTLDR)\n"); | ||||||
|  |         BlCloseVolume(DiskHandle); | ||||||
|  |         return Status; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* Open EFI device path protocol */ | ||||||
|  |     Status = EfiSystemTable->BootServices->HandleProtocol(DiskHandle, &DevicePathGuid, (PVOID *)&DevicePath); | ||||||
|  |     if(Status != STATUS_EFI_SUCCESS) | ||||||
|  |     { | ||||||
|  |         /* Close volume */ | ||||||
|  |         BlCloseVolume(DiskHandle); | ||||||
|  |         return Status; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* Iterate through files inside XTLDR directory */ | ||||||
|  |     while(TRUE) | ||||||
|  |     { | ||||||
|  |         /* Read directory */ | ||||||
|  |         DirSize = sizeof(Buffer); | ||||||
|  |         Status = ModulesDir->Read(ModulesDir, &DirSize, Buffer); | ||||||
|  |         if(Status != STATUS_EFI_SUCCESS) | ||||||
|  |         { | ||||||
|  |             /* Failed to read directory */ | ||||||
|  |             BlDbgPrint(L"\n"); | ||||||
|  |  | ||||||
|  |             /* Close directory and volume */ | ||||||
|  |             ModulesDir->Close(ModulesDir); | ||||||
|  |             BlCloseVolume(DiskHandle); | ||||||
|  |             return Status; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         /* Check if read anything */ | ||||||
|  |         if(DirSize == 0) | ||||||
|  |         { | ||||||
|  |             /* Already read all contents, break loop execution */ | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         /* Take filename and its length */ | ||||||
|  |         ModuleName = ((PEFI_FILE_INFO)Buffer)->FileName; | ||||||
|  |         Length = RtlWideStringLength(ModuleName, 0); | ||||||
|  |  | ||||||
|  |         /* Make sure we deal with .EFI executable file */ | ||||||
|  |         if(Length < 4 || ModuleName[Length - 4] != '.' || | ||||||
|  |            (ModuleName[Length - 3] != 'E' && ModuleName[Length - 3] != 'e') || | ||||||
|  |            (ModuleName[Length - 2] != 'F' && ModuleName[Length - 2] != 'f') || | ||||||
|  |            (ModuleName[Length - 1] != 'I' && ModuleName[Length - 1] != 'i')) | ||||||
|  |         { | ||||||
|  |             /* Skip non .EFI file */ | ||||||
|  |             continue; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         /* Print debug message */ | ||||||
|  |         BlDbgPrint(L"Loading module '%S' ... ", ModuleName); | ||||||
|  |  | ||||||
|  |         /* Set correct path to the module file */ | ||||||
|  |         RtlWideStringConcatenate(ModulePath, ModulesDirPath, 0); | ||||||
|  |         RtlWideStringConcatenate(ModulePath, ModuleName, 0); | ||||||
|  |  | ||||||
|  |         /* Find valid device path */ | ||||||
|  |         Status = BlFindVolumeDevicePath(DevicePath, ModulePath, &VolumeDevicePath); | ||||||
|  |         if(Status != STATUS_EFI_SUCCESS) | ||||||
|  |         { | ||||||
|  |             /* Failed to set path */ | ||||||
|  |             BlDbgPrint(L"FAIL\n"); | ||||||
|  |             BlDbgPrint(L"ERROR: Unable to set valid device path\n"); | ||||||
|  |  | ||||||
|  |             /* Close directory and volume */ | ||||||
|  |             ModulesDir->Close(ModulesDir); | ||||||
|  |             BlCloseVolume(DiskHandle); | ||||||
|  |             return Status; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         /* Load the module into memory */ | ||||||
|  |         Status = EfiSystemTable->BootServices->LoadImage(FALSE, EfiImageHandle, VolumeDevicePath, NULL, 0, &ModuleHandle); | ||||||
|  |         if(Status != STATUS_EFI_SUCCESS) | ||||||
|  |         { | ||||||
|  |             /* Module failed */ | ||||||
|  |             BlDbgPrint(L"FAIL\n"); | ||||||
|  |  | ||||||
|  |             /* Check if caused by secure boot */ | ||||||
|  |             if(Status == STATUS_EFI_ACCESS_DENIED && EfiSecureBoot >= 1) | ||||||
|  |             { | ||||||
|  |                 BlDbgPrint(L"ERROR: SecureBoot signature validation failed\n"); | ||||||
|  |             } | ||||||
|  |             else | ||||||
|  |             { | ||||||
|  |                 BlDbgPrint(L"ERROR: Unable to load module\n"); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             /* Free memory and skip module */ | ||||||
|  |             BlEfiMemoryFreePool(VolumeDevicePath); | ||||||
|  |             continue; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         /* Free memory */ | ||||||
|  |         BlEfiMemoryFreePool(VolumeDevicePath); | ||||||
|  |  | ||||||
|  |         /* Access module interface for further module type check */ | ||||||
|  |         Status = EfiSystemTable->BootServices->OpenProtocol(ModuleHandle, &LIPGuid, (PVOID *)&LoadedImage, | ||||||
|  |                                                             EfiImageHandle, NULL, EFI_OPEN_PROTOCOL_GET_PROTOCOL); | ||||||
|  |         if(Status != STATUS_EFI_SUCCESS) | ||||||
|  |         { | ||||||
|  |             /* Failed to open protocol */ | ||||||
|  |             BlDbgPrint(L"FAIL\n"); | ||||||
|  |             BlDbgPrint(L"ERROR: Unable to access module interface\n"); | ||||||
|  |  | ||||||
|  |             /* Skip to the next module */ | ||||||
|  |             continue; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         /* Some firmwares do not allow to start drivers which are not of 'boot system driver' type, so check it */ | ||||||
|  |         if(LoadedImage->ImageCodeType != EfiBootServicesCode) | ||||||
|  |         { | ||||||
|  |             /* Different type set, probably 'runtime driver', refuse to load it */ | ||||||
|  |             BlDbgPrint(L"FAIL\n"); | ||||||
|  |             BlDbgPrint(L"ERROR: Loaded module is not a boot system driver\n"); | ||||||
|  |  | ||||||
|  |             /* Close protocol and skip module */ | ||||||
|  |             EfiSystemTable->BootServices->CloseProtocol(LoadedImage, &LIPGuid, LoadedImage, NULL); | ||||||
|  |             continue; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         /* Close loaded image protocol */ | ||||||
|  |         EfiSystemTable->BootServices->CloseProtocol(LoadedImage, &LIPGuid, LoadedImage, NULL); | ||||||
|  |  | ||||||
|  |         /* Start the module */ | ||||||
|  |         Status = EfiSystemTable->BootServices->StartImage(ModuleHandle, NULL, NULL); | ||||||
|  |         if(Status != STATUS_EFI_SUCCESS) | ||||||
|  |         { | ||||||
|  |             /* Module failed */ | ||||||
|  |             BlDbgPrint(L"FAIL\n"); | ||||||
|  |             BlDbgPrint(L"ERROR: Unable to start module\n"); | ||||||
|  |  | ||||||
|  |             /* Skip module */ | ||||||
|  |             continue; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         /* Module loaded successfully */ | ||||||
|  |         BlDbgPrint(L"OK\n"); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* Get list of all handles */ | ||||||
|  |     Status = EfiSystemTable->BootServices->LocateHandleBuffer(AllHandles, NULL, NULL, &ModulesCount, &Handles); | ||||||
|  |     if(Status != STATUS_EFI_SUCCESS) | ||||||
|  |     { | ||||||
|  |         /* Failed to get list of handles */ | ||||||
|  |         BlDbgPrint(L"WARNING: Unable to get a list of handles, some modules might not work properly\n"); | ||||||
|  |     } | ||||||
|  |     else | ||||||
|  |     { | ||||||
|  |         /* Iterate through a list of handles */ | ||||||
|  |         BlDbgPrint(L"Starting services for %lu handles\n", ModulesCount); | ||||||
|  |         for(Index = 0; Index < ModulesCount; Index++) | ||||||
|  |         { | ||||||
|  |             /* Start services for all loaded modules */ | ||||||
|  |             EfiSystemTable->BootServices->ConnectController(Handles[Index], NULL, NULL, TRUE); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /* Free memory */ | ||||||
|  |     BlEfiMemoryFreePool(Handles); | ||||||
|  |  | ||||||
|  |     /* Close directory and volume */ | ||||||
|  |     ModulesDir->Close(ModulesDir); | ||||||
|  |     BlCloseVolume(DiskHandle); | ||||||
|  |  | ||||||
|  |     /* Return success */ | ||||||
|     return STATUS_EFI_SUCCESS; |     return STATUS_EFI_SUCCESS; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -52,6 +262,7 @@ BlRegisterXtLoaderProtocol() | |||||||
|     LoaderProtocol.EfiPrint = BlEfiPrint; |     LoaderProtocol.EfiPrint = BlEfiPrint; | ||||||
|  |  | ||||||
|     /* Register loader protocol */ |     /* Register loader protocol */ | ||||||
|  |     BlDbgPrint(L"Registering XT loader protocol\n"); | ||||||
|     return EfiSystemTable->BootServices->InstallProtocolInterface(&Handle, &Guid, EFI_NATIVE_INTERFACE, &LoaderProtocol); |     return EfiSystemTable->BootServices->InstallProtocolInterface(&Handle, &Guid, EFI_NATIVE_INTERFACE, &LoaderProtocol); | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user