TEST=Build and boot on intel/archercity CRB Change-Id: I84f07c16e24e441a885144df8c805f1310acae29 Signed-off-by: Shuo Liu <shuo.liu@intel.com> Co-authored-by: Ziang Wang <ziang.wang@intel.com> Co-authored-by: Gang Chen <gang.c.chen@intel.com> Reviewed-on: https://review.coreboot.org/c/coreboot/+/81439 Reviewed-by: Angel Pons <th3fanbus@gmail.com> Tested-by: build bot (Jenkins) <no-reply@coreboot.org>
120 lines
3.9 KiB
C
120 lines
3.9 KiB
C
/* SPDX-License-Identifier: GPL-2.0-only */
|
|
|
|
#include <console/console.h>
|
|
#include <device/device.h>
|
|
#include <device/pci_ops.h>
|
|
#include <device/pci.h>
|
|
#include <device/pciexp.h>
|
|
#include <soc/numa.h>
|
|
#include <soc/soc_util.h>
|
|
#include <soc/util.h>
|
|
#include <types.h>
|
|
|
|
void dump_pds(void)
|
|
{
|
|
printk(BIOS_DEBUG, "====== Proximity Domain Dump ======\n");
|
|
printk(BIOS_DEBUG, "number of proximity domains: %d\n", pds.num_pds);
|
|
for (uint8_t i = 0; i < pds.num_pds; i++) {
|
|
printk(BIOS_DEBUG, "\tproximity domain %d:\n", i);
|
|
printk(BIOS_DEBUG, "\t\ttype:%d\n", pds.pds[i].pd_type);
|
|
printk(BIOS_DEBUG, "\t\tsocket_bitmap:0x%x\n", pds.pds[i].socket_bitmap);
|
|
printk(BIOS_DEBUG, "\t\tdevice:%s\n", pds.pds[i].dev ? dev_path(pds.pds[i].dev) : "");
|
|
printk(BIOS_DEBUG, "\t\tbase(64MB):0x%x\n", pds.pds[i].base);
|
|
printk(BIOS_DEBUG, "\t\tsize(64MB):0x%x\n", pds.pds[i].size);
|
|
}
|
|
}
|
|
|
|
void fill_pds(void)
|
|
{
|
|
uint8_t num_sockets = soc_get_num_cpus();
|
|
uint8_t num_cxlnodes = get_cxl_node_count();
|
|
const IIO_UDS *hob = get_iio_uds();
|
|
|
|
/*
|
|
* Rules/assumptions:
|
|
* 1. Each processor has a processor proximity domain regardless whether
|
|
* a processor has DIMM attached to it or not.
|
|
* 2. All system memory map elements are either from processor attached memory,
|
|
* or from CXL memory. Each CXL node info entry has a corresponding entry
|
|
* in system memory map elements.
|
|
* 3. Each CXL device may have multiple HDMs (Host-managed Device Memory). Each
|
|
* HDM has one and only one CXL node info entry. Each CXL node info entry
|
|
* represents a generic initiator proximity domain.
|
|
*/
|
|
pds.num_pds = num_cxlnodes + num_sockets;
|
|
pds.pds = xmalloc(sizeof(struct proximity_domain) * pds.num_pds);
|
|
if (!pds.pds)
|
|
die("%s %d out of memory.", __FILE__, __LINE__);
|
|
|
|
memset(pds.pds, 0, sizeof(struct proximity_domain) * pds.num_pds);
|
|
|
|
/* Fill in processor domains */
|
|
uint8_t i, j, socket;
|
|
for (socket = 0, i = 0; i < num_sockets; socket++) {
|
|
if (!soc_cpu_is_enabled(socket))
|
|
continue;
|
|
pds.pds[i].pd_type = PD_TYPE_PROCESSOR;
|
|
pds.pds[i].socket_bitmap = 1 << hob->PlatformData.IIO_resource[socket].SocketID;
|
|
pds.pds[i].distances = malloc(sizeof(uint8_t) * pds.num_pds);
|
|
if (!pds.pds[i].distances)
|
|
die("%s %d out of memory.", __FILE__, __LINE__);
|
|
/* hard code the distances for now, till we know how to calculate them. */
|
|
for (j = 0; j < pds.num_pds; j++) {
|
|
if (j == i)
|
|
pds.pds[i].distances[j] = 0x0a;
|
|
else
|
|
pds.pds[i].distances[j] = 0x0e;
|
|
}
|
|
i++;
|
|
}
|
|
|
|
/* If there are no CXL nodes, we are done */
|
|
if (num_cxlnodes == 0)
|
|
return;
|
|
|
|
#if CONFIG(SOC_INTEL_HAS_CXL)
|
|
/* There are CXL nodes, fill in generic initiator domain after the processors pds */
|
|
uint8_t skt_id, cxl_id;
|
|
const CXL_NODE_SOCKET *cxl_hob = get_cxl_node();
|
|
for (skt_id = 0, i = num_sockets; skt_id < MAX_SOCKET; skt_id++, i++) {
|
|
for (cxl_id = 0; cxl_id < cxl_hob[skt_id].CxlNodeCount; ++cxl_id) {
|
|
const CXL_NODE_INFO node = cxl_hob[skt_id].CxlNodeInfo[cxl_id];
|
|
pds.pds[i].pd_type = PD_TYPE_GENERIC_INITIATOR;
|
|
pds.pds[i].socket_bitmap = node.SocketBitmap;
|
|
pds.pds[i].base = node.Address;
|
|
pds.pds[i].size = node.Size;
|
|
struct device *dev = pcie_find_dsn(node.SerialNumber, node.VendorId, 0);
|
|
pds.pds[i].dev = dev;
|
|
pds.pds[i].distances = malloc(sizeof(uint8_t) * pds.num_pds);
|
|
if (!pds.pds[i].distances)
|
|
die("%s %d out of memory.", __FILE__, __LINE__);
|
|
/* hard code the distances until we know how to calculate them */
|
|
for (j = 0; j < pds.num_pds; j++) {
|
|
if (j == i)
|
|
pds.pds[i].distances[j] = 0x0a;
|
|
else
|
|
pds.pds[i].distances[j] = 0x0e;
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
|
|
/*
|
|
* Return the total size of memory regions in generic initiator affinity domains.
|
|
* The size is in unit of 64MB.
|
|
*/
|
|
uint32_t get_generic_initiator_mem_size(void)
|
|
{
|
|
uint8_t i;
|
|
uint32_t size = 0;
|
|
|
|
for (i = 0; i < pds.num_pds; i++) {
|
|
if (pds.pds[i].pd_type == PD_TYPE_PROCESSOR)
|
|
continue;
|
|
size += pds.pds[i].size;
|
|
}
|
|
|
|
return size;
|
|
}
|