2019-01-29 09:43:41 +01:00
|
|
|
/** @file
|
|
|
|
The device path help function.
|
|
|
|
|
|
|
|
Copyright (c) 2019, Intel Corporation. All rights reserved.<BR>
|
|
|
|
|
2019-04-04 01:05:13 +02:00
|
|
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
2019-01-29 09:43:41 +01:00
|
|
|
|
|
|
|
**/
|
|
|
|
|
|
|
|
#include "NvmExpressPei.h"
|
|
|
|
|
|
|
|
//
|
|
|
|
// Template for an Nvm Express Device Path node
|
|
|
|
//
|
|
|
|
NVME_NAMESPACE_DEVICE_PATH mNvmeDevicePathNodeTemplate = {
|
|
|
|
{ // Header
|
|
|
|
MESSAGING_DEVICE_PATH,
|
|
|
|
MSG_NVME_NAMESPACE_DP,
|
|
|
|
{
|
2021-12-05 23:54:02 +01:00
|
|
|
(UINT8)(sizeof (NVME_NAMESPACE_DEVICE_PATH)),
|
|
|
|
(UINT8)((sizeof (NVME_NAMESPACE_DEVICE_PATH)) >> 8)
|
2019-01-29 09:43:41 +01:00
|
|
|
}
|
|
|
|
},
|
|
|
|
0x0, // NamespaceId
|
|
|
|
0x0 // NamespaceUuid
|
|
|
|
};
|
|
|
|
|
|
|
|
//
|
|
|
|
// Template for an End of entire Device Path node
|
|
|
|
//
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL mNvmeEndDevicePathNodeTemplate = {
|
|
|
|
END_DEVICE_PATH_TYPE,
|
|
|
|
END_ENTIRE_DEVICE_PATH_SUBTYPE,
|
|
|
|
{
|
2021-12-05 23:54:02 +01:00
|
|
|
(UINT8)(sizeof (EFI_DEVICE_PATH_PROTOCOL)),
|
|
|
|
(UINT8)((sizeof (EFI_DEVICE_PATH_PROTOCOL)) >> 8)
|
2019-01-29 09:43:41 +01:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
Returns the 16-bit Length field of a device path node.
|
|
|
|
|
|
|
|
Returns the 16-bit Length field of the device path node specified by Node.
|
|
|
|
Node is not required to be aligned on a 16-bit boundary, so it is recommended
|
|
|
|
that a function such as ReadUnaligned16() be used to extract the contents of
|
|
|
|
the Length field.
|
|
|
|
|
|
|
|
If Node is NULL, then ASSERT().
|
|
|
|
|
|
|
|
@param Node A pointer to a device path node data structure.
|
|
|
|
|
|
|
|
@return The 16-bit Length field of the device path node specified by Node.
|
|
|
|
|
|
|
|
**/
|
|
|
|
UINTN
|
|
|
|
DevicePathNodeLength (
|
|
|
|
IN CONST VOID *Node
|
|
|
|
)
|
|
|
|
{
|
|
|
|
ASSERT (Node != NULL);
|
|
|
|
return ReadUnaligned16 ((UINT16 *)&((EFI_DEVICE_PATH_PROTOCOL *)(Node))->Length[0]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Returns a pointer to the next node in a device path.
|
|
|
|
|
|
|
|
If Node is NULL, then ASSERT().
|
|
|
|
|
|
|
|
@param Node A pointer to a device path node data structure.
|
|
|
|
|
|
|
|
@return a pointer to the device path node that follows the device path node
|
|
|
|
specified by Node.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL *
|
|
|
|
NextDevicePathNode (
|
|
|
|
IN CONST VOID *Node
|
|
|
|
)
|
|
|
|
{
|
|
|
|
ASSERT (Node != NULL);
|
2021-12-05 23:54:02 +01:00
|
|
|
return (EFI_DEVICE_PATH_PROTOCOL *)((UINT8 *)(Node) + DevicePathNodeLength (Node));
|
2019-01-29 09:43:41 +01:00
|
|
|
}
|
|
|
|
|
2019-01-21 07:14:19 +01:00
|
|
|
/**
|
|
|
|
Get the size of the current device path instance.
|
|
|
|
|
|
|
|
@param[in] DevicePath A pointer to the EFI_DEVICE_PATH_PROTOCOL
|
|
|
|
structure.
|
|
|
|
@param[out] InstanceSize The size of the current device path instance.
|
|
|
|
@param[out] EntireDevicePathEnd Indicate whether the instance is the last
|
|
|
|
one in the device path strucure.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS The size of the current device path instance is fetched.
|
|
|
|
@retval Others Fails to get the size of the current device path instance.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
GetDevicePathInstanceSize (
|
2021-12-05 23:54:02 +01:00
|
|
|
IN EFI_DEVICE_PATH_PROTOCOL *DevicePath,
|
|
|
|
OUT UINTN *InstanceSize,
|
|
|
|
OUT BOOLEAN *EntireDevicePathEnd
|
2019-01-21 07:14:19 +01:00
|
|
|
)
|
|
|
|
{
|
2021-12-05 23:54:02 +01:00
|
|
|
EFI_DEVICE_PATH_PROTOCOL *Walker;
|
2019-01-21 07:14:19 +01:00
|
|
|
|
2021-12-05 23:54:02 +01:00
|
|
|
if ((DevicePath == NULL) || (InstanceSize == NULL) || (EntireDevicePathEnd == NULL)) {
|
2019-01-21 07:14:19 +01:00
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Find the end of the device path instance
|
|
|
|
//
|
|
|
|
Walker = DevicePath;
|
|
|
|
while (Walker->Type != END_DEVICE_PATH_TYPE) {
|
|
|
|
Walker = NextDevicePathNode (Walker);
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Check if 'Walker' points to the end of an entire device path
|
|
|
|
//
|
|
|
|
if (Walker->SubType == END_ENTIRE_DEVICE_PATH_SUBTYPE) {
|
|
|
|
*EntireDevicePathEnd = TRUE;
|
|
|
|
} else if (Walker->SubType == END_INSTANCE_DEVICE_PATH_SUBTYPE) {
|
|
|
|
*EntireDevicePathEnd = FALSE;
|
|
|
|
} else {
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Compute the size of the device path instance
|
|
|
|
//
|
2021-12-05 23:54:02 +01:00
|
|
|
*InstanceSize = ((UINTN)Walker - (UINTN)(DevicePath)) + sizeof (EFI_DEVICE_PATH_PROTOCOL);
|
2019-01-21 07:14:19 +01:00
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2019-01-29 09:43:41 +01:00
|
|
|
/**
|
|
|
|
Check the validity of the device path of a NVM Express host controller.
|
|
|
|
|
|
|
|
@param[in] DevicePath A pointer to the EFI_DEVICE_PATH_PROTOCOL
|
|
|
|
structure.
|
|
|
|
@param[in] DevicePathLength The length of the device path.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS The device path is valid.
|
|
|
|
@retval EFI_INVALID_PARAMETER The device path is invalid.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
NvmeIsHcDevicePathValid (
|
2021-12-05 23:54:02 +01:00
|
|
|
IN EFI_DEVICE_PATH_PROTOCOL *DevicePath,
|
|
|
|
IN UINTN DevicePathLength
|
2019-01-29 09:43:41 +01:00
|
|
|
)
|
|
|
|
{
|
2021-12-05 23:54:02 +01:00
|
|
|
EFI_DEVICE_PATH_PROTOCOL *Start;
|
|
|
|
UINTN Size;
|
2019-01-29 09:43:41 +01:00
|
|
|
|
|
|
|
if (DevicePath == NULL) {
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Validate the DevicePathLength is big enough to touch the first node.
|
|
|
|
//
|
|
|
|
if (DevicePathLength < sizeof (EFI_DEVICE_PATH_PROTOCOL)) {
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
Start = DevicePath;
|
|
|
|
while (!(DevicePath->Type == END_DEVICE_PATH_TYPE &&
|
2021-12-05 23:54:02 +01:00
|
|
|
DevicePath->SubType == END_ENTIRE_DEVICE_PATH_SUBTYPE))
|
|
|
|
{
|
2019-01-29 09:43:41 +01:00
|
|
|
DevicePath = NextDevicePathNode (DevicePath);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Prevent overflow and invalid zero in the 'Length' field of a device path
|
|
|
|
// node.
|
|
|
|
//
|
2021-12-05 23:54:02 +01:00
|
|
|
if ((UINTN)DevicePath <= (UINTN)Start) {
|
2019-01-29 09:43:41 +01:00
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Prevent touching memory beyond given DevicePathLength.
|
|
|
|
//
|
2021-12-05 23:54:02 +01:00
|
|
|
if ((UINTN)DevicePath - (UINTN)Start >
|
|
|
|
DevicePathLength - sizeof (EFI_DEVICE_PATH_PROTOCOL))
|
|
|
|
{
|
2019-01-29 09:43:41 +01:00
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Check if the device path and its size match exactly with each other.
|
|
|
|
//
|
2021-12-05 23:54:02 +01:00
|
|
|
Size = ((UINTN)DevicePath - (UINTN)Start) + sizeof (EFI_DEVICE_PATH_PROTOCOL);
|
2019-01-29 09:43:41 +01:00
|
|
|
if (Size != DevicePathLength) {
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Build the device path for an Nvm Express device with given namespace identifier
|
|
|
|
and namespace extended unique identifier.
|
|
|
|
|
|
|
|
@param[in] Private A pointer to the PEI_NVME_CONTROLLER_PRIVATE_DATA
|
|
|
|
data structure.
|
|
|
|
@param[in] NamespaceId The given namespace identifier.
|
|
|
|
@param[in] NamespaceUuid The given namespace extended unique identifier.
|
|
|
|
@param[out] DevicePathLength The length of the device path in bytes specified
|
|
|
|
by DevicePath.
|
|
|
|
@param[out] DevicePath The device path of Nvm Express device.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS The operation succeeds.
|
|
|
|
@retval EFI_INVALID_PARAMETER The parameters are invalid.
|
|
|
|
@retval EFI_OUT_OF_RESOURCES The operation fails due to lack of resources.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
NvmeBuildDevicePath (
|
2021-12-05 23:54:02 +01:00
|
|
|
IN PEI_NVME_CONTROLLER_PRIVATE_DATA *Private,
|
|
|
|
IN UINT32 NamespaceId,
|
|
|
|
IN UINT64 NamespaceUuid,
|
|
|
|
OUT UINTN *DevicePathLength,
|
|
|
|
OUT EFI_DEVICE_PATH_PROTOCOL **DevicePath
|
2019-01-29 09:43:41 +01:00
|
|
|
)
|
|
|
|
{
|
2021-12-05 23:54:02 +01:00
|
|
|
EFI_DEVICE_PATH_PROTOCOL *DevicePathWalker;
|
|
|
|
NVME_NAMESPACE_DEVICE_PATH *NvmeDeviceNode;
|
2019-01-29 09:43:41 +01:00
|
|
|
|
2021-12-05 23:54:02 +01:00
|
|
|
if ((DevicePathLength == NULL) || (DevicePath == NULL)) {
|
2019-01-29 09:43:41 +01:00
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
*DevicePathLength = Private->DevicePathLength + sizeof (NVME_NAMESPACE_DEVICE_PATH);
|
|
|
|
*DevicePath = AllocatePool (*DevicePathLength);
|
|
|
|
if (*DevicePath == NULL) {
|
|
|
|
*DevicePathLength = 0;
|
|
|
|
return EFI_OUT_OF_RESOURCES;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Construct the host controller part device nodes
|
|
|
|
//
|
|
|
|
DevicePathWalker = *DevicePath;
|
|
|
|
CopyMem (
|
|
|
|
DevicePathWalker,
|
|
|
|
Private->DevicePath,
|
|
|
|
Private->DevicePathLength - sizeof (EFI_DEVICE_PATH_PROTOCOL)
|
|
|
|
);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Construct the Nvm Express device node
|
|
|
|
//
|
2021-12-05 23:54:02 +01:00
|
|
|
DevicePathWalker = (EFI_DEVICE_PATH_PROTOCOL *)((UINT8 *)DevicePathWalker +
|
|
|
|
(Private->DevicePathLength - sizeof (EFI_DEVICE_PATH_PROTOCOL)));
|
2019-01-29 09:43:41 +01:00
|
|
|
CopyMem (
|
|
|
|
DevicePathWalker,
|
|
|
|
&mNvmeDevicePathNodeTemplate,
|
|
|
|
sizeof (mNvmeDevicePathNodeTemplate)
|
|
|
|
);
|
|
|
|
NvmeDeviceNode = (NVME_NAMESPACE_DEVICE_PATH *)DevicePathWalker;
|
|
|
|
NvmeDeviceNode->NamespaceId = NamespaceId;
|
|
|
|
NvmeDeviceNode->NamespaceUuid = NamespaceUuid;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Construct the end device node
|
|
|
|
//
|
2021-12-05 23:54:02 +01:00
|
|
|
DevicePathWalker = (EFI_DEVICE_PATH_PROTOCOL *)((UINT8 *)DevicePathWalker +
|
|
|
|
sizeof (NVME_NAMESPACE_DEVICE_PATH));
|
2019-01-29 09:43:41 +01:00
|
|
|
CopyMem (
|
|
|
|
DevicePathWalker,
|
|
|
|
&mNvmeEndDevicePathNodeTemplate,
|
|
|
|
sizeof (mNvmeEndDevicePathNodeTemplate)
|
|
|
|
);
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|