/** @file
Configuration Manager Dxe
Copyright (c) 2021, ARM Limited. All rights reserved.
SPDX-License-Identifier: BSD-2-Clause-Patent
@par Glossary:
- Cm or CM - Configuration Manager
- Obj or OBJ - Object
**/
#include
#include
#include
#include
#include
#include
#include
#include "ConfigurationManagerFvp.h"
#include "Platform.h"
EDKII_FVP_PLATFORM_REPOSITORY_INFO MorelloFvpRepositoryInfo = {
// ACPI Table List
{
// FADT Table
{
EFI_ACPI_6_3_FIXED_ACPI_DESCRIPTION_TABLE_SIGNATURE,
EFI_ACPI_6_3_FIXED_ACPI_DESCRIPTION_TABLE_REVISION,
CREATE_STD_ACPI_TABLE_GEN_ID (EStdAcpiTableIdFadt),
NULL
},
// GTDT Table
{
EFI_ACPI_6_3_GENERIC_TIMER_DESCRIPTION_TABLE_SIGNATURE,
EFI_ACPI_6_3_GENERIC_TIMER_DESCRIPTION_TABLE_REVISION,
CREATE_STD_ACPI_TABLE_GEN_ID (EStdAcpiTableIdGtdt),
NULL
},
// MADT Table
{
EFI_ACPI_6_3_MULTIPLE_APIC_DESCRIPTION_TABLE_SIGNATURE,
EFI_ACPI_6_3_MULTIPLE_APIC_DESCRIPTION_TABLE_REVISION,
CREATE_STD_ACPI_TABLE_GEN_ID (EStdAcpiTableIdMadt),
NULL
},
// SPCR Table
{
EFI_ACPI_6_3_SERIAL_PORT_CONSOLE_REDIRECTION_TABLE_SIGNATURE,
EFI_ACPI_SERIAL_PORT_CONSOLE_REDIRECTION_TABLE_REVISION,
CREATE_STD_ACPI_TABLE_GEN_ID (EStdAcpiTableIdSpcr),
NULL
},
// DSDT Table
{
EFI_ACPI_6_3_DIFFERENTIATED_SYSTEM_DESCRIPTION_TABLE_SIGNATURE,
0, // Unused
CREATE_STD_ACPI_TABLE_GEN_ID (EStdAcpiTableIdDsdt),
(EFI_ACPI_DESCRIPTION_HEADER*)dsdtfvp_aml_code
},
// DBG2 Table
{
EFI_ACPI_6_3_DEBUG_PORT_2_TABLE_SIGNATURE,
EFI_ACPI_DBG2_DEBUG_DEVICE_INFORMATION_STRUCT_REVISION,
CREATE_STD_ACPI_TABLE_GEN_ID (EStdAcpiTableIdDbg2),
NULL
},
// PPTT Table
{
EFI_ACPI_6_3_PROCESSOR_PROPERTIES_TOPOLOGY_TABLE_STRUCTURE_SIGNATURE,
EFI_ACPI_6_3_PROCESSOR_PROPERTIES_TOPOLOGY_TABLE_REVISION,
CREATE_STD_ACPI_TABLE_GEN_ID (EStdAcpiTableIdPptt),
NULL
},
// IORT Table
{
EFI_ACPI_6_3_IO_REMAPPING_TABLE_SIGNATURE,
EFI_ACPI_IO_REMAPPING_TABLE_REVISION,
CREATE_STD_ACPI_TABLE_GEN_ID (EStdAcpiTableIdIort),
NULL
},
// PCI MCFG Table
{
EFI_ACPI_6_3_PCI_EXPRESS_MEMORY_MAPPED_CONFIGURATION_SPACE_BASE_ADDRESS_DESCRIPTION_TABLE_SIGNATURE,
EFI_ACPI_MEMORY_MAPPED_CONFIGURATION_SPACE_ACCESS_TABLE_REVISION,
CREATE_STD_ACPI_TABLE_GEN_ID (EStdAcpiTableIdMcfg),
NULL,
},
// SSDT table describing the PCI root complex
{
EFI_ACPI_6_3_SECONDARY_SYSTEM_DESCRIPTION_TABLE_SIGNATURE,
0, // Unused
CREATE_STD_ACPI_TABLE_GEN_ID (EStdAcpiTableIdSsdt),
(EFI_ACPI_DESCRIPTION_HEADER*)ssdtpcifvp_aml_code
},
},
// GIC ITS
{
// GIC ITS - PCIe TCU
{
// The GIC ITS ID.
0,
// The physical address for the Interrupt Translation Service
0x30060000,
//Proximity Domain
0
},
// GIC ITS - PCIe RC
{
// The GIC ITS ID.
1,
// The physical address for the Interrupt Translation Service
0x300A0000,
//Proximity Domain
0
},
},
// ITS group node
{
{
// Reference token for this Iort node
REFERENCE_TOKEN_FVP (ItsGroupInfo[0]),
// The number of ITS identifiers in the ITS node.
1,
// Reference token for the ITS identifier array
REFERENCE_TOKEN_FVP (ItsIdentifierArray[0])
},
{
// Reference token for this Iort node
REFERENCE_TOKEN_FVP (ItsGroupInfo[1]),
// The number of ITS identifiers in the ITS node.
1,
// Reference token for the ITS identifier array
REFERENCE_TOKEN_FVP (ItsIdentifierArray[1])
},
},
// ITS identifier array
{
{
// The ITS Identifier
0
},
{
// The ITS Identifier
1
},
},
// SMMUv3 Node
{
{
// Reference token for this Iort node
REFERENCE_TOKEN_FVP (SmmuV3Info[0]),
// Number of ID mappings
2,
// Reference token for the ID mapping array
REFERENCE_TOKEN_FVP (DeviceIdMapping[0][0]),
// SMMU Base Address
0x4F400000,
// SMMU flags
EFI_ACPI_IORT_SMMUv3_FLAG_COHAC_OVERRIDE,
// VATOS address
0,
// Model
EFI_ACPI_IORT_SMMUv3_MODEL_GENERIC,
// GSIV of the Event interrupt if SPI based
0x10B,
// PRI Interrupt if SPI based
0,
// GERR interrupt if GSIV based
0x10D,
// Sync interrupt if GSIV based
0x10C,
// Proximity domain flag, ignored in this case
0,
// Index into the array of ID mapping
1
},
},
// Root Complex node info
{
{
// Reference token for this Iort node
REFERENCE_TOKEN_FVP (RootComplexInfo[0]),
// Number of ID mappings
1,
// Reference token for the ID mapping array
REFERENCE_TOKEN_FVP (DeviceIdMapping[1][0]),
// Memory access properties : Cache coherent attributes
EFI_ACPI_IORT_MEM_ACCESS_PROP_CCA,
// Memory access properties : Allocation hints
0,
// Memory access properties : Memory access flags
0,
// ATS attributes
EFI_ACPI_IORT_ROOT_COMPLEX_ATS_SUPPORTED,
// PCI segment number
0,
// Memory address size limit
42
},
},
// Array of Device ID mappings
{
// DeviceIdMapping[0][0] - [0][1]
{
/* Mapping SMMUv3 -> ITS Group
*/
// SMMUv3 device ID mapping
{
// Input base
0x0,
// Number of input IDs
0x0000FFFF,
// Output Base
0x0,
// Output reference
REFERENCE_TOKEN_FVP (ItsGroupInfo[1]),
// Flags
0
},
// SMMUv3 device ID mapping
{
// Input base
0x0,
// Number of input IDs
0x00000001,
// Output Base
0x0,
// Output reference token for the IORT node
REFERENCE_TOKEN_FVP (ItsGroupInfo[0]),
// Flags
EFI_ACPI_IORT_ID_MAPPING_FLAGS_SINGLE
}
},
// DeviceIdMapping[1][0]
{
// Mapping for RootComplex -> SMMUv3
// Device ID mapping for Root complex node
{
// Input base
0x0,
// Number of input IDs
0x0000FFFF,
// Output Base
0x0,
// Output reference
REFERENCE_TOKEN_FVP (SmmuV3Info[0]),
// Flags
0
},
},
},
// PCI Configuration Space Info
{
// PCIe ECAM
{
FixedPcdGet64 (PcdPciExpressBaseAddress), // Base Address
0x0, // Segment Group Number
FixedPcdGet32 (PcdPciBusMin), // Start Bus Number
FixedPcdGet32 (PcdPciBusMax) // End Bus Number
},
},
};
EDKII_PLATFORM_REPOSITORY_INFO MorelloRepositoryInfo = {
&CommonPlatformInfo,
&MorelloFvpRepositoryInfo
};
/** Return a device Id mapping array.
@param [in] This Pointer to the Configuration Manager Protocol.
@param [in] CmObjectId The Configuration Manager Object ID.
@param [in] Token A token for identifying the object
@param [in, out] CmObject Pointer to the Configuration Manager Object
descriptor describing the requested Object.
@retval EFI_SUCCESS Success.
@retval EFI_INVALID_PARAMETER A parameter is invalid.
@retval EFI_NOT_FOUND The required object information is not found.
**/
EFI_STATUS
EFIAPI
GetDeviceIdMappingArray (
IN CONST EDKII_CONFIGURATION_MANAGER_PROTOCOL * CONST This,
IN CONST CM_OBJECT_ID CmObjectId,
IN CONST CM_OBJECT_TOKEN Token,
IN OUT CM_OBJ_DESCRIPTOR * CONST CmObject
)
{
EDKII_FVP_PLATFORM_REPOSITORY_INFO * PlatformRepo;
UINTN Count;
if ((This == NULL) || (CmObject == NULL)) {
ASSERT (This != NULL);
ASSERT (CmObject != NULL);
return EFI_INVALID_PARAMETER;
}
PlatformRepo = This->PlatRepoInfo->FvpPlatRepoInfo;
DEBUG ((DEBUG_INFO, "DeviceIdMapping - Token = %p\n"));
if (Token == (CM_OBJECT_TOKEN)&PlatformRepo->DeviceIdMapping[0][0]) {
Count = 2;
DEBUG ((DEBUG_INFO, "DeviceIdMapping - Found DeviceIdMapping[0][0]\n"));
} else if (Token ==
(CM_OBJECT_TOKEN)&PlatformRepo->DeviceIdMapping[1][0]) {
Count = 1;
DEBUG ((DEBUG_INFO, "DeviceIdMapping - Found DeviceIdMapping[1][0]\n"));
} else {
DEBUG ((DEBUG_INFO, "DeviceIdMapping - Not Found\n"));
return EFI_NOT_FOUND;
}
CmObject->Data = (VOID*)Token;
CmObject->ObjectId = CmObjectId;
CmObject->Count = Count;
CmObject->Size = Count * sizeof (CM_ARM_ID_MAPPING);
return EFI_SUCCESS;
}
/** Return an ITS identifier array.
@param [in] This Pointer to the Configuration Manager Protocol.
@param [in] CmObjectId The Configuration Manager Object ID.
@param [in] Token A token for identifying the object
@param [in, out] CmObject Pointer to the Configuration Manager Object
descriptor describing the requested Object.
@retval EFI_SUCCESS Success.
@retval EFI_INVALID_PARAMETER A parameter is invalid.
@retval EFI_NOT_FOUND The required object information is not found.
**/
EFI_STATUS
EFIAPI
GetItsIdentifierArray (
IN CONST EDKII_CONFIGURATION_MANAGER_PROTOCOL * CONST This,
IN CONST CM_OBJECT_ID CmObjectId,
IN CONST CM_OBJECT_TOKEN Token,
IN OUT CM_OBJ_DESCRIPTOR * CONST CmObject
)
{
EDKII_FVP_PLATFORM_REPOSITORY_INFO * PlatformRepo;
UINTN Count;
UINTN Index;
if ((This == NULL) || (CmObject == NULL)) {
ASSERT (This != NULL);
ASSERT (CmObject != NULL);
return EFI_INVALID_PARAMETER;
}
PlatformRepo = This->PlatRepoInfo->FvpPlatRepoInfo;
Count = ARRAY_SIZE (PlatformRepo->ItsIdentifierArray);
for (Index = 0; Index < Count; Index++) {
if (Token == (CM_OBJECT_TOKEN)&PlatformRepo->ItsIdentifierArray[Index]) {
CmObject->ObjectId = CmObjectId;
CmObject->Size = sizeof (PlatformRepo->ItsIdentifierArray[0]);
CmObject->Data = (VOID*)&PlatformRepo->ItsIdentifierArray[Index];
CmObject->Count = 1;
return EFI_SUCCESS;
}
}
return EFI_NOT_FOUND;
}
/** Return an ITS group info.
@param [in] This Pointer to the Configuration Manager Protocol.
@param [in] CmObjectId The Configuration Manager Object ID.
@param [in] Token A token for identifying the object
@param [in, out] CmObject Pointer to the Configuration Manager Object
descriptor describing the requested Object.
@retval EFI_SUCCESS Success.
@retval EFI_INVALID_PARAMETER A parameter is invalid.
@retval EFI_NOT_FOUND The required object information is not found.
**/
EFI_STATUS
EFIAPI
GetItsGroupInfo (
IN CONST EDKII_CONFIGURATION_MANAGER_PROTOCOL * CONST This,
IN CONST CM_OBJECT_ID CmObjectId,
IN CONST CM_OBJECT_TOKEN Token,
IN OUT CM_OBJ_DESCRIPTOR * CONST CmObject
)
{
EDKII_FVP_PLATFORM_REPOSITORY_INFO * PlatformRepo;
UINTN Count;
UINTN Index;
if ((This == NULL) || (CmObject == NULL)) {
ASSERT (This != NULL);
ASSERT (CmObject != NULL);
return EFI_INVALID_PARAMETER;
}
PlatformRepo = This->PlatRepoInfo->FvpPlatRepoInfo;
Count = ARRAY_SIZE (PlatformRepo->ItsGroupInfo);
for (Index = 0; Index < Count; Index++) {
if (Token == (CM_OBJECT_TOKEN)&PlatformRepo->ItsGroupInfo[Index]) {
CmObject->ObjectId = CmObjectId;
CmObject->Size = sizeof (PlatformRepo->ItsGroupInfo[0]);
CmObject->Data = (VOID*)&PlatformRepo->ItsGroupInfo[Index];
CmObject->Count = 1;
return EFI_SUCCESS;
}
}
return EFI_NOT_FOUND;
}
/** Return platform specific ARM namespace object.
@param [in] This Pointer to the Configuration Manager Protocol.
@param [in] CmObjectId The Configuration Manager Object ID.
@param [in] Token An optional token identifying the object. If
unused this must be CM_NULL_TOKEN.
@param [in, out] CmObject Pointer to the Configuration Manager Object
descriptor describing the requested Object.
@retval EFI_SUCCESS Success.
@retval EFI_INVALID_PARAMETER A parameter is invalid.
@retval EFI_NOT_FOUND The required object information is not found.
**/
EFI_STATUS
EFIAPI
GetArmNameSpaceObjectPlat (
IN CONST EDKII_CONFIGURATION_MANAGER_PROTOCOL * CONST This,
IN CONST CM_OBJECT_ID CmObjectId,
IN CONST CM_OBJECT_TOKEN Token OPTIONAL,
IN OUT CM_OBJ_DESCRIPTOR * CONST CmObject
)
{
EFI_STATUS Status;
EDKII_FVP_PLATFORM_REPOSITORY_INFO * PlatformRepo;
if ((This == NULL) || (CmObject == NULL)) {
ASSERT (This != NULL);
ASSERT (CmObject != NULL);
return EFI_INVALID_PARAMETER;
}
PlatformRepo = This->PlatRepoInfo->FvpPlatRepoInfo;
Status = EFI_NOT_FOUND;
switch (GET_CM_OBJECT_ID (CmObjectId)) {
case EArmObjGicItsInfo:
Status = HandleCmObject (
CmObjectId,
&PlatformRepo->GicItsInfo,
sizeof (PlatformRepo->GicItsInfo),
ARRAY_SIZE (PlatformRepo->GicItsInfo),
CmObject
);
break;
case EArmObjSmmuV3:
Status = HandleCmObject (
CmObjectId,
PlatformRepo->SmmuV3Info,
sizeof (PlatformRepo->SmmuV3Info),
1,
CmObject
);
break;
case EArmObjItsGroup:
Status = HandleCmObjectRefByToken (
This,
CmObjectId,
PlatformRepo->ItsGroupInfo,
sizeof (PlatformRepo->ItsGroupInfo),
ARRAY_SIZE (PlatformRepo->ItsGroupInfo),
Token,
GetItsGroupInfo,
CmObject
);
break;
case EArmObjGicItsIdentifierArray:
Status = HandleCmObjectRefByToken (
This,
CmObjectId,
PlatformRepo->ItsIdentifierArray,
sizeof (PlatformRepo->ItsIdentifierArray),
ARRAY_SIZE (PlatformRepo->ItsIdentifierArray),
Token,
GetItsIdentifierArray,
CmObject
);
break;
case EArmObjRootComplex:
Status = HandleCmObject (
CmObjectId,
PlatformRepo->RootComplexInfo,
sizeof (PlatformRepo->RootComplexInfo),
1,
CmObject
);
break;
case EArmObjIdMappingArray:
Status = HandleCmObjectRefByToken (
This,
CmObjectId,
PlatformRepo->DeviceIdMapping,
sizeof (PlatformRepo->DeviceIdMapping),
ARRAY_SIZE (PlatformRepo->DeviceIdMapping),
Token,
GetDeviceIdMappingArray,
CmObject
);
break;
case EArmObjPciConfigSpaceInfo:
Status = HandleCmObject (
CmObjectId,
PlatformRepo->PciConfigInfo,
sizeof (PlatformRepo->PciConfigInfo),
1,
CmObject
);
break;
default: {
break;
}
}//switch
return Status;
}
/** Return platform specific standard namespace object.
@param [in] This Pointer to the Configuration Manager Protocol.
@param [in] CmObjectId The Configuration Manager Object ID.
@param [in] Token An optional token identifying the object. If
unused this must be CM_NULL_TOKEN.
@param [in, out] CmObject Pointer to the Configuration Manager Object
descriptor describing the requested Object.
@retval EFI_SUCCESS Success.
@retval EFI_INVALID_PARAMETER A parameter is invalid.
@retval EFI_NOT_FOUND The required object information is not found.
**/
EFI_STATUS
EFIAPI
GetStandardNameSpaceObjectPlat (
IN CONST EDKII_CONFIGURATION_MANAGER_PROTOCOL * CONST This,
IN CONST CM_OBJECT_ID CmObjectId,
IN CONST CM_OBJECT_TOKEN Token OPTIONAL,
IN OUT CM_OBJ_DESCRIPTOR * CONST CmObject
)
{
EFI_STATUS Status;
EDKII_FVP_PLATFORM_REPOSITORY_INFO * PlatformRepo;
if ((This == NULL) || (CmObject == NULL)) {
ASSERT (This != NULL);
ASSERT (CmObject != NULL);
return EFI_INVALID_PARAMETER;
}
Status = EFI_NOT_FOUND;
PlatformRepo = This->PlatRepoInfo->FvpPlatRepoInfo;
switch (GET_CM_OBJECT_ID (CmObjectId)) {
case EStdObjAcpiTableList:
Status = HandleCmObject (
CmObjectId,
PlatformRepo->CmAcpiTableList,
sizeof (PlatformRepo->CmAcpiTableList),
ARRAY_SIZE (PlatformRepo->CmAcpiTableList),
CmObject
);
break;
default: {
break;
}
}
return Status;
}