OpenCorePkg/Library/OcAppleDiskImageLib/OcAppleDiskImageBlockIo.c
2019-03-30 18:50:08 +01:00

337 lines
10 KiB
C

/** @file
Copyright (C) 2019, Goldfish64. 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 <Uefi.h>
#include <Protocol/BlockIo.h>
#include <Library/BaseMemoryLib.h>
#include <Library/DebugLib.h>
#include <Library/DevicePathLib.h>
#include <Library/MemoryAllocationLib.h>
#include <Library/OcAppleDiskImageLib.h>
#include <Library/OcStringLib.h>
#include <Library/PrintLib.h>
#include <Library/UefiBootServicesTableLib.h>
#include "OcAppleDiskImageLibInternal.h"
#define DMG_FILE_PATH_LEN (L_STR_LEN (L"DMG_.dmg") + 16 + 1)
typedef struct {
EFI_DEVICE_PATH_PROTOCOL Header;
///
/// A NULL-terminated Path string including directory and file names.
///
CHAR16 PathName[DMG_FILE_PATH_LEN];
} DMG_FILEPATH_DEVICE_PATH;
typedef struct {
DMG_CONTROLLER_DEVICE_PATH Controller;
MEMMAP_DEVICE_PATH MemMap;
DMG_FILEPATH_DEVICE_PATH FilePath;
DMG_SIZE_DEVICE_PATH Size;
EFI_DEVICE_PATH_PROTOCOL End;
} DMG_DEVICE_PATH;
//
// Mounted disk image private data.
//
#define OC_APPLE_DISK_IMAGE_MOUNTED_DATA_SIGNATURE SIGNATURE_32('D','m','g','I')
typedef struct {
// Signature.
UINTN Signature;
// Protocols.
EFI_BLOCK_IO_PROTOCOL BlockIo;
EFI_BLOCK_IO_MEDIA BlockIoMedia;
DMG_DEVICE_PATH DevicePath;
EFI_HANDLE Handle;
// Disk image data.
OC_APPLE_DISK_IMAGE_CONTEXT *ImageContext;
RAM_DMG_HEADER *RamDmgHeader;
} OC_APPLE_DISK_IMAGE_MOUNTED_DATA;
#define OC_APPLE_DISK_IMAGE_MOUNTED_DATA_FROM_THIS(This) \
CR(This, OC_APPLE_DISK_IMAGE_MOUNTED_DATA, BlockIo, OC_APPLE_DISK_IMAGE_MOUNTED_DATA_SIGNATURE)
//
// Block I/O protocol functions.
//
EFI_STATUS
EFIAPI
DiskImageBlockIoReset (
IN EFI_BLOCK_IO_PROTOCOL *This,
IN BOOLEAN ExtendedVerification
)
{
return EFI_SUCCESS;
}
EFI_STATUS
EFIAPI
DiskImageBlockIoReadBlocks (
IN EFI_BLOCK_IO_PROTOCOL *This,
IN UINT32 MediaId,
IN EFI_LBA Lba,
IN UINTN BufferSize,
OUT VOID *Buffer
)
{
OC_APPLE_DISK_IMAGE_MOUNTED_DATA *DiskImageData;
if ((This == NULL) || (Buffer == NULL)) {
return EFI_INVALID_PARAMETER;
}
if ((BufferSize % APPLE_DISK_IMAGE_SECTOR_SIZE) != 0) {
return EFI_BAD_BUFFER_SIZE;
}
DiskImageData = OC_APPLE_DISK_IMAGE_MOUNTED_DATA_FROM_THIS (This);
if (Lba >= DiskImageData->ImageContext->Trailer.SectorCount) {
return EFI_INVALID_PARAMETER;
}
//
// Read data from image.
//
return OcAppleDiskImageRead(DiskImageData->ImageContext, Lba, BufferSize, Buffer);
}
EFI_STATUS
EFIAPI
DiskImageBlockIoWriteBlocks (
IN EFI_BLOCK_IO_PROTOCOL *This,
IN UINT32 MediaId,
IN EFI_LBA Lba,
IN UINTN BufferSize,
IN VOID *Buffer
)
{
return EFI_WRITE_PROTECTED;
}
EFI_STATUS
EFIAPI
DiskImageBlockIoFlushBlocks (
IN EFI_BLOCK_IO_PROTOCOL *This
)
{
return EFI_SUCCESS;
}
//
// Block I/O protocol template.
//
EFI_BLOCK_IO_PROTOCOL mDiskImageBlockIo = {
EFI_BLOCK_IO_PROTOCOL_REVISION,
NULL,
DiskImageBlockIoReset,
DiskImageBlockIoReadBlocks,
DiskImageBlockIoWriteBlocks,
DiskImageBlockIoFlushBlocks
};
STATIC
VOID
InternalConstructDmgDevicePath (
IN OUT OC_APPLE_DISK_IMAGE_MOUNTED_DATA *DiskImageData,
IN EFI_PHYSICAL_ADDRESS RamDmgPhysAddr
)
{
UINT64 DmgSize;
DMG_DEVICE_PATH *DevicePath;
ASSERT (DiskImageData != NULL);
DmgSize = DiskImageData->ImageContext->Length;
DevicePath = &DiskImageData->DevicePath;
// Create DMG controller device node.
DevicePath->Controller.Header.Type = HARDWARE_DEVICE_PATH;
DevicePath->Controller.Header.SubType = HW_VENDOR_DP;
SetDevicePathNodeLength (&DevicePath->Controller, sizeof (DevicePath->Controller));
DevicePath->Controller.Guid = gDmgControllerDpGuid;
DevicePath->Controller.Key = 0;
// Allocate memmap node.
DevicePath->MemMap.Header.Type = HARDWARE_DEVICE_PATH;
DevicePath->MemMap.Header.SubType = HW_MEMMAP_DP;
SetDevicePathNodeLength (&DevicePath->MemMap, sizeof (DevicePath->MemMap));
// Set memmap node properties.
DevicePath->MemMap.MemoryType = EfiACPIMemoryNVS;
DevicePath->MemMap.StartingAddress = RamDmgPhysAddr;
DevicePath->MemMap.EndingAddress = DevicePath->MemMap.StartingAddress + sizeof(RAM_DMG_HEADER);
// Allocate filepath node. Length is struct length (includes null terminator) and name length.
DevicePath->FilePath.Header.Type = MEDIA_DEVICE_PATH;
DevicePath->FilePath.Header.Type = MEDIA_FILEPATH_DP;
SetDevicePathNodeLength (&DevicePath->FilePath, sizeof (DevicePath->FilePath));
UnicodeSPrint (DevicePath->FilePath.PathName, sizeof (DevicePath->FilePath.PathName), L"DMG_%16X.dmg", DmgSize);
// Allocate DMG size node.
DevicePath->Size.Header.Type = MESSAGING_DEVICE_PATH;
DevicePath->Size.Header.SubType = MSG_VENDOR_DP;
SetDevicePathNodeLength (&DevicePath->Size, sizeof (DevicePath->Size));
DevicePath->Size.Guid = gDmgSizeDpGuid;
DevicePath->Size.Length = DmgSize;
SetDevicePathEndNode (&DevicePath->End);
}
EFI_STATUS
EFIAPI
OcAppleDiskImageInstallBlockIo(
IN OC_APPLE_DISK_IMAGE_CONTEXT *Context) {
// Create variables.
EFI_STATUS Status;
OC_APPLE_DISK_IMAGE_MOUNTED_DATA *DiskImageData = NULL;
// RAM DMG.
EFI_PHYSICAL_ADDRESS RamDmgPhysAddr;
RAM_DMG_HEADER *RamDmgHeader = NULL;
ASSERT (Context != NULL);
ASSERT (Context->BlockIoHandle != NULL);
// Allocate page for RAM DMG header used by boot.efi.
Status = gBS->AllocatePages(AllocateAnyPages, EfiACPIMemoryNVS, EFI_SIZE_TO_PAGES(sizeof(RAM_DMG_HEADER)), &RamDmgPhysAddr);
if (EFI_ERROR(Status)) {
return EFI_OUT_OF_RESOURCES;
}
RamDmgHeader = (RAM_DMG_HEADER*)RamDmgPhysAddr;
ZeroMem(RamDmgHeader, sizeof(RAM_DMG_HEADER));
// Fill RAM DMG header.
RamDmgHeader->Signature = RamDmgHeader->Signature2 = RAM_DMG_SIGNATURE;
RamDmgHeader->Version = RAM_DMG_VERSION;
RamDmgHeader->ExtentCount = 1;
RamDmgHeader->ExtentInfo[0].Start = (UINT64)Context->Buffer;
RamDmgHeader->ExtentInfo[0].Length = Context->Length;
DEBUG((DEBUG_INFO, "DMG extent @ 0x%lx, length 0x%lx\n", RamDmgHeader->ExtentInfo[0].Start, RamDmgHeader->ExtentInfo[0].Length));
// Allocate installed DMG info.
DiskImageData = AllocateZeroPool(sizeof(OC_APPLE_DISK_IMAGE_MOUNTED_DATA));
if (!DiskImageData) {
gBS->FreePages(RamDmgPhysAddr, EFI_SIZE_TO_PAGES(sizeof(RAM_DMG_HEADER)));
return EFI_OUT_OF_RESOURCES;
}
// Fill disk image data.
DiskImageData->Signature = OC_APPLE_DISK_IMAGE_MOUNTED_DATA_SIGNATURE;
CopyMem(&(DiskImageData->BlockIo), &mDiskImageBlockIo, sizeof(EFI_BLOCK_IO_PROTOCOL));
DiskImageData->Handle = NULL;
DiskImageData->ImageContext = Context;
DiskImageData->RamDmgHeader = RamDmgHeader;
// Allocate media info.
DiskImageData->BlockIo.Media = &DiskImageData->BlockIoMedia;
// Fill media info.
DiskImageData->BlockIoMedia.MediaPresent = TRUE;
DiskImageData->BlockIoMedia.ReadOnly = TRUE;
DiskImageData->BlockIoMedia.BlockSize = APPLE_DISK_IMAGE_SECTOR_SIZE;
DiskImageData->BlockIoMedia.LastBlock = Context->Trailer.SectorCount - 1;
InternalConstructDmgDevicePath (DiskImageData, RamDmgPhysAddr);
// Install protocols on child.
Status = gBS->InstallMultipleProtocolInterfaces(&(DiskImageData->Handle),
&gEfiBlockIoProtocolGuid, &DiskImageData->BlockIo,
&gEfiDevicePathProtocolGuid, &DiskImageData->DevicePath, NULL);
if (EFI_ERROR(Status)) {
FreePool(DiskImageData);
gBS->FreePages(RamDmgPhysAddr, EFI_SIZE_TO_PAGES(sizeof(RAM_DMG_HEADER)));
return Status;
}
// Connect controller.
Status = gBS->ConnectController(DiskImageData->Handle, NULL, NULL, TRUE);
if (EFI_ERROR(Status)) {
gBS->UninstallMultipleProtocolInterfaces (DiskImageData->Handle,
&gEfiBlockIoProtocolGuid, &(DiskImageData->BlockIo),
&gEfiDevicePathProtocolGuid, &DiskImageData->DevicePath, NULL);
FreePool(DiskImageData);
gBS->FreePages(RamDmgPhysAddr, EFI_SIZE_TO_PAGES(sizeof(RAM_DMG_HEADER)));
return Status;
}
// Success.
Context->BlockIoHandle = DiskImageData->Handle;
return EFI_SUCCESS;
}
EFI_STATUS
EFIAPI
OcAppleDiskImageUninstallBlockIo (
IN OC_APPLE_DISK_IMAGE_CONTEXT *Context
)
{
EFI_STATUS Status;
EFI_BLOCK_IO_PROTOCOL *BlockIo;
OC_APPLE_DISK_IMAGE_MOUNTED_DATA *DiskImageData;
ASSERT (Context != NULL);
if (Context->BlockIoHandle == NULL)
return EFI_NOT_STARTED;
//
// Get block I/O.
//
Status = gBS->HandleProtocol (Context->BlockIoHandle, &gEfiBlockIoProtocolGuid, (VOID**)&BlockIo);
if (EFI_ERROR (Status))
return Status;
//
// Get disk image data.
//
DiskImageData = OC_APPLE_DISK_IMAGE_MOUNTED_DATA_FROM_THIS (BlockIo);
//
// Ensure context matches.
//
ASSERT (DiskImageData->Handle == Context->BlockIoHandle && DiskImageData->ImageContext == Context);
//
// Disconnect controller.
//
Status = gBS->DisconnectController (DiskImageData->Handle, NULL, NULL);
if (EFI_ERROR (Status))
return Status;
//
// Uninstall protocols.
//
Status = gBS->UninstallMultipleProtocolInterfaces (DiskImageData->Handle,
&gEfiBlockIoProtocolGuid, &(DiskImageData->BlockIo),
&gEfiDevicePathProtocolGuid, &DiskImageData->DevicePath, NULL);
if (EFI_ERROR (Status))
return Status;
Context->BlockIoHandle = NULL;
// Free RAM DMG header pages.
if (DiskImageData->RamDmgHeader != NULL)
FreePages (DiskImageData->RamDmgHeader, EFI_SIZE_TO_PAGES (sizeof(RAM_DMG_HEADER)));
//
// Free disk image data.
//
FreePool (DiskImageData);
return EFI_SUCCESS;
}