mirror of https://github.com/acidanthera/audk.git
340 lines
9.7 KiB
C
340 lines
9.7 KiB
C
|
/** @file
|
||
|
Platform VTd Sample driver.
|
||
|
|
||
|
Copyright (c) 2017, Intel Corporation. All rights reserved.<BR>
|
||
|
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 <PiDxe.h>
|
||
|
|
||
|
#include <IndustryStandard/Vtd.h>
|
||
|
#include <Protocol/PlatformVtdPolicy.h>
|
||
|
#include <Protocol/PciIo.h>
|
||
|
#include <Protocol/DevicePath.h>
|
||
|
|
||
|
#include <Library/IoLib.h>
|
||
|
#include <Library/BaseLib.h>
|
||
|
#include <Library/BaseMemoryLib.h>
|
||
|
#include <Library/MemoryAllocationLib.h>
|
||
|
#include <Library/DebugLib.h>
|
||
|
#include <Library/UefiBootServicesTableLib.h>
|
||
|
#include <Library/DevicePathLib.h>
|
||
|
|
||
|
typedef struct {
|
||
|
ACPI_EXTENDED_HID_DEVICE_PATH I2cController;
|
||
|
UINT8 HidStr[8];
|
||
|
UINT8 UidStr[1];
|
||
|
UINT8 CidStr[8];
|
||
|
} PLATFORM_I2C_CONTROLLER_DEVICE_PATH;
|
||
|
|
||
|
typedef struct {
|
||
|
ACPI_EXTENDED_HID_DEVICE_PATH I2cDevice;
|
||
|
UINT8 HidStr[13];
|
||
|
UINT8 UidStr[1];
|
||
|
UINT8 CidStr[13];
|
||
|
} PLATFORM_I2C_DEVICE_DEVICE_PATH;
|
||
|
|
||
|
typedef struct {
|
||
|
PLATFORM_I2C_CONTROLLER_DEVICE_PATH I2cController;
|
||
|
PLATFORM_I2C_DEVICE_DEVICE_PATH I2cDevice;
|
||
|
EFI_DEVICE_PATH_PROTOCOL End;
|
||
|
} PLATFORM_I2C_DEVICE_PATH;
|
||
|
|
||
|
typedef struct {
|
||
|
ACPI_HID_DEVICE_PATH PciRootBridge;
|
||
|
PCI_DEVICE_PATH PciDevice;
|
||
|
EFI_DEVICE_PATH_PROTOCOL EndDevicePath;
|
||
|
} PLATFORM_PCI_DEVICE_PATH;
|
||
|
|
||
|
typedef struct {
|
||
|
ACPI_HID_DEVICE_PATH PciRootBridge;
|
||
|
PCI_DEVICE_PATH PciBridge;
|
||
|
PCI_DEVICE_PATH PciDevice;
|
||
|
EFI_DEVICE_PATH_PROTOCOL EndDevicePath;
|
||
|
} PLATFORM_PCI_BRIDGE_DEVICE_PATH;
|
||
|
|
||
|
typedef struct {
|
||
|
EFI_DEVICE_PATH_PROTOCOL *DevicePath;
|
||
|
UINT16 Segment;
|
||
|
VTD_SOURCE_ID SourceId;
|
||
|
} PLATFORM_ACPI_DEVICE_MAPPING;
|
||
|
|
||
|
#define PLATFORM_PCI_ROOT_BRIDGE \
|
||
|
{ \
|
||
|
{ \
|
||
|
ACPI_DEVICE_PATH, \
|
||
|
ACPI_DP, \
|
||
|
{ \
|
||
|
(UINT8) (sizeof (ACPI_HID_DEVICE_PATH)), \
|
||
|
(UINT8) ((sizeof (ACPI_HID_DEVICE_PATH)) >> 8) \
|
||
|
}, \
|
||
|
}, \
|
||
|
EISA_PNP_ID (0x0A03), \
|
||
|
0 \
|
||
|
}
|
||
|
|
||
|
#define PLATFORM_END_ENTIRE \
|
||
|
{ \
|
||
|
END_DEVICE_PATH_TYPE, END_ENTIRE_DEVICE_PATH_SUBTYPE, { END_DEVICE_PATH_LENGTH, 0 } \
|
||
|
}
|
||
|
|
||
|
#define PLATFORM_PCI(Device, Function) \
|
||
|
{ \
|
||
|
{ \
|
||
|
HARDWARE_DEVICE_PATH, \
|
||
|
HW_PCI_DP, \
|
||
|
{ \
|
||
|
(UINT8) (sizeof (PCI_DEVICE_PATH)), \
|
||
|
(UINT8) ((sizeof (PCI_DEVICE_PATH)) >> 8) \
|
||
|
} \
|
||
|
}, \
|
||
|
(Function), \
|
||
|
(Device) \
|
||
|
}
|
||
|
|
||
|
#define PLATFORM_I2C(Hid, Uid, Cid, HidStr, UidStr, CidStr) \
|
||
|
{ \
|
||
|
{ \
|
||
|
{ \
|
||
|
ACPI_DEVICE_PATH, \
|
||
|
ACPI_EXTENDED_DP, \
|
||
|
{sizeof(ACPI_EXTENDED_HID_DEVICE_PATH) + sizeof(HidStr) + sizeof(UidStr) + sizeof(CidStr), 0} \
|
||
|
}, \
|
||
|
Hid, \
|
||
|
Uid, \
|
||
|
Cid \
|
||
|
}, \
|
||
|
HidStr, \
|
||
|
UidStr, \
|
||
|
CidStr \
|
||
|
}
|
||
|
|
||
|
PLATFORM_I2C_DEVICE_PATH mPlatformI2CDevicePath = {
|
||
|
PLATFORM_I2C(0, 2, 0, "INT33C3", "", "INT33C3"),
|
||
|
PLATFORM_I2C(0, 1, 0, "I2C01\\TPANEL", "", "I2C01\\TPANEL"),
|
||
|
PLATFORM_END_ENTIRE
|
||
|
};
|
||
|
|
||
|
PLATFORM_ACPI_DEVICE_MAPPING mAcpiDeviceMapping[] = {
|
||
|
{
|
||
|
(EFI_DEVICE_PATH_PROTOCOL *)&mPlatformI2CDevicePath,
|
||
|
0x0, // Segment
|
||
|
{{0x01, 0x15, 0x00}} // Function, Device, Bus
|
||
|
}
|
||
|
};
|
||
|
|
||
|
PLATFORM_PCI_BRIDGE_DEVICE_PATH mPlatformPciBridgeDevicePath = {
|
||
|
PLATFORM_PCI_ROOT_BRIDGE,
|
||
|
PLATFORM_PCI(0x1C, 1),
|
||
|
PLATFORM_PCI(0, 0),
|
||
|
PLATFORM_END_ENTIRE
|
||
|
};
|
||
|
|
||
|
EDKII_PLATFORM_VTD_DEVICE_INFO mExceptionDeviceList[] = {
|
||
|
{
|
||
|
0x0, // Segment
|
||
|
{{0x00, 0x00, 0x02}} // Function, Device, Bus
|
||
|
},
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
Compares 2 device path.
|
||
|
|
||
|
@param[in] DevicePath1 A device path with EndDevicePath node.
|
||
|
@param[in] DevicePath2 A device path with EndDevicePath node.
|
||
|
|
||
|
@retval TRUE 2 device path are identical.
|
||
|
@retval FALSE 2 device path are not identical.
|
||
|
**/
|
||
|
BOOLEAN
|
||
|
CompareDevicePath (
|
||
|
IN CONST EFI_DEVICE_PATH_PROTOCOL *DevicePath1,
|
||
|
IN CONST EFI_DEVICE_PATH_PROTOCOL *DevicePath2
|
||
|
)
|
||
|
{
|
||
|
UINTN Size1;
|
||
|
UINTN Size2;
|
||
|
|
||
|
Size1 = GetDevicePathSize (DevicePath1);
|
||
|
Size2 = GetDevicePathSize (DevicePath2);
|
||
|
if (Size1 != Size2) {
|
||
|
return FALSE;
|
||
|
}
|
||
|
if (CompareMem (DevicePath1, DevicePath2, Size1) != 0) {
|
||
|
return FALSE;
|
||
|
}
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
Get the VTD SourceId from the device handler.
|
||
|
This function is required for non PCI device handler.
|
||
|
|
||
|
Pseudo-algo in Intel VTd driver:
|
||
|
Status = PlatformGetVTdDeviceId ();
|
||
|
if (EFI_ERROR(Status)) {
|
||
|
if (DeviceHandle is PCI) {
|
||
|
Get SourceId from Bus/Device/Function
|
||
|
} else {
|
||
|
return EFI_UNSUPPORTED
|
||
|
}
|
||
|
}
|
||
|
Get VTd engine by Segment/Bus/Device/Function.
|
||
|
|
||
|
@param[in] This The protocol instance pointer.
|
||
|
@param[in] DeviceHandle Device Identifier in UEFI.
|
||
|
@param[out] DeviceInfo DeviceInfo for indentify the VTd engine in ACPI Table
|
||
|
and the VTd page entry.
|
||
|
|
||
|
@retval EFI_SUCCESS The VtdIndex and SourceId are returned.
|
||
|
@retval EFI_INVALID_PARAMETER DeviceHandle is not a valid handler.
|
||
|
@retval EFI_INVALID_PARAMETER DeviceInfo is NULL.
|
||
|
@retval EFI_NOT_FOUND The Segment or SourceId information is NOT found.
|
||
|
@retval EFI_UNSUPPORTED This function is not supported.
|
||
|
|
||
|
**/
|
||
|
EFI_STATUS
|
||
|
EFIAPI
|
||
|
PlatformVTdGetDeviceId (
|
||
|
IN EDKII_PLATFORM_VTD_POLICY_PROTOCOL *This,
|
||
|
IN EFI_HANDLE DeviceHandle,
|
||
|
OUT EDKII_PLATFORM_VTD_DEVICE_INFO *DeviceInfo
|
||
|
)
|
||
|
{
|
||
|
EFI_PCI_IO_PROTOCOL *PciIo;
|
||
|
UINTN Seg;
|
||
|
UINTN Bus;
|
||
|
UINTN Dev;
|
||
|
UINTN Func;
|
||
|
EFI_STATUS Status;
|
||
|
EFI_DEVICE_PATH_PROTOCOL *DevicePath;
|
||
|
UINTN Index;
|
||
|
|
||
|
DEBUG ((DEBUG_VERBOSE, "PlatformVTdGetDeviceId\n"));
|
||
|
|
||
|
if (DeviceInfo == NULL) {
|
||
|
return EFI_INVALID_PARAMETER;
|
||
|
}
|
||
|
|
||
|
if (DeviceHandle == NULL) {
|
||
|
return EFI_INVALID_PARAMETER;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Handle PCI device
|
||
|
//
|
||
|
Status = gBS->HandleProtocol (DeviceHandle, &gEfiPciIoProtocolGuid, (VOID **)&PciIo);
|
||
|
if (!EFI_ERROR(Status)) {
|
||
|
Status = PciIo->GetLocation (PciIo, &Seg, &Bus, &Dev, &Func);
|
||
|
if (EFI_ERROR(Status)) {
|
||
|
return EFI_UNSUPPORTED;
|
||
|
}
|
||
|
DeviceInfo->Segment = (UINT16)Seg;
|
||
|
DeviceInfo->SourceId.Bits.Bus = (UINT8)Bus;
|
||
|
DeviceInfo->SourceId.Bits.Device = (UINT8)Dev;
|
||
|
DeviceInfo->SourceId.Bits.Function = (UINT8)Func;
|
||
|
|
||
|
return EFI_SUCCESS;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Handle ACPI device
|
||
|
//
|
||
|
Status = gBS->HandleProtocol (DeviceHandle, &gEfiDevicePathProtocolGuid, (VOID **)&DevicePath);
|
||
|
if (!EFI_ERROR(Status)) {
|
||
|
for (Index = 0; Index < ARRAY_SIZE(mAcpiDeviceMapping); Index++) {
|
||
|
if (CompareDevicePath (mAcpiDeviceMapping[Index].DevicePath, DevicePath)) {
|
||
|
DeviceInfo->Segment = mAcpiDeviceMapping[Index].Segment;
|
||
|
DeviceInfo->SourceId = mAcpiDeviceMapping[Index].SourceId;
|
||
|
return EFI_SUCCESS;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return EFI_NOT_FOUND;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
Get a list of the exception devices.
|
||
|
|
||
|
The VTd driver should always set ALLOW for the device in this list.
|
||
|
|
||
|
@param[in] This The protocol instance pointer.
|
||
|
@param[out] DeviceInfoCount The count of the list of DeviceInfo.
|
||
|
@param[out] DeviceInfo A callee allocated buffer to hold a list of DeviceInfo.
|
||
|
|
||
|
@retval EFI_SUCCESS The DeviceInfoCount and DeviceInfo are returned.
|
||
|
@retval EFI_INVALID_PARAMETER DeviceInfoCount is NULL, or DeviceInfo is NULL.
|
||
|
@retval EFI_UNSUPPORTED This function is not supported.
|
||
|
|
||
|
**/
|
||
|
EFI_STATUS
|
||
|
EFIAPI
|
||
|
PlatformVTdGetExceptionDeviceList (
|
||
|
IN EDKII_PLATFORM_VTD_POLICY_PROTOCOL *This,
|
||
|
OUT UINTN *DeviceInfoCount,
|
||
|
OUT EDKII_PLATFORM_VTD_DEVICE_INFO **DeviceInfo
|
||
|
)
|
||
|
{
|
||
|
DEBUG ((DEBUG_VERBOSE, "PlatformVTdGetExceptionDeviceList\n"));
|
||
|
|
||
|
if (DeviceInfoCount == NULL || DeviceInfo == NULL) {
|
||
|
return EFI_INVALID_PARAMETER;
|
||
|
}
|
||
|
|
||
|
*DeviceInfo = AllocateZeroPool (sizeof(mExceptionDeviceList));
|
||
|
if (*DeviceInfo == NULL) {
|
||
|
return EFI_OUT_OF_RESOURCES;
|
||
|
}
|
||
|
CopyMem (*DeviceInfo, mExceptionDeviceList, sizeof(mExceptionDeviceList));
|
||
|
|
||
|
*DeviceInfoCount = ARRAY_SIZE(mExceptionDeviceList);
|
||
|
|
||
|
return EFI_SUCCESS;
|
||
|
}
|
||
|
|
||
|
EDKII_PLATFORM_VTD_POLICY_PROTOCOL mPlatformVTdSample = {
|
||
|
EDKII_PLATFORM_VTD_POLICY_PROTOCOL_REVISION,
|
||
|
PlatformVTdGetDeviceId,
|
||
|
PlatformVTdGetExceptionDeviceList,
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
Platform VTd sample driver.
|
||
|
|
||
|
@param[in] ImageHandle ImageHandle of the loaded driver
|
||
|
@param[in] SystemTable Pointer to the System Table
|
||
|
|
||
|
@retval EFI_SUCCESS The Protocol is installed.
|
||
|
@retval EFI_OUT_OF_RESOURCES Not enough resources available to initialize driver.
|
||
|
@retval EFI_DEVICE_ERROR A device error occurred attempting to initialize the driver.
|
||
|
|
||
|
**/
|
||
|
EFI_STATUS
|
||
|
EFIAPI
|
||
|
PlatformVTdSampleInitialize (
|
||
|
IN EFI_HANDLE ImageHandle,
|
||
|
IN EFI_SYSTEM_TABLE *SystemTable
|
||
|
)
|
||
|
{
|
||
|
EFI_STATUS Status;
|
||
|
EFI_HANDLE Handle;
|
||
|
|
||
|
Handle = NULL;
|
||
|
Status = gBS->InstallMultipleProtocolInterfaces (
|
||
|
&Handle,
|
||
|
&gEdkiiPlatformVTdPolicyProtocolGuid, &mPlatformVTdSample,
|
||
|
NULL
|
||
|
);
|
||
|
ASSERT_EFI_ERROR (Status);
|
||
|
|
||
|
return Status;
|
||
|
}
|