/** @file
|
*
|
* Copyright (c) 2018, Hisilicon Limited. All rights reserved.
|
* Copyright (c) 2018, Linaro Limited. All rights reserved.
|
*
|
* SPDX-License-Identifier: BSD-2-Clause-Patent
|
*
|
* Based on the files under Platform/ARM/JunoPkg/AcpiTables/
|
*
|
**/
|
|
#include "Pptt.h"
|
|
typedef EFI_ACPI_5_1_GIC_STRUCTURE ACPI_GIC_STRUCTURE;
|
typedef EFI_ACPI_5_1_MULTIPLE_APIC_DESCRIPTION_TABLE_HEADER ACPI_MADT_TABLE_HEADER;
|
|
EFI_ACPI_TABLE_PROTOCOL *mAcpiTableProtocol = NULL;
|
EFI_ACPI_SDT_PROTOCOL *mAcpiSdtProtocol = NULL;
|
|
EFI_ACPI_DESCRIPTION_HEADER mPpttHeader =
|
ARM_ACPI_HEADER (
|
EFI_ACPI_6_2_PROCESSOR_PROPERTIES_TOPOLOGY_TABLE_STRUCTURE_SIGNATURE,
|
EFI_ACPI_DESCRIPTION_HEADER,
|
EFI_ACPI_6_2_PROCESSOR_PROPERTIES_TOPOLOGY_TABLE_REVISION
|
);
|
|
EFI_ACPI_6_2_PPTT_STRUCTURE_ID mPpttSocketType2[PPTT_SOCKET_COMPONENT_NO] =
|
{
|
{2, sizeof (EFI_ACPI_6_2_PPTT_STRUCTURE_ID), {0, 0}, PPTT_VENDOR_ID, 0, 0, 0, 0, 0}
|
};
|
|
EFI_ACPI_6_2_PPTT_STRUCTURE_CACHE mPpttCacheType1[PPTT_CACHE_NO];
|
|
STATIC UINT32 mSocketOffset[MAX_SOCKET];
|
STATIC UINT32 mScclOffset[MAX_SCL];
|
STATIC UINT32 mClusterOffset[MAX_SCL][MAX_CLUSTER_PER_SCL];
|
|
STATIC
|
VOID
|
InitCacheInfo (
|
VOID
|
)
|
{
|
UINT8 Index;
|
EFI_ACPI_6_2_PPTT_STRUCTURE_CACHE_ATTRIBUTES Type1Attributes;
|
CSSELR_DATA CsselrData;
|
CCSIDR_DATA CcsidrData;
|
|
for (Index = 0; Index < PPTT_CACHE_NO; Index++) {
|
CsselrData.Data = 0;
|
CcsidrData.Data = 0;
|
SetMem (
|
&Type1Attributes,
|
sizeof (EFI_ACPI_6_2_PPTT_STRUCTURE_CACHE_ATTRIBUTES),
|
0
|
);
|
|
if (Index == 0) { //L1I
|
CsselrData.Bits.InD = 1;
|
CsselrData.Bits.Level = 0;
|
Type1Attributes.CacheType = 1;
|
} else if (Index == 1) {
|
Type1Attributes.CacheType = 0;
|
CsselrData.Bits.Level = Index - 1;
|
} else {
|
Type1Attributes.CacheType = 2;
|
CsselrData.Bits.Level = Index - 1;
|
}
|
|
CcsidrData.Data = ReadCCSIDR (CsselrData.Data);
|
|
if (CcsidrData.Bits.Wa == 1) {
|
Type1Attributes.AllocationType = EFI_ACPI_6_2_CACHE_ATTRIBUTES_ALLOCATION_WRITE;
|
if (CcsidrData.Bits.Ra == 1) {
|
Type1Attributes.AllocationType = EFI_ACPI_6_2_CACHE_ATTRIBUTES_ALLOCATION_READ_WRITE;
|
}
|
}
|
|
if (CcsidrData.Bits.Wt == 1) {
|
Type1Attributes.WritePolicy = 1;
|
}
|
DEBUG ((DEBUG_INFO,
|
"[Acpi PPTT] Level = %x!CcsidrData = %x!\n",
|
CsselrData.Bits.Level,
|
CcsidrData.Data));
|
|
mPpttCacheType1[Index].Type = EFI_ACPI_6_2_PPTT_TYPE_CACHE;
|
mPpttCacheType1[Index].Length = sizeof (EFI_ACPI_6_2_PPTT_STRUCTURE_CACHE);
|
mPpttCacheType1[Index].Reserved[0] = 0;
|
mPpttCacheType1[Index].Reserved[1] = 0;
|
mPpttCacheType1[Index].Flags.SizePropertyValid = 1;
|
mPpttCacheType1[Index].Flags.NumberOfSetsValid = 1;
|
mPpttCacheType1[Index].Flags.AssociativityValid = 1;
|
mPpttCacheType1[Index].Flags.AllocationTypeValid = 1;
|
mPpttCacheType1[Index].Flags.CacheTypeValid = 1;
|
mPpttCacheType1[Index].Flags.WritePolicyValid = 1;
|
mPpttCacheType1[Index].Flags.LineSizeValid = 1;
|
mPpttCacheType1[Index].Flags.Reserved = 0;
|
mPpttCacheType1[Index].NextLevelOfCache = 0;
|
|
if (Index != PPTT_CACHE_NO - 1) {
|
mPpttCacheType1[Index].NumberOfSets = (UINT16)CcsidrData.Bits.NumSets + 1;
|
mPpttCacheType1[Index].Associativity = (UINT16)CcsidrData.Bits.Associativity + 1;
|
mPpttCacheType1[Index].LineSize = (UINT16)( 1 << (CcsidrData.Bits.LineSize + 4));
|
mPpttCacheType1[Index].Size = mPpttCacheType1[Index].LineSize * \
|
mPpttCacheType1[Index].Associativity * \
|
mPpttCacheType1[Index].NumberOfSets;
|
CopyMem (
|
&mPpttCacheType1[Index].Attributes,
|
&Type1Attributes,
|
sizeof (EFI_ACPI_6_2_PPTT_STRUCTURE_CACHE_ATTRIBUTES)
|
);
|
} else {
|
// L3 cache
|
mPpttCacheType1[Index].Size = 0x2000000; // 32MB
|
mPpttCacheType1[Index].NumberOfSets = 0x800;
|
mPpttCacheType1[Index].Associativity = 0x0F; // CacheAssociativity16Way
|
SetMem (
|
&mPpttCacheType1[Index].Attributes,
|
sizeof (EFI_ACPI_6_2_PPTT_STRUCTURE_CACHE_ATTRIBUTES),
|
0x0A
|
);
|
mPpttCacheType1[Index].LineSize = 0x80; // 128byte
|
}
|
}
|
}
|
|
STATIC
|
EFI_STATUS
|
AddCoreTable (
|
IN EFI_ACPI_DESCRIPTION_HEADER *PpttTable,
|
IN OUT UINT32 *PpttTableLengthRemain,
|
IN UINT32 Parent,
|
IN UINT32 ResourceNo,
|
IN UINT32 ProcessorId
|
)
|
{
|
EFI_ACPI_6_2_PPTT_STRUCTURE_PROCESSOR *PpttType0;
|
EFI_ACPI_6_2_PPTT_STRUCTURE_CACHE *PpttType1;
|
UINT32 *PrivateResource;
|
UINT8 Index;
|
UINT32 NextLevelCacheOffset;
|
|
if (*PpttTableLengthRemain <
|
(sizeof (EFI_ACPI_6_2_PPTT_STRUCTURE_PROCESSOR) + ResourceNo * 4)) {
|
return EFI_OUT_OF_RESOURCES;
|
}
|
PpttType0 = (EFI_ACPI_6_2_PPTT_STRUCTURE_PROCESSOR *)((UINT8 *)PpttTable +
|
PpttTable->Length);
|
PpttType0->Type = 0;
|
SetMem (&PpttType0->Flags, sizeof (PpttType0->Flags), 0);
|
PpttType0->Flags.AcpiProcessorIdValid = EFI_ACPI_6_2_PPTT_PROCESSOR_ID_VALID;
|
PpttType0->Parent= Parent;
|
PpttType0->AcpiProcessorId = ProcessorId;
|
PpttType0->NumberOfPrivateResources = ResourceNo;
|
PpttType0->Length = sizeof (EFI_ACPI_6_2_PPTT_STRUCTURE_PROCESSOR) +
|
ResourceNo * 4;
|
|
*PpttTableLengthRemain -= (UINTN)PpttType0->Length;
|
PpttTable->Length += PpttType0->Length;
|
PrivateResource = (UINT32 *)((UINT8 *)PpttType0 +
|
sizeof (EFI_ACPI_6_2_PPTT_STRUCTURE_PROCESSOR));
|
|
// Add cache type structure
|
for (Index = 0; Index < ResourceNo; Index++, PrivateResource++) {
|
if (*PpttTableLengthRemain < sizeof (EFI_ACPI_6_2_PPTT_STRUCTURE_CACHE)) {
|
return EFI_OUT_OF_RESOURCES;
|
}
|
*PrivateResource = PpttTable->Length;
|
PpttType1 = (EFI_ACPI_6_2_PPTT_STRUCTURE_CACHE *)((UINT8 *)PpttTable +
|
PpttTable->Length);
|
gBS->CopyMem (
|
PpttType1,
|
&mPpttCacheType1[Index],
|
sizeof (EFI_ACPI_6_2_PPTT_STRUCTURE_CACHE)
|
);
|
*PpttTableLengthRemain -= PpttType1->Length;
|
PpttTable->Length += PpttType1->Length;
|
}
|
|
NextLevelCacheOffset = *(PrivateResource - 1);
|
PrivateResource = (UINT32 *)(PpttType0 + 1);
|
// Set the next level to L2 for L1I and L1D
|
PpttType1 = (EFI_ACPI_6_2_PPTT_STRUCTURE_CACHE *)((UINT8 *) PpttTable + *PrivateResource++);
|
PpttType1->NextLevelOfCache = NextLevelCacheOffset;
|
PpttType1 = (EFI_ACPI_6_2_PPTT_STRUCTURE_CACHE *)((UINT8 *) PpttTable + *PrivateResource++);
|
PpttType1->NextLevelOfCache = NextLevelCacheOffset;
|
|
return EFI_SUCCESS;
|
}
|
|
STATIC
|
EFI_STATUS
|
AddClusterTable (
|
IN EFI_ACPI_DESCRIPTION_HEADER *PpttTable,
|
IN OUT UINT32 *PpttTableLengthRemain,
|
IN UINT32 Parent,
|
IN UINT32 ResourceNo
|
)
|
{
|
EFI_ACPI_6_2_PPTT_STRUCTURE_PROCESSOR *PpttType0;
|
|
if ((*PpttTableLengthRemain) <
|
(sizeof (EFI_ACPI_6_2_PPTT_STRUCTURE_PROCESSOR) + ResourceNo * 4)) {
|
return EFI_OUT_OF_RESOURCES;
|
}
|
PpttType0 = (EFI_ACPI_6_2_PPTT_STRUCTURE_PROCESSOR *)((UINT8 *)PpttTable +
|
PpttTable->Length);
|
PpttType0->Type = 0;
|
SetMem (&PpttType0->Flags, sizeof (PpttType0->Flags), 0);
|
PpttType0->Parent= Parent;
|
PpttType0->NumberOfPrivateResources = ResourceNo;
|
PpttType0->Length = sizeof (EFI_ACPI_6_2_PPTT_STRUCTURE_PROCESSOR) +
|
ResourceNo * 4;
|
|
*PpttTableLengthRemain -= PpttType0->Length;
|
PpttTable->Length += PpttType0->Length;
|
|
return EFI_SUCCESS;
|
}
|
|
STATIC
|
EFI_STATUS
|
AddScclTable (
|
IN EFI_ACPI_DESCRIPTION_HEADER *PpttTable,
|
IN OUT UINT32 *PpttTableLengthRemain,
|
IN UINT32 Parent,
|
IN UINT32 ResourceNo
|
)
|
{
|
EFI_ACPI_6_2_PPTT_STRUCTURE_PROCESSOR *PpttType0;
|
EFI_ACPI_6_2_PPTT_STRUCTURE_CACHE *PpttType1;
|
UINT32 *PrivateResource;
|
|
if (*PpttTableLengthRemain <
|
(sizeof (EFI_ACPI_6_2_PPTT_STRUCTURE_PROCESSOR) + ResourceNo * 4)) {
|
return EFI_OUT_OF_RESOURCES;
|
}
|
PpttType0 = (EFI_ACPI_6_2_PPTT_STRUCTURE_PROCESSOR *)((UINT8 *)PpttTable +
|
PpttTable->Length);
|
PpttType0->Type = 0;
|
SetMem (&PpttType0->Flags, sizeof (PpttType0->Flags), 0);
|
PpttType0->Parent= Parent;
|
PpttType0->NumberOfPrivateResources = ResourceNo;
|
PpttType0->Length = sizeof (EFI_ACPI_6_2_PPTT_STRUCTURE_PROCESSOR) +
|
ResourceNo * 4;
|
|
*PpttTableLengthRemain -= PpttType0->Length;
|
PpttTable->Length += PpttType0->Length;
|
PrivateResource = (UINT32 *)((UINT8 *)PpttType0 +
|
sizeof (EFI_ACPI_6_2_PPTT_STRUCTURE_PROCESSOR));
|
|
// Add cache type structure
|
if (*PpttTableLengthRemain < sizeof (EFI_ACPI_6_2_PPTT_STRUCTURE_CACHE)) {
|
return EFI_OUT_OF_RESOURCES;
|
}
|
*PrivateResource = PpttTable->Length;
|
PpttType1 = (EFI_ACPI_6_2_PPTT_STRUCTURE_CACHE *)((UINT8 *)PpttTable +
|
PpttTable->Length);
|
gBS->CopyMem (
|
PpttType1,
|
&mPpttCacheType1[3],
|
sizeof (EFI_ACPI_6_2_PPTT_STRUCTURE_CACHE)
|
);
|
*PpttTableLengthRemain -= PpttType1->Length;
|
PpttTable->Length += PpttType1->Length;
|
|
return EFI_SUCCESS;
|
}
|
|
STATIC
|
EFI_STATUS
|
AddSocketTable (
|
IN EFI_ACPI_DESCRIPTION_HEADER *PpttTable,
|
IN OUT UINT32 *PpttTableLengthRemain,
|
IN UINT32 Parent,
|
IN UINT32 ResourceNo
|
)
|
{
|
EFI_ACPI_6_2_PPTT_STRUCTURE_PROCESSOR *PpttType0;
|
EFI_ACPI_6_2_PPTT_STRUCTURE_ID *PpttType2;
|
UINT32 *PrivateResource;
|
UINT8 Index;
|
|
if (*PpttTableLengthRemain < sizeof (EFI_ACPI_6_2_PPTT_STRUCTURE_PROCESSOR)) {
|
return EFI_OUT_OF_RESOURCES;
|
}
|
PpttType0 = (EFI_ACPI_6_2_PPTT_STRUCTURE_PROCESSOR *)((UINT8 *)PpttTable +
|
PpttTable->Length);
|
PpttType0->Type = 0;
|
SetMem (&PpttType0->Flags, sizeof (PpttType0->Flags), 0);
|
PpttType0->Flags.PhysicalPackage = EFI_ACPI_6_2_PPTT_PROCESSOR_ID_VALID;
|
PpttType0->Parent= Parent;
|
PpttType0->NumberOfPrivateResources = ResourceNo;
|
PpttType0->Length = sizeof (EFI_ACPI_6_2_PPTT_STRUCTURE_PROCESSOR) +
|
ResourceNo * 4;
|
PpttTable->Length += PpttType0->Length;
|
|
*PpttTableLengthRemain -= PpttType0->Length;
|
if (*PpttTableLengthRemain < ResourceNo * 4) {
|
return EFI_OUT_OF_RESOURCES;
|
}
|
PrivateResource = (UINT32 *)((UINT8 *)PpttType0 +
|
sizeof (EFI_ACPI_6_2_PPTT_STRUCTURE_PROCESSOR));
|
DEBUG ((DEBUG_INFO,
|
"[Acpi PPTT] sizeof(EFI_ACPI_6_2_PPTT_STRUCTURE_ID) = %x!\n",
|
sizeof (EFI_ACPI_6_2_PPTT_STRUCTURE_ID)));
|
|
for (Index = 0; Index < ResourceNo; Index++, PrivateResource++) {
|
if (*PpttTableLengthRemain < sizeof (EFI_ACPI_6_2_PPTT_STRUCTURE_ID)) {
|
return EFI_OUT_OF_RESOURCES;
|
}
|
*PrivateResource = PpttTable->Length;
|
PpttType2 = (EFI_ACPI_6_2_PPTT_STRUCTURE_ID *)((UINT8 *)PpttTable +
|
PpttTable->Length);
|
gBS->CopyMem (
|
PpttType2,
|
&mPpttSocketType2[Index],
|
sizeof (EFI_ACPI_6_2_PPTT_STRUCTURE_ID)
|
);
|
*PpttTableLengthRemain -= PpttType2->Length;
|
PpttTable->Length += PpttType2->Length;
|
}
|
|
return EFI_SUCCESS;
|
}
|
|
STATIC
|
VOID
|
GetAffLvl (
|
IN UINT64 Mpidr,
|
IN OUT UINT8 *Level3,
|
IN OUT UINT8 *Level2,
|
IN OUT UINT8 *Level1,
|
IN OUT UINT8 *Level0
|
)
|
{
|
*Level3 = BitFieldRead64 (Mpidr, 32, 39);
|
*Level2 = BitFieldRead64 (Mpidr, 16, 23);
|
*Level1 = BitFieldRead64 (Mpidr, 8, 15);
|
*Level0 = BitFieldRead64 (Mpidr, 0, 7);
|
}
|
|
|
STATIC
|
VOID
|
GetApic (
|
IN ACPI_MADT_TABLE_HEADER *ApicTable,
|
IN OUT EFI_ACPI_DESCRIPTION_HEADER *PpttTable,
|
IN UINT32 PpttTableLengthRemain
|
)
|
{
|
UINT32 Parent = 0;
|
UINT32 ResourceNo = 0;
|
ACPI_GIC_STRUCTURE *Ptr;
|
UINT8 AffLvl3 = 0;
|
UINT8 AffLvl2 = 0;
|
UINT8 AffLvl1 = 0;
|
UINT8 AffLvl0 = 0;
|
UINTN SocketIndex;
|
|
for (Ptr = (ACPI_GIC_STRUCTURE *) (ApicTable + 1);
|
(UINTN) Ptr < (UINTN) ApicTable + ApicTable->Header.Length;
|
Ptr = (ACPI_GIC_STRUCTURE *) ((UINTN) Ptr + Ptr->Length)) {
|
|
// Avoid dead loop due to corrupted MADT
|
if (Ptr->Length == 0) {
|
DEBUG ((DEBUG_ERROR, "[%a:%d] - Invalid MADT sub structure at 0x%x\n",
|
__FUNCTION__, __LINE__, (UINTN) Ptr - (UINTN) ApicTable));
|
break;
|
}
|
|
if (Ptr->Type != EFI_ACPI_5_1_GIC ||
|
(Ptr->Flags & EFI_ACPI_5_1_GIC_ENABLED) == 0) {
|
continue;
|
}
|
GetAffLvl (Ptr->MPIDR, &AffLvl3, &AffLvl2, &AffLvl1, &AffLvl0);
|
// AffLvl3 is not used for Hi1620
|
// And socket index is calculated by AffLvl2
|
|
SocketIndex = AffLvl2 / MAX_SCL_PER_SOCKET;
|
if (mSocketOffset[SocketIndex] == 0) {
|
//Add socket for type0 table
|
ResourceNo = PPTT_SOCKET_COMPONENT_NO;
|
mSocketOffset[SocketIndex] = PpttTable->Length;
|
Parent = 0;
|
AddSocketTable (
|
PpttTable,
|
&PpttTableLengthRemain,
|
Parent,
|
ResourceNo
|
);
|
}
|
|
if (mScclOffset[AffLvl2] == 0) {
|
//Add SCCL for type0 table
|
ResourceNo = 1;
|
mScclOffset[AffLvl2] = PpttTable->Length ;
|
Parent = mSocketOffset[SocketIndex];
|
AddScclTable (
|
PpttTable,
|
&PpttTableLengthRemain,
|
Parent,
|
ResourceNo
|
);
|
}
|
|
if (mClusterOffset[AffLvl2][AffLvl1] == 0) {
|
// Add cluster for type0 table
|
// No private resource for cluster on Hi1620
|
ResourceNo = 0;
|
mClusterOffset[AffLvl2][AffLvl1] = PpttTable->Length ;
|
Parent = mScclOffset[AffLvl2];
|
AddClusterTable (
|
PpttTable,
|
&PpttTableLengthRemain,
|
Parent,
|
ResourceNo
|
);
|
}
|
|
//Add core for type0 table
|
ResourceNo = 3;
|
Parent = mClusterOffset[AffLvl2][AffLvl1];
|
AddCoreTable (
|
PpttTable,
|
&PpttTableLengthRemain,
|
Parent,
|
ResourceNo,
|
Ptr->AcpiProcessorUid
|
);
|
}
|
}
|
|
|
STATIC
|
VOID
|
PpttSetAcpiTable(
|
IN EFI_EVENT Event,
|
IN VOID *Context
|
)
|
{
|
UINTN AcpiTableHandle;
|
EFI_STATUS Status;
|
UINT8 Checksum;
|
EFI_ACPI_SDT_HEADER *Table;
|
ACPI_MADT_TABLE_HEADER *ApicTable;
|
EFI_ACPI_TABLE_VERSION TableVersion;
|
EFI_ACPI_DESCRIPTION_HEADER *PpttTable;
|
UINTN TableKey;
|
UINT32 Index0;
|
UINT32 PpttTableLengthRemain = 0;
|
|
gBS->CloseEvent (Event);
|
|
InitCacheInfo ();
|
|
PpttTable = (EFI_ACPI_DESCRIPTION_HEADER *)AllocateZeroPool (PPTT_TABLE_MAX_LEN);
|
gBS->CopyMem (
|
(VOID *)PpttTable,
|
&mPpttHeader,
|
sizeof (EFI_ACPI_DESCRIPTION_HEADER)
|
);
|
PpttTableLengthRemain = PPTT_TABLE_MAX_LEN - sizeof (EFI_ACPI_DESCRIPTION_HEADER);
|
|
for (Index0 = 0; Index0 < EFI_ACPI_MAX_NUM_TABLES; Index0++) {
|
Status = mAcpiSdtProtocol->GetAcpiTable (
|
Index0,
|
&Table,
|
&TableVersion,
|
&TableKey
|
);
|
if (EFI_ERROR (Status)) {
|
break;
|
}
|
|
// Find APIC table
|
if (Table->Signature == EFI_ACPI_6_1_MULTIPLE_APIC_DESCRIPTION_TABLE_SIGNATURE) {
|
break;
|
}
|
|
}
|
|
if (!EFI_ERROR (Status) && (Index0 != EFI_ACPI_MAX_NUM_TABLES)) {
|
ApicTable = (ACPI_MADT_TABLE_HEADER *)Table;
|
|
GetApic (ApicTable, PpttTable, PpttTableLengthRemain);
|
|
Checksum = CalculateCheckSum8 ((UINT8 *)(PpttTable), PpttTable->Length);
|
PpttTable->Checksum = Checksum;
|
|
AcpiTableHandle = 0;
|
Status = mAcpiTableProtocol->InstallAcpiTable (
|
mAcpiTableProtocol,
|
PpttTable,
|
PpttTable->Length,
|
&AcpiTableHandle);
|
}
|
|
FreePool (PpttTable);
|
return ;
|
}
|
|
EFI_STATUS
|
EFIAPI
|
PpttEntryPoint(
|
IN EFI_HANDLE ImageHandle,
|
IN EFI_SYSTEM_TABLE *SystemTable
|
)
|
{
|
EFI_STATUS Status;
|
EFI_EVENT ReadyToBootEvent;
|
|
Status = gBS->LocateProtocol (
|
&gEfiAcpiTableProtocolGuid,
|
NULL,
|
(VOID **)&mAcpiTableProtocol);
|
ASSERT_EFI_ERROR (Status);
|
|
Status = gBS->LocateProtocol (
|
&gEfiAcpiSdtProtocolGuid,
|
NULL,
|
(VOID **)&mAcpiSdtProtocol);
|
ASSERT_EFI_ERROR (Status);
|
|
Status = EfiCreateEventReadyToBootEx (
|
TPL_NOTIFY,
|
PpttSetAcpiTable,
|
NULL,
|
&ReadyToBootEvent
|
);
|
ASSERT_EFI_ERROR (Status);
|
|
DEBUG ((DEBUG_INFO, "Acpi Pptt init done.\n"));
|
|
return Status;
|
}
|