soc/intel/xeon_sp/spr: Add ACPI support for Sapphire Rapids

Add ACPI support for Sapphire Rapids. Passes FWTS ACPI tests.
The code was written from scratch because there are Xeon-SP specific
implementation especially Integrated Input/Output (IIO).

Change-Id: Ic2a9be0222e122ae087b9cc8e1859d257e3411d6
Signed-off-by: Jonathan Zhang <jonzhang@meta.com>
Signed-off-by: Johnny Lin <johnny_lin@wiwynn.com>
Signed-off-by: Marc Jones <marcj303@gmail.com>
Reviewed-on: https://review.coreboot.org/c/coreboot/+/71967
Reviewed-by: Jonathan Zhang <jon.zhixiong.zhang@gmail.com>
Tested-by: build bot (Jenkins) <no-reply@coreboot.org>
Reviewed-by: Christian Walter <christian.walter@9elements.com>
Reviewed-by: Lean Sheng Tan <sheng.tan@9elements.com>
diff --git a/src/soc/intel/xeon_sp/spr/soc_acpi.c b/src/soc/intel/xeon_sp/spr/soc_acpi.c
new file mode 100644
index 0000000..d9d82ac
--- /dev/null
+++ b/src/soc/intel/xeon_sp/spr/soc_acpi.c
@@ -0,0 +1,591 @@
+/* SPDX-License-Identifier: GPL-2.0-only */
+
+#include <acpi/acpigen.h>
+#include <arch/smp/mpspec.h>
+#include <assert.h>
+#include <cbmem.h>
+#include <cpu/intel/turbo.h>
+#include <device/mmio.h>
+#include <device/pci.h>
+#include <intelblocks/acpi.h>
+#include <intelblocks/cpulib.h>
+#include <intelblocks/pmclib.h>
+#include <soc/acpi.h>
+#include <soc/iomap.h>
+#include <soc/msr.h>
+#include <soc/pci_devs.h>
+#include <soc/pm.h>
+#include <soc/soc_util.h>
+#include <soc/util.h>
+#include <hob_iiouds.h>
+
+int soc_madt_sci_irq_polarity(int sci)
+{
+	if (sci >= 20)
+		return MP_IRQ_POLARITY_LOW;
+	else
+		return MP_IRQ_POLARITY_HIGH;
+}
+
+uint32_t soc_read_sci_irq_select(void)
+{
+	/* PMC controller is hidden - hence PWRMBASE can't be accessbile using PCI cfg space */
+	uintptr_t pmc_bar = PCH_PWRM_BASE_ADDRESS;
+	return read32((void *)pmc_bar + PMC_ACPI_CNT);
+}
+
+void soc_fill_fadt(acpi_fadt_t *fadt)
+{
+	const uint16_t pmbase = ACPI_BASE_ADDRESS;
+
+	fadt->FADT_MinorVersion = 1;
+	fadt->pm_tmr_blk = pmbase + PM1_TMR;
+	fadt->pm_tmr_len = 4;
+	/* Clear flags set by common/block/acpi/acpi.c acpi_fill_fadt() */
+	fadt->flags &= ~ACPI_FADT_SEALED_CASE;
+
+	fadt->preferred_pm_profile = PM_ENTERPRISE_SERVER;
+	fadt->pm2_cnt_blk = pmbase + PM2_CNT;
+	fadt->pm2_cnt_len = 1;
+	fadt->gpe0_blk = pmbase + 0x60;
+	fadt->p_lvl2_lat = ACPI_FADT_C2_NOT_SUPPORTED;
+	fadt->p_lvl3_lat = ACPI_FADT_C3_NOT_SUPPORTED;
+	fadt->duty_offset = 1;
+
+	fadt->x_pm1b_evt_blk.space_id = ACPI_ADDRESS_SPACE_IO;
+	fadt->x_pm1b_cnt_blk.space_id = ACPI_ADDRESS_SPACE_IO;
+
+	/* PM2 Control Registers */
+	fadt->x_pm2_cnt_blk.space_id = ACPI_ADDRESS_SPACE_IO;
+	fadt->x_pm2_cnt_blk.bit_width = fadt->pm2_cnt_len * 8;
+	fadt->x_pm2_cnt_blk.bit_offset = 0;
+	fadt->x_pm2_cnt_blk.access_size = ACPI_ACCESS_SIZE_UNDEFINED;
+	fadt->x_pm2_cnt_blk.addrl = pmbase + PM2_CNT;
+	fadt->x_pm2_cnt_blk.addrh = 0x0;
+
+	/* PM1 Timer Register */
+	fadt->x_pm_tmr_blk.space_id = ACPI_ADDRESS_SPACE_IO;
+	fadt->x_pm_tmr_blk.bit_width = fadt->pm_tmr_len * 8;
+	fadt->x_pm_tmr_blk.bit_offset = 0;
+	fadt->x_pm_tmr_blk.access_size = ACPI_ACCESS_SIZE_DWORD_ACCESS;
+	fadt->x_pm_tmr_blk.addrl = pmbase + PM1_TMR;
+	fadt->x_pm_tmr_blk.addrh = 0x0;
+
+	fadt->x_gpe0_blk.space_id = ACPI_ADDRESS_SPACE_IO;
+	fadt->x_gpe0_blk.bit_width = fadt->gpe0_blk_len * 8;
+	fadt->x_gpe0_blk.bit_offset = 0;
+	fadt->x_gpe0_blk.access_size = ACPI_ACCESS_SIZE_BYTE_ACCESS;
+	fadt->x_gpe0_blk.addrl = fadt->gpe0_blk;
+	fadt->x_gpe0_blk.addrh = 0x0;
+
+	fadt->x_gpe1_blk.space_id = ACPI_ADDRESS_SPACE_IO;
+	fadt->x_gpe1_blk.bit_width = fadt->gpe1_blk_len * 8;
+	fadt->x_gpe1_blk.bit_offset = 0;
+	fadt->x_gpe1_blk.access_size = ACPI_ACCESS_SIZE_BYTE_ACCESS;
+	fadt->x_gpe1_blk.addrl = fadt->gpe1_blk;
+	fadt->x_gpe1_blk.addrh = 0x0;
+}
+
+static void create_dsdt_iou_pci_resource(uint8_t socket, uint8_t stack, const STACK_RES *ri)
+{
+	/*
+	   Stacks 0 (TYPE_UBOX_IIO)
+		Scope: PC<socket><stack>, ResourceTemplate: P0RS
+	   Stacks 1 .. 5 (TYPE_UBOX_IIO)
+		Scope: PC<socket><stack>, ResourceTemplate: RBRS
+	*/
+
+	/* Write ResourceTemplate resource name */
+	char tres[16];
+	snprintf(tres, sizeof(tres), "PT%d%X", socket, stack);
+	acpigen_write_name(tres);
+
+	printk(BIOS_DEBUG, "\tCreating ResourceTemplate %s for socket: %d, stack: %d\n", tres,
+	       socket, stack);
+
+	acpigen_write_resourcetemplate_header();
+
+	/*
+	 * ACPI spec requires that each bus device (stack) has a valid
+	 * resource template. Disabled devices still need a valid _CRS
+	 * although the values are ignored. The FWTS sanity check does
+	 * not pass when using the HOB stack resources to generate the
+	 * _CRS resource template for disabled stacks, so we provide a
+	 * zeroed resource template to satisfy both the spec and FWTS.
+	 */
+	const bool stack_enabled = ri->Personality < TYPE_RESERVED;
+
+	/* Bus Resource */
+	if (stack_enabled) {
+		/* For stack with CXL device, the PCIe bus resource is BusBase only. */
+		if (is_iio_cxl_stack_res(ri))
+			acpigen_resource_word(2, 0xc, 0, 0, ri->BusBase, ri->BusBase, 0x0, 1);
+		else
+			acpigen_resource_word(2, 0xc, 0, 0, ri->BusBase, ri->BusLimit, 0x0,
+					      (ri->BusLimit - ri->BusBase + 1));
+	} else {
+		acpigen_resource_word(2, 0, 0, 0, 0, 0, 0, 0);
+	}
+
+	/* Additional IO resources on socket 0 bus 0 */
+	if (socket == 0 && stack == 0) {
+		/* ACPI 6.4.2.5 I/O Port Descriptor */
+		acpigen_write_io16(0xCF8, 0xCFF, 0x1, 0x8, 1);
+
+		/* IO decode  CF8-CFF */
+		acpigen_resource_word(1, 0xc, 0x3, 0, 0x0000, 0x03AF, 0, 0x03B0);
+		acpigen_resource_word(1, 0xc, 0x3, 0, 0x03E0, 0x0CF7, 0, 0x0918);
+		acpigen_resource_word(1, 0xc, 0x3, 0, 0x03B0, 0x03BB, 0, 0x000C);
+		acpigen_resource_word(1, 0xc, 0x3, 0, 0x03C0, 0x03DF, 0, 0x0020);
+	}
+
+	/* IO resource */
+	if (stack_enabled) {
+		acpigen_resource_word(1, 0xc, 0x3, 0, ri->PciResourceIoBase,
+				      ri->PciResourceIoLimit, 0x0,
+				      (ri->PciResourceIoLimit - ri->PciResourceIoBase + 1));
+
+		/* Additional Mem32 resources on socket 0 bus 0 */
+		if (socket == 0 && stack == 0) {
+			acpigen_resource_dword(0, 0xc, 3, 0, VGA_BASE_ADDRESS,
+					       (VGA_BASE_ADDRESS + VGA_BASE_SIZE - 1), 0x0,
+					       VGA_BASE_SIZE);
+			acpigen_resource_dword(0, 0xc, 1, 0, SPI_BASE_ADDRESS,
+					       (SPI_BASE_ADDRESS + SPI_BASE_SIZE - 1), 0x0,
+					       SPI_BASE_SIZE);
+		}
+
+		/* Mem32 resource */
+		acpigen_resource_dword(
+			0, 0xc, 1, 0, ri->PciResourceMem32Base, ri->PciResourceMem32Limit, 0x0,
+			(ri->PciResourceMem32Limit - ri->PciResourceMem32Base + 1));
+
+		/* Mem64 resource */
+		acpigen_resource_qword(
+			0, 0xc, 1, 0, ri->PciResourceMem64Base, ri->PciResourceMem64Limit, 0x0,
+			(ri->PciResourceMem64Limit - ri->PciResourceMem64Base + 1));
+	} else {
+		/* Zeroed IO resource */
+		acpigen_resource_word(1, 0, 3, 0, 0, 0, 0, 0);
+
+		/* Zeroed Mem32 resource */
+		acpigen_resource_dword(0, 0, 1, 0, 0, 0, 0, 0);
+
+		/* Zeroed Mem64 resource */
+		acpigen_resource_dword(0, 0, 1, 0, 0, 0, 0, 0);
+	}
+
+	acpigen_write_resourcetemplate_footer();
+}
+
+static void create_dsdt_iou_cxl_resource(uint8_t socket, uint8_t stack, const STACK_RES *ri)
+{
+	/*
+	   Stacks 1 .. 5 (TYPE_UBOX_IIO)
+		Scope: CX<socket><stack>, ResourceTemplate: RBRS
+	*/
+	/* write ResourceTemplate resource name */
+	char tres[16];
+	snprintf(tres, sizeof(tres), "CT%d%X", socket, stack);
+	acpigen_write_name(tres);
+
+	printk(BIOS_DEBUG, "\tCreating ResourceTemplate %s for socket: %d, stack: %d\n", tres,
+	       socket, stack);
+
+	acpigen_write_resourcetemplate_header();
+
+	if (ri->Personality < TYPE_RESERVED) {
+		/* bus resource, from (BusBase + 1) to BusLimit */
+		acpigen_resource_word(2, 0xc, 0, 0, ri->BusBase + 1, ri->BusLimit, 0x0,
+				      (ri->BusLimit - ri->BusBase));
+
+		/* IO resource */
+		acpigen_resource_word(1, 0xc, 0x3, 0, ri->IoBase, ri->PciResourceIoBase - 1,
+				      0x0, ri->PciResourceIoBase - ri->IoBase);
+
+		/* Mem32 resource */
+		acpigen_resource_dword(0, 0xc, 1, 0, ri->Mmio32Base,
+				       ri->PciResourceMem32Base - 1, 0x0,
+				       ri->PciResourceMem32Base - ri->Mmio32Base);
+
+		/* Mem64 resource */
+		acpigen_resource_qword(0, 0xc, 1, 0, ri->Mmio64Base,
+				       ri->PciResourceMem64Base - 1, 0x0,
+				       ri->PciResourceMem64Base - ri->Mmio64Base);
+	} else {
+		/* bus resource, from (BusBase + 1) to BusLimit */
+		acpigen_resource_word(2, 0, 0, 0, 0, 0, 0, 0);
+
+		/* IO resource */
+		acpigen_resource_word(1, 0, 3, 0, 0, 0, 0, 0);
+
+		/* Mem32 resource */
+		acpigen_resource_dword(0, 0, 1, 0, 0, 0, 0, 0);
+
+		/* Mem64 resource */
+		acpigen_resource_qword(0, 0, 1, 0, 0, 0, 0, 0);
+	}
+
+	acpigen_write_resourcetemplate_footer();
+}
+
+static void create_dsdt_dino_resource(uint8_t socket, uint8_t stack, const STACK_RES *ri)
+{
+	/*
+	   Stacks 8 .. B (TYPE_DINO)
+		Scope: DI<socket><stack> for DINO, ResourceTemplate: DT
+		Scope: CP<socket><stack> for CPM (i.e., QAT), ResourceTemplate: MT
+		Scope: HQ<socket><stack> for HQM (i.e., DLB), ResourceTemplate: HT
+	*/
+
+	enum {
+		DSDT_DINO = 0,
+		DSDT_CPM,
+		DSDT_HQM,
+		DSDT_CPM1,
+		DSDT_HQM1
+	};
+	uint8_t rlist[] = {DSDT_DINO, DSDT_CPM, DSDT_HQM, DSDT_CPM1, DSDT_HQM1};
+
+	for (int i = 0; i < ARRAY_SIZE(rlist); ++i) {
+		uint8_t bus_base, bus_limit;
+		uint64_t mem64_base, mem64_limit;
+		char tres[16];
+
+		/* Note, This allocates the resources in a different order than
+		 * coreboot (DINO base is last). This causes the kernel to
+		 * reallocate the DINO BARs.
+		 * TODO: Use the resource settings from coreboot */
+		if (rlist[i] == DSDT_DINO) {
+			bus_base = ri->BusBase;
+			bus_limit = ri->BusBase;
+			mem64_base = ri->PciResourceMem64Base + CPM_MMIO_SIZE + HQM_MMIO_SIZE
+				     + CPM_MMIO_SIZE + HQM_MMIO_SIZE;
+			mem64_limit = ri->PciResourceMem64Limit;
+			snprintf(tres, sizeof(tres), "DT%d%X", socket, stack);
+		} else if (rlist[i] == DSDT_CPM) {
+			bus_base = ri->BusBase + CPM_BUS_OFFSET;
+			bus_limit = bus_base + CPM_RESERVED_BUS;
+			mem64_base = ri->PciResourceMem64Base;
+			mem64_limit = mem64_base + CPM_MMIO_SIZE - 1;
+			snprintf(tres, sizeof(tres), "MT%d%X", socket, stack);
+		} else if (rlist[i] == DSDT_HQM) {
+			bus_base = ri->BusBase + HQM_BUS_OFFSET;
+			bus_limit = bus_base + HQM_RESERVED_BUS;
+			mem64_base = ri->PciResourceMem64Base + CPM_MMIO_SIZE;
+			mem64_limit = mem64_base + HQM_MMIO_SIZE - 1;
+			snprintf(tres, sizeof(tres), "HT%d%X", socket, stack);
+		} else if (rlist[i] == DSDT_CPM1) {
+			bus_base = ri->BusBase + CPM1_BUS_OFFSET;
+			bus_limit = bus_base + CPM_RESERVED_BUS;
+			mem64_base = ri->PciResourceMem64Base + CPM_MMIO_SIZE + HQM_MMIO_SIZE;
+			mem64_limit = mem64_base + CPM_MMIO_SIZE - 1;
+			snprintf(tres, sizeof(tres), "MU%d%X", socket, stack);
+		} else { // DSDT_HQM1
+			bus_base = ri->BusBase + HQM1_BUS_OFFSET;
+			bus_limit = bus_base + HQM_RESERVED_BUS;
+			mem64_base = ri->PciResourceMem64Base + CPM_MMIO_SIZE + HQM_MMIO_SIZE
+				     + CPM_MMIO_SIZE;
+			mem64_limit = mem64_base + HQM_MMIO_SIZE - 1;
+			snprintf(tres, sizeof(tres), "HU%d%X", socket, stack);
+		}
+
+		printk(BIOS_DEBUG,
+		       "\tCreating Dino ResourceTemplate %s for socket: %d, "
+		       "stack: %d\n bus_base:0x%x, bus_limit:0x%x\n",
+		       tres, socket, stack, bus_base, bus_limit);
+
+		acpigen_write_name(tres);
+		acpigen_write_resourcetemplate_header();
+
+		if (ri->Personality < TYPE_RESERVED) {
+			acpigen_resource_word(2, 0xc, 0, 0, bus_base, bus_limit, 0x0,
+					      (bus_limit - bus_base + 1));
+
+			/* Mem32 resource */
+			if (rlist[i] == DSDT_DINO)
+				acpigen_resource_dword(0, 0xc, 1, 0, ri->PciResourceMem32Base,
+						       ri->PciResourceMem32Limit, 0x0,
+						       (ri->PciResourceMem32Limit
+							- ri->PciResourceMem32Base + 1));
+
+			/* Mem64 resource */
+			acpigen_resource_qword(0, 0xc, 1, 0, mem64_base, mem64_limit, 0,
+					       (mem64_limit - mem64_base + 1));
+		} else {
+			acpigen_resource_word(2, 0, 0, 0, 0, 0, 0, 0);
+
+			/* Mem32 resource */
+			if (rlist[i] == DSDT_DINO)
+				acpigen_resource_dword(0, 0, 1, 0, 0, 0, 0, 0);
+
+			/* Mem64 resource */
+			acpigen_resource_qword(0, 0, 1, 0, 0, 0, 0, 0);
+		}
+
+		acpigen_write_resourcetemplate_footer();
+	}
+}
+
+static void create_dsdt_ubox_resource(uint8_t socket, uint8_t stack, const STACK_RES *ri)
+{
+	/*
+	   Stacks D .. E (TYPE_UBOX)
+		Scope: UC/UD<socket><0..1> for UBOX[1-2], ResourceTemplate: UT/UU
+	*/
+
+	for (int i = 0; i < 2; ++i) {
+		char tres[16];
+		/* write ResourceTemplate resource name */
+		if (i == 0)
+			snprintf(tres, sizeof(tres), "UT%d%X", socket, stack);
+		else
+			snprintf(tres, sizeof(tres), "UU%d%X", socket, stack);
+
+		printk(BIOS_DEBUG, "\tCreating ResourceTemplate %s for socket: %d, stack: %d\n",
+		       tres, socket, stack);
+
+		acpigen_write_name(tres);
+		acpigen_write_resourcetemplate_header();
+
+		if (ri->Personality >= TYPE_RESERVED)
+			acpigen_resource_word(2, 0, 0, 0, 0, 0, 0, 0);
+		else if (i == 0)
+			acpigen_resource_word(2, 0xc, 0, 0, ri->BusBase, ri->BusBase, 0x0, 1);
+		else
+			acpigen_resource_word(2, 0xc, 0, 0, ri->BusBase + 1, ri->BusBase + 1,
+					      0x0, 1);
+
+		acpigen_write_resourcetemplate_footer();
+	}
+}
+
+
+/*
+ * Add a DSDT ACPI Name field for STACK enable setting.
+ *  This is retrieved by the device _STA defined in iiostack.asl
+ */
+static void create_dsdt_stack_sta(uint8_t socket, uint8_t stack, const STACK_RES *ri)
+{
+	char stack_sta[16];
+	snprintf(stack_sta, sizeof(stack_sta), "ST%d%X", socket, stack);
+
+	if (ri->Personality >= TYPE_RESERVED)
+		acpigen_write_name_integer(stack_sta, ACPI_STATUS_DEVICE_ALL_OFF);
+	else
+		acpigen_write_name_integer(stack_sta, ACPI_STATUS_DEVICE_ALL_ON);
+}
+
+void uncore_inject_dsdt(const struct device *device)
+{
+	/* Only add RTxx entries once. */
+	if (device->bus->secondary != 0)
+		return;
+
+	/*
+	   Write stack scope - this needs to match RP ACPI scopes.
+	   Stacks 0 (TYPE_UBOX_IIO)
+		Scope: PC<socket><stack>, ResourceTemplate: P0RS
+	   Stacks 1 .. 5 (TYPE_UBOX_IIO)
+		Scope: PC<socket><stack> & CX<socket><stack>, ResourceTemplate: RBRS
+	   Stacks 8 .. B (TYPE_DINO)
+		Scope: DI<socket><stack> for DINO, ResourceTemplate: RBRS
+		Scope: CP<socket><stack> for CPM (i.e., QAT), ResourceTemplate: RBRS
+		Scope: HQ<socket><stack> for HQM (i.e., DLB), ResourceTemplate: RBRS
+	   Stacks D .. E (TYPE_UBOX)
+		Scope: UC<socket><0..1> for UBOX[1-2], ResourceTemplate: UNRS
+	*/
+
+	printk(BIOS_DEBUG, "%s device: %s\n", __func__, dev_path(device));
+
+	acpigen_write_scope("\\_SB");
+
+	/* The _CSR generation must match SPR iiostack.asl. */
+	const IIO_UDS *hob = get_iio_uds();
+	/* TODO: DSDT uses CONFIG_MAX_SOCKET while PlatformData.numofIIO is the actual number
+	   of CPUs plugged in, although it doesn't cause serious errors when not all CPUs are
+	   plugged in, they should be in sync. */
+	for (uint8_t socket = 0; socket < hob->PlatformData.numofIIO; ++socket) {
+		for (int stack = 0; stack < MAX_LOGIC_IIO_STACK; ++stack) {
+			const STACK_RES *ri =
+				&hob->PlatformData.IIO_resource[socket].StackRes[stack];
+
+			printk(BIOS_DEBUG, "%s processing socket: %d, stack: %d, type: %d\n",
+			       __func__, socket, stack, ri->Personality);
+
+			if (stack <= IioStack5) { // TYPE_UBOX_IIO
+				create_dsdt_iou_pci_resource(socket, stack, ri);
+				create_dsdt_iou_cxl_resource(socket, stack, ri);
+				create_dsdt_stack_sta(socket, stack, ri);
+			} else if (stack >= IioStack8 && stack <= IioStack11) { // TYPE_DINO
+				create_dsdt_dino_resource(socket, stack, ri);
+				create_dsdt_stack_sta(socket, stack, ri);
+			} else if (stack == IioStack13) { // TYPE_UBOX
+				create_dsdt_ubox_resource(socket, stack, ri);
+				create_dsdt_stack_sta(socket, stack, ri);
+			}
+		}
+	}
+
+	acpigen_pop_len();
+}
+
+/* TODO: See if we can use the common generate_p_state_entries */
+void soc_power_states_generation(int core, int cores_per_package)
+{
+	int ratio_min, ratio_max, ratio_turbo, ratio_step;
+	int coord_type, power_max, power_unit, num_entries;
+	int ratio, power, clock, clock_max;
+	msr_t msr;
+
+	/* Determine P-state coordination type from MISC_PWR_MGMT[0] */
+	msr = rdmsr(MSR_MISC_PWR_MGMT);
+	if (msr.lo & MISC_PWR_MGMT_EIST_HW_DIS)
+		coord_type = SW_ANY;
+	else
+		coord_type = HW_ALL;
+
+	/* Get bus ratio limits and calculate clock speeds */
+	msr = rdmsr(MSR_PLATFORM_INFO);
+	ratio_min = (msr.hi >> (40 - 32)) & 0xff; /* Max Efficiency Ratio */
+
+	/* Determine if this CPU has configurable TDP */
+	if (cpu_config_tdp_levels()) {
+		/* Set max ratio to nominal TDP ratio */
+		msr = rdmsr(MSR_CONFIG_TDP_NOMINAL);
+		ratio_max = msr.lo & 0xff;
+	} else {
+		/* Max Non-Turbo Ratio */
+		ratio_max = (msr.lo >> 8) & 0xff;
+	}
+	clock_max = ratio_max * CONFIG_CPU_BCLK_MHZ;
+
+	/* Calculate CPU TDP in mW */
+	msr = rdmsr(MSR_PKG_POWER_SKU_UNIT);
+	power_unit = 2 << ((msr.lo & 0xf) - 1);
+	msr = rdmsr(MSR_PKG_POWER_SKU);
+	power_max = ((msr.lo & 0x7fff) / power_unit) * 1000;
+
+	/* Write _PCT indicating use of FFixedHW */
+	acpigen_write_empty_PCT();
+
+	/* Write _PPC with no limit on supported P-state */
+	acpigen_write_PPC_NVS();
+
+	/* Write PSD indicating configured coordination type */
+	acpigen_write_PSD_package(core, 1, coord_type);
+
+	/* Add P-state entries in _PSS table */
+	acpigen_write_name("_PSS");
+
+	/* Determine ratio points */
+	ratio_step = PSS_RATIO_STEP;
+	num_entries = ((ratio_max - ratio_min) / ratio_step) + 1;
+	if (num_entries > PSS_MAX_ENTRIES) {
+		ratio_step += 1;
+		num_entries = ((ratio_max - ratio_min) / ratio_step) + 1;
+	}
+
+	/* P[T] is Turbo state if enabled */
+	if (get_turbo_state() == TURBO_ENABLED) {
+		/* _PSS package count including Turbo */
+		acpigen_write_package(num_entries + 2);
+
+		msr = rdmsr(MSR_TURBO_RATIO_LIMIT);
+		ratio_turbo = msr.lo & 0xff;
+
+		/* Add entry for Turbo ratio */
+		acpigen_write_PSS_package(clock_max + 1,	  /* MHz */
+					  power_max,		  /* mW */
+					  PSS_LATENCY_TRANSITION, /* lat1 */
+					  PSS_LATENCY_BUSMASTER,  /* lat2 */
+					  ratio_turbo << 8,	  /* control */
+					  ratio_turbo << 8);	  /* status */
+	} else {
+		/* _PSS package count without Turbo */
+		acpigen_write_package(num_entries + 1);
+	}
+
+	/* First regular entry is max non-turbo ratio */
+	acpigen_write_PSS_package(clock_max,		  /* MHz */
+				  power_max,		  /* mW */
+				  PSS_LATENCY_TRANSITION, /* lat1 */
+				  PSS_LATENCY_BUSMASTER,  /* lat2 */
+				  ratio_max << 8,	  /* control */
+				  ratio_max << 8);	  /* status */
+
+	/* Generate the remaining entries */
+	for (ratio = ratio_min + ((num_entries - 1) * ratio_step); ratio >= ratio_min;
+	     ratio -= ratio_step) {
+
+		/* Calculate power at this ratio */
+		power = common_calculate_power_ratio(power_max, ratio_max, ratio);
+		clock = ratio * CONFIG_CPU_BCLK_MHZ;
+		// clock = 1;
+		acpigen_write_PSS_package(clock,		  /* MHz */
+					  power,		  /* mW */
+					  PSS_LATENCY_TRANSITION, /* lat1 */
+					  PSS_LATENCY_BUSMASTER,  /* lat2 */
+					  ratio << 8,		  /* control */
+					  ratio << 8);		  /* status */
+	}
+
+	/* Fix package length */
+	acpigen_pop_len();
+}
+
+unsigned long xeonsp_acpi_create_madt_lapics(unsigned long current)
+{
+	struct device *cpu;
+	uint8_t num_cpus = 0;
+
+	for (cpu = all_devices; cpu; cpu = cpu->next) {
+		if ((cpu->path.type != DEVICE_PATH_APIC)
+		    || (cpu->bus->dev->path.type != DEVICE_PATH_CPU_CLUSTER)) {
+			continue;
+		}
+		if (!cpu->enabled)
+			continue;
+		current += acpi_create_madt_lapic((acpi_madt_lapic_t *)current, num_cpus,
+						  cpu->path.apic.apic_id);
+		num_cpus++;
+	}
+
+	return current;
+}
+
+unsigned long acpi_fill_cedt(unsigned long current)
+{
+	const IIO_UDS *hob = get_iio_uds();
+	union uid {
+		uint32_t data;
+		struct {
+			uint8_t byte0;
+			uint8_t byte1;
+			uint8_t byte2;
+			uint8_t byte3;
+		};
+	} cxl_uid;
+	u32 cxl_ver;
+	u64 base;
+
+	cxl_uid.byte0 = 'C';
+	cxl_uid.byte1 = 'X';
+	/* Loop through all sockets and stacks, add CHBS for each CXL IIO stack */
+	for (uint8_t s = 0; s < hob->PlatformData.numofIIO; ++s) {
+		for (int x = 0; x < MAX_LOGIC_IIO_STACK; ++x) {
+			const STACK_RES *ri = &hob->PlatformData.IIO_resource[s].StackRes[x];
+			if (!is_iio_cxl_stack_res(ri))
+				continue;
+			/* uid needs to match with ACPI CXL device ID, eg. acpi/iiostack.asl */
+			cxl_uid.byte2 = s + '0';
+			cxl_uid.byte3 = x + '0';
+			cxl_ver = ACPI_CEDT_CHBS_CXL_VER_1_1;
+			base = ri->Mmio32Base; /* DP RCRB base */
+			current += acpi_create_cedt_chbs((acpi_cedt_chbs_t *)current,
+							 cxl_uid.data, cxl_ver, base);
+		}
+	}
+
+	return current;
+}