2016-10-31 15:58:18 +01:00
|
|
|
/** @file
|
|
|
|
Copyright (c) 2016, Linaro, Ltd. All rights reserved.<BR>
|
|
|
|
|
2019-04-04 01:05:13 +02:00
|
|
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
2016-10-31 15:58:18 +01:00
|
|
|
|
|
|
|
**/
|
|
|
|
|
|
|
|
#include <PiDxe.h>
|
|
|
|
|
|
|
|
#include <Guid/NonDiscoverableDevice.h>
|
|
|
|
|
|
|
|
#include <Library/BaseMemoryLib.h>
|
|
|
|
#include <Library/DebugLib.h>
|
|
|
|
#include <Library/DevicePathLib.h>
|
|
|
|
#include <Library/MemoryAllocationLib.h>
|
|
|
|
#include <Library/NonDiscoverableDeviceRegistrationLib.h>
|
|
|
|
#include <Library/UefiBootServicesTableLib.h>
|
|
|
|
|
|
|
|
#include <Protocol/DevicePath.h>
|
|
|
|
#include <Protocol/NonDiscoverableDevice.h>
|
|
|
|
|
2016-12-19 08:02:24 +01:00
|
|
|
/**
|
|
|
|
Get Guid form the type of non-discoverable device.
|
|
|
|
|
|
|
|
@param[in] Type The type of non-discoverable device.
|
|
|
|
|
|
|
|
@retval Return the Guid.
|
|
|
|
|
|
|
|
**/
|
2016-10-31 15:58:18 +01:00
|
|
|
STATIC
|
|
|
|
CONST EFI_GUID *
|
|
|
|
GetGuidFromType (
|
|
|
|
IN NON_DISCOVERABLE_DEVICE_TYPE Type
|
|
|
|
)
|
|
|
|
{
|
|
|
|
switch (Type) {
|
2021-12-05 23:54:02 +01:00
|
|
|
case NonDiscoverableDeviceTypeAhci:
|
|
|
|
return &gEdkiiNonDiscoverableAhciDeviceGuid;
|
2016-10-31 15:58:18 +01:00
|
|
|
|
2021-12-05 23:54:02 +01:00
|
|
|
case NonDiscoverableDeviceTypeAmba:
|
|
|
|
return &gEdkiiNonDiscoverableAmbaDeviceGuid;
|
2016-10-31 15:58:18 +01:00
|
|
|
|
2021-12-05 23:54:02 +01:00
|
|
|
case NonDiscoverableDeviceTypeEhci:
|
|
|
|
return &gEdkiiNonDiscoverableEhciDeviceGuid;
|
2016-10-31 15:58:18 +01:00
|
|
|
|
2021-12-05 23:54:02 +01:00
|
|
|
case NonDiscoverableDeviceTypeNvme:
|
|
|
|
return &gEdkiiNonDiscoverableNvmeDeviceGuid;
|
2016-10-31 15:58:18 +01:00
|
|
|
|
2021-12-05 23:54:02 +01:00
|
|
|
case NonDiscoverableDeviceTypeOhci:
|
|
|
|
return &gEdkiiNonDiscoverableOhciDeviceGuid;
|
2016-10-31 15:58:18 +01:00
|
|
|
|
2021-12-05 23:54:02 +01:00
|
|
|
case NonDiscoverableDeviceTypeSdhci:
|
|
|
|
return &gEdkiiNonDiscoverableSdhciDeviceGuid;
|
2016-10-31 15:58:18 +01:00
|
|
|
|
2021-12-05 23:54:02 +01:00
|
|
|
case NonDiscoverableDeviceTypeUfs:
|
|
|
|
return &gEdkiiNonDiscoverableUfsDeviceGuid;
|
2016-10-31 15:58:18 +01:00
|
|
|
|
2021-12-05 23:54:02 +01:00
|
|
|
case NonDiscoverableDeviceTypeUhci:
|
|
|
|
return &gEdkiiNonDiscoverableUhciDeviceGuid;
|
2016-10-31 15:58:18 +01:00
|
|
|
|
2021-12-05 23:54:02 +01:00
|
|
|
case NonDiscoverableDeviceTypeXhci:
|
|
|
|
return &gEdkiiNonDiscoverableXhciDeviceGuid;
|
2016-10-31 15:58:18 +01:00
|
|
|
|
2021-12-05 23:54:02 +01:00
|
|
|
default:
|
|
|
|
return NULL;
|
2016-10-31 15:58:18 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#pragma pack (1)
|
|
|
|
typedef struct {
|
2021-12-05 23:54:02 +01:00
|
|
|
VENDOR_DEVICE_PATH Vendor;
|
|
|
|
UINT64 BaseAddress;
|
|
|
|
UINT8 ResourceType;
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL End;
|
2016-10-31 15:58:18 +01:00
|
|
|
} NON_DISCOVERABLE_DEVICE_PATH;
|
|
|
|
#pragma pack ()
|
|
|
|
|
|
|
|
/**
|
2016-12-19 08:02:24 +01:00
|
|
|
Register a non-discoverable MMIO device.
|
2016-10-31 15:58:18 +01:00
|
|
|
|
2016-12-19 08:02:24 +01:00
|
|
|
@param[in] Type The type of non-discoverable device
|
2016-10-31 15:58:18 +01:00
|
|
|
@param[in] DmaType Whether the device is DMA coherent
|
|
|
|
@param[in] InitFunc Initialization routine to be invoked when
|
|
|
|
the device is enabled
|
|
|
|
@param[in,out] Handle The handle onto which to install the
|
|
|
|
non-discoverable device protocol.
|
|
|
|
If Handle is NULL or *Handle is NULL, a
|
|
|
|
new handle will be allocated.
|
|
|
|
@param[in] NumMmioResources The number of UINTN base/size pairs that
|
|
|
|
follow, each describing an MMIO region
|
|
|
|
owned by the device
|
2016-12-19 08:02:24 +01:00
|
|
|
@param[in] ... The variable argument list which contains the
|
|
|
|
info about MmioResources.
|
2016-10-31 15:58:18 +01:00
|
|
|
|
|
|
|
@retval EFI_SUCCESS The registration succeeded.
|
|
|
|
@retval EFI_INVALID_PARAMETER An invalid argument was given
|
|
|
|
@retval Other The registration failed.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
RegisterNonDiscoverableMmioDevice (
|
|
|
|
IN NON_DISCOVERABLE_DEVICE_TYPE Type,
|
|
|
|
IN NON_DISCOVERABLE_DEVICE_DMA_TYPE DmaType,
|
|
|
|
IN NON_DISCOVERABLE_DEVICE_INIT InitFunc,
|
|
|
|
IN OUT EFI_HANDLE *Handle OPTIONAL,
|
|
|
|
IN UINTN NumMmioResources,
|
|
|
|
...
|
|
|
|
)
|
|
|
|
{
|
2021-12-05 23:54:02 +01:00
|
|
|
NON_DISCOVERABLE_DEVICE *Device;
|
|
|
|
NON_DISCOVERABLE_DEVICE_PATH *DevicePath;
|
|
|
|
EFI_HANDLE LocalHandle;
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINTN AllocSize;
|
|
|
|
UINTN Index;
|
|
|
|
VA_LIST Args;
|
|
|
|
EFI_ACPI_ADDRESS_SPACE_DESCRIPTOR *Desc;
|
|
|
|
EFI_ACPI_END_TAG_DESCRIPTOR *End;
|
|
|
|
UINTN Base, Size;
|
|
|
|
|
|
|
|
if ((Type >= NonDiscoverableDeviceTypeMax) ||
|
|
|
|
(DmaType >= NonDiscoverableDeviceDmaTypeMax) ||
|
|
|
|
(NumMmioResources == 0))
|
|
|
|
{
|
2016-10-31 15:58:18 +01:00
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Handle == NULL) {
|
2021-12-05 23:54:02 +01:00
|
|
|
Handle = &LocalHandle;
|
2016-10-31 15:58:18 +01:00
|
|
|
LocalHandle = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
AllocSize = sizeof *Device +
|
|
|
|
NumMmioResources * sizeof (EFI_ACPI_ADDRESS_SPACE_DESCRIPTOR) +
|
|
|
|
sizeof (EFI_ACPI_END_TAG_DESCRIPTOR);
|
|
|
|
Device = (NON_DISCOVERABLE_DEVICE *)AllocateZeroPool (AllocSize);
|
|
|
|
if (Device == NULL) {
|
|
|
|
return EFI_OUT_OF_RESOURCES;
|
|
|
|
}
|
|
|
|
|
|
|
|
Device->Type = GetGuidFromType (Type);
|
|
|
|
ASSERT (Device->Type != NULL);
|
|
|
|
|
2021-12-05 23:54:02 +01:00
|
|
|
Device->DmaType = DmaType;
|
2016-10-31 15:58:18 +01:00
|
|
|
Device->Initialize = InitFunc;
|
2021-12-05 23:54:02 +01:00
|
|
|
Device->Resources = (EFI_ACPI_ADDRESS_SPACE_DESCRIPTOR *)(Device + 1);
|
2016-10-31 15:58:18 +01:00
|
|
|
|
|
|
|
VA_START (Args, NumMmioResources);
|
|
|
|
for (Index = 0; Index < NumMmioResources; Index++) {
|
2021-12-05 23:54:02 +01:00
|
|
|
Desc = &Device->Resources[Index];
|
2016-10-31 15:58:18 +01:00
|
|
|
Base = VA_ARG (Args, UINTN);
|
|
|
|
Size = VA_ARG (Args, UINTN);
|
|
|
|
|
|
|
|
Desc->Desc = ACPI_ADDRESS_SPACE_DESCRIPTOR;
|
|
|
|
Desc->Len = sizeof *Desc - 3;
|
|
|
|
Desc->AddrRangeMin = Base;
|
|
|
|
Desc->AddrLen = Size;
|
|
|
|
Desc->AddrRangeMax = Base + Size - 1;
|
|
|
|
Desc->ResType = ACPI_ADDRESS_SPACE_TYPE_MEM;
|
2017-01-11 02:54:39 +01:00
|
|
|
Desc->AddrSpaceGranularity = ((EFI_PHYSICAL_ADDRESS)Base + Size > SIZE_4GB) ? 64 : 32;
|
2016-10-31 15:58:18 +01:00
|
|
|
Desc->AddrTranslationOffset = 0;
|
|
|
|
}
|
2021-12-05 23:54:02 +01:00
|
|
|
|
2016-10-31 15:58:18 +01:00
|
|
|
VA_END (Args);
|
|
|
|
|
2021-12-05 23:54:02 +01:00
|
|
|
End = (EFI_ACPI_END_TAG_DESCRIPTOR *)&Device->Resources[NumMmioResources];
|
2016-10-31 15:58:18 +01:00
|
|
|
|
|
|
|
End->Desc = ACPI_END_TAG_DESCRIPTOR;
|
|
|
|
End->Checksum = 0;
|
|
|
|
|
|
|
|
DevicePath = (NON_DISCOVERABLE_DEVICE_PATH *)CreateDeviceNode (
|
|
|
|
HARDWARE_DEVICE_PATH,
|
|
|
|
HW_VENDOR_DP,
|
2021-12-05 23:54:02 +01:00
|
|
|
sizeof (*DevicePath)
|
|
|
|
);
|
2016-10-31 15:58:18 +01:00
|
|
|
if (DevicePath == NULL) {
|
|
|
|
Status = EFI_OUT_OF_RESOURCES;
|
|
|
|
goto FreeDevice;
|
|
|
|
}
|
|
|
|
|
|
|
|
CopyGuid (&DevicePath->Vendor.Guid, &gEdkiiNonDiscoverableDeviceProtocolGuid);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Use the base address and type of the first region to
|
|
|
|
// make the device path unique
|
|
|
|
//
|
2021-12-05 23:54:02 +01:00
|
|
|
DevicePath->BaseAddress = Device->Resources[0].AddrRangeMin;
|
|
|
|
DevicePath->ResourceType = Device->Resources[0].ResType;
|
2016-10-31 15:58:18 +01:00
|
|
|
|
2021-12-05 23:54:02 +01:00
|
|
|
SetDevicePathNodeLength (
|
|
|
|
&DevicePath->Vendor,
|
|
|
|
sizeof (*DevicePath) - sizeof (DevicePath->End)
|
|
|
|
);
|
2016-10-31 15:58:18 +01:00
|
|
|
SetDevicePathEndNode (&DevicePath->End);
|
|
|
|
|
2021-12-05 23:54:02 +01:00
|
|
|
Status = gBS->InstallMultipleProtocolInterfaces (
|
|
|
|
Handle,
|
|
|
|
&gEdkiiNonDiscoverableDeviceProtocolGuid,
|
|
|
|
Device,
|
|
|
|
&gEfiDevicePathProtocolGuid,
|
|
|
|
DevicePath,
|
|
|
|
NULL
|
|
|
|
);
|
2016-10-31 15:58:18 +01:00
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
goto FreeDevicePath;
|
|
|
|
}
|
2021-12-05 23:54:02 +01:00
|
|
|
|
2016-10-31 15:58:18 +01:00
|
|
|
return EFI_SUCCESS;
|
|
|
|
|
|
|
|
FreeDevicePath:
|
|
|
|
FreePool (DevicePath);
|
|
|
|
|
|
|
|
FreeDevice:
|
|
|
|
FreePool (Device);
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|