mirror of
https://github.com/acidanthera/OpenCorePkg.git
synced 2025-12-08 19:25:01 +00:00
194 lines
4.9 KiB
C
194 lines
4.9 KiB
C
/** @file
|
|
Copyright (C) 2020, vit9696. All rights reserved.
|
|
|
|
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 "OcApfsInternal.h"
|
|
#include <Library/OcApfsLib.h>
|
|
#include <Library/BaseMemoryLib.h>
|
|
#include <Library/DebugLib.h>
|
|
#include <Library/DevicePathLib.h>
|
|
#include <Library/MemoryAllocationLib.h>
|
|
#include <Library/UefiBootServicesTableLib.h>
|
|
#include <Library/UefiLib.h>
|
|
#include <Protocol/BlockIo.h>
|
|
#include <Protocol/PartitionInfo.h>
|
|
|
|
STATIC VOID *mApfsNewPartitionsEventKey;
|
|
|
|
STATIC
|
|
VOID
|
|
EFIAPI
|
|
ApfsNewPartitionArrived (
|
|
IN EFI_EVENT Event,
|
|
IN VOID *Context
|
|
)
|
|
{
|
|
EFI_STATUS Status;
|
|
UINTN BufferSize;
|
|
EFI_HANDLE Handle;
|
|
|
|
while (TRUE) {
|
|
BufferSize = sizeof (EFI_HANDLE);
|
|
Status = gBS->LocateHandle (
|
|
ByRegisterNotify,
|
|
NULL,
|
|
mApfsNewPartitionsEventKey,
|
|
&BufferSize,
|
|
&Handle
|
|
);
|
|
if (!EFI_ERROR (Status)) {
|
|
OcApfsConnectHandle (Handle, TRUE);
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
STATIC
|
|
EFI_STATUS
|
|
ApfsMonitorNewPartitions (
|
|
VOID
|
|
)
|
|
{
|
|
EFI_STATUS Status;
|
|
EFI_EVENT Event;
|
|
|
|
Status = gBS->CreateEvent (
|
|
EVT_NOTIFY_SIGNAL,
|
|
TPL_CALLBACK,
|
|
ApfsNewPartitionArrived,
|
|
NULL,
|
|
&Event
|
|
);
|
|
|
|
if (!EFI_ERROR (Status)) {
|
|
Status = gBS->RegisterProtocolNotify (
|
|
&gEfiBlockIoProtocolGuid,
|
|
Event,
|
|
&mApfsNewPartitionsEventKey
|
|
);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
gBS->CloseEvent (Event);
|
|
}
|
|
}
|
|
|
|
return Status;
|
|
}
|
|
|
|
EFI_STATUS
|
|
OcApfsConnectParentDevice (
|
|
IN EFI_HANDLE Handle OPTIONAL,
|
|
IN BOOLEAN VerifyPolicy
|
|
)
|
|
{
|
|
EFI_STATUS Status;
|
|
EFI_STATUS Status2;
|
|
UINTN HandleCount;
|
|
EFI_HANDLE *HandleBuffer;
|
|
EFI_DEVICE_PATH *ParentDevicePath;
|
|
EFI_DEVICE_PATH *ChildDevicePath;
|
|
UINTN Index;
|
|
UINTN PrefixLength;
|
|
|
|
HandleCount = 0;
|
|
Status = gBS->LocateHandleBuffer (
|
|
ByProtocol,
|
|
&gEfiBlockIoProtocolGuid,
|
|
NULL,
|
|
&HandleCount,
|
|
&HandleBuffer
|
|
);
|
|
|
|
ParentDevicePath = NULL;
|
|
PrefixLength = 0;
|
|
if (Handle != NULL) {
|
|
Status2 = gBS->HandleProtocol (
|
|
Handle,
|
|
&gEfiDevicePathProtocolGuid,
|
|
(VOID **)&ParentDevicePath
|
|
);
|
|
if (!EFI_ERROR (Status2)) {
|
|
PrefixLength = GetDevicePathSize (ParentDevicePath) - END_DEVICE_PATH_LENGTH;
|
|
} else {
|
|
DEBUG ((DEBUG_INFO, "OCJS: No parent device path - %r\n", Status2));
|
|
ParentDevicePath = NULL;
|
|
}
|
|
}
|
|
|
|
if (!EFI_ERROR (Status)) {
|
|
Status = EFI_NOT_FOUND;
|
|
|
|
for (Index = 0; Index < HandleCount; ++Index) {
|
|
if ((ParentDevicePath != NULL) && (PrefixLength > 0)) {
|
|
Status2 = gBS->HandleProtocol (
|
|
HandleBuffer[Index],
|
|
&gEfiDevicePathProtocolGuid,
|
|
(VOID **)&ChildDevicePath
|
|
);
|
|
if (EFI_ERROR (Status2)) {
|
|
DEBUG ((DEBUG_INFO, "OCJS: No child device path - %r\n", Status2));
|
|
continue;
|
|
}
|
|
|
|
if (CompareMem (ParentDevicePath, ChildDevicePath, PrefixLength) == 0) {
|
|
DEBUG ((DEBUG_INFO, "OCJS: Matched device path\n"));
|
|
} else {
|
|
continue;
|
|
}
|
|
}
|
|
|
|
Status2 = OcApfsConnectHandle (
|
|
HandleBuffer[Index],
|
|
VerifyPolicy
|
|
);
|
|
if (!EFI_ERROR (Status2)) {
|
|
Status = Status2;
|
|
}
|
|
}
|
|
|
|
FreePool (HandleBuffer);
|
|
} else {
|
|
DEBUG ((DEBUG_INFO, "OCJS: BlockIo buffer error - %r\n", Status));
|
|
}
|
|
|
|
return Status;
|
|
}
|
|
|
|
EFI_STATUS
|
|
OcApfsConnectDevices (
|
|
IN BOOLEAN Monitor
|
|
)
|
|
{
|
|
EFI_STATUS Status;
|
|
VOID *PartitionInfoInterface;
|
|
|
|
DEBUG_CODE_BEGIN ();
|
|
Status = gBS->LocateProtocol (
|
|
&gEfiPartitionInfoProtocolGuid,
|
|
NULL,
|
|
&PartitionInfoInterface
|
|
);
|
|
DEBUG ((DEBUG_INFO, "OCJS: PartitionInfo is %r\n", Status));
|
|
DEBUG_CODE_END ();
|
|
|
|
if (Monitor) {
|
|
Status = ApfsMonitorNewPartitions ();
|
|
if (EFI_ERROR (Status)) {
|
|
DEBUG ((DEBUG_INFO, "OCJS: Failed to setup drive monitoring - %r\n", Status));
|
|
}
|
|
}
|
|
|
|
return OcApfsConnectParentDevice (NULL, TRUE);
|
|
}
|