[Patch v4 9/9] soc: tegra: cbb: Add support for tegra-grace SOC
From: Sumit Gupta
Date: Thu May 05 2022 - 13:08:07 EST
Adding support for Tegra Grace SOC which uses CBB2.0 architecture
based fabrics. Also, adding ACPI support required for Grace.
Fabrics reporting errors in Grace are "CBB Fabric" and "BPMP Fabric".
"CBB Fabric" connects various other CBB2.0 based fabrics and also
services the Initiators and Targets which are connected to itself.
"BPMP Fabric" is present in BPMP cluster.
Signed-off-by: Sumit Gupta <sumitg@xxxxxxxxxx>
---
drivers/soc/tegra/cbb/tegra234-cbb.c | 90 ++++++++++-
include/soc/tegra/tegra-grace-cbb.h | 219 +++++++++++++++++++++++++++
2 files changed, 308 insertions(+), 1 deletion(-)
create mode 100644 include/soc/tegra/tegra-grace-cbb.h
diff --git a/drivers/soc/tegra/cbb/tegra234-cbb.c b/drivers/soc/tegra/cbb/tegra234-cbb.c
index 782d28e50ca1..1889331ddb22 100644
--- a/drivers/soc/tegra/cbb/tegra234-cbb.c
+++ b/drivers/soc/tegra/cbb/tegra234-cbb.c
@@ -11,6 +11,7 @@
*/
#include <asm/cpufeature.h>
+#include <linux/acpi.h>
#include <linux/clk.h>
#include <linux/debugfs.h>
#include <linux/module.h>
@@ -27,6 +28,7 @@
#include <soc/tegra/fuse.h>
#include <soc/tegra/tegra-cbb.h>
#include <soc/tegra/tegra234-cbb.h>
+#include <soc/tegra/tegra-grace-cbb.h>
static LIST_HEAD(cbb_errmon_list);
static DEFINE_SPINLOCK(cbb_errmon_lock);
@@ -258,14 +260,33 @@ static void print_errlog_err(struct seq_file *file, struct tegra_cbb_errmon_reco
u8 cache_type = 0, prot_type = 0, burst_length = 0;
u8 mstr_id = 0, grpsec = 0, vqc = 0, falconsec = 0;
u8 beat_size = 0, access_type = 0, access_id = 0;
+ u8 requester_socket_id = 0, local_socket_id = 0;
u8 slave_id = 0, fab_id = 0, burst_type = 0;
char fabric_name[20];
+ bool is_numa = 0;
+
+ if (num_possible_nodes() > 1)
+ is_numa = true;
mstr_id = FIELD_GET(FAB_EM_EL_MSTRID, errmon->mn_user_bits);
vqc = FIELD_GET(FAB_EM_EL_VQC, errmon->mn_user_bits);
grpsec = FIELD_GET(FAB_EM_EL_GRPSEC, errmon->mn_user_bits);
falconsec = FIELD_GET(FAB_EM_EL_FALCONSEC, errmon->mn_user_bits);
+ /*
+ * For SOC with multiple NUMA nodes, print cross socket access
+ * errors only if initiator/master_id is CCPLEX, CPMU or GPU.
+ */
+ if (is_numa) {
+ local_socket_id = numa_node_id();
+ requester_socket_id = FIELD_GET(REQ_SOCKET_ID, errmon->mn_attr2);
+
+ if (requester_socket_id != local_socket_id) {
+ if ((mstr_id != 0x1) && (mstr_id != 0x2) && (mstr_id != 0xB))
+ return;
+ }
+ }
+
fab_id = FIELD_GET(FAB_EM_EL_FABID, errmon->mn_attr2);
slave_id = FIELD_GET(FAB_EM_EL_SLAVEID, errmon->mn_attr2);
@@ -298,6 +319,15 @@ static void print_errlog_err(struct seq_file *file, struct tegra_cbb_errmon_reco
else
strcpy(fabric_name, errmon->name);
+ if (is_numa) {
+ tegra_cbb_print_err(file, "\t Requester_Socket_Id\t: 0x%x\n",
+ requester_socket_id);
+ tegra_cbb_print_err(file, "\t Local_Socket_Id\t: 0x%x\n",
+ local_socket_id);
+ tegra_cbb_print_err(file, "\t No. of NUMA_NODES\t: 0x%x\n",
+ num_possible_nodes());
+ }
+
tegra_cbb_print_err(file, "\t Fabric\t\t: %s\n", fabric_name);
tegra_cbb_print_err(file, "\t Slave_Id\t\t: 0x%x\n", slave_id);
tegra_cbb_print_err(file, "\t Burst_length\t\t: 0x%x\n", burst_length);
@@ -585,6 +615,23 @@ static struct tegra_cbb_fabric_data tegra234_sce_fab_data = {
.err_notifier_base = 0x19000
};
+static struct tegra_cbb_fabric_data tegra_grace_cbb_fab_data = {
+ .name = "cbb-fabric",
+ .tegra_cbb_master_id = tegra_grace_master_id,
+ .sn_addr_map = tegra_grace_cbb_sn_lookup,
+ .noc_errors = tegra_grace_errmon_errors,
+ .err_notifier_base = 0x60000,
+ .off_mask_erd = 0x40004
+};
+
+static struct tegra_cbb_fabric_data tegra_grace_bpmp_fab_data = {
+ .name = "bpmp-fabric",
+ .tegra_cbb_master_id = tegra_grace_master_id,
+ .sn_addr_map = tegra_grace_bpmp_sn_lookup,
+ .noc_errors = tegra_grace_errmon_errors,
+ .err_notifier_base = 0x19000
+};
+
static const struct of_device_id tegra234_cbb_dt_ids[] = {
{.compatible = "nvidia,tegra234-cbb-fabric",
.data = &tegra234_cbb_fab_data},
@@ -602,6 +649,36 @@ static const struct of_device_id tegra234_cbb_dt_ids[] = {
};
MODULE_DEVICE_TABLE(of, tegra234_cbb_dt_ids);
+struct cbb_acpi_uid_noc {
+ const char *hid;
+ const char *uid;
+ const struct tegra_cbb_fabric_data *fab;
+};
+
+static const struct cbb_acpi_uid_noc cbb_acpi_uids[] = {
+ { "NVDA1070", "1", &tegra_grace_cbb_fab_data },
+ { "NVDA1070", "2", &tegra_grace_bpmp_fab_data },
+ { },
+};
+
+static const struct acpi_device_id tegra_grace_cbb_acpi_ids[] = {
+ { "NVDA1070" },
+ { },
+};
+MODULE_DEVICE_TABLE(acpi, tegra_grace_cbb_acpi_ids);
+
+static const struct
+tegra_cbb_fabric_data *cbb_acpi_get_fab_data(struct acpi_device *adev)
+{
+ const struct cbb_acpi_uid_noc *u;
+
+ for (u = cbb_acpi_uids; u->hid; u++) {
+ if (acpi_dev_hid_uid_match(adev, u->hid, u->uid))
+ return u->fab;
+ }
+ return NULL;
+}
+
static int
tegra234_cbb_errmon_init(const struct tegra_cbb_fabric_data *pdata,
struct tegra_cbb *cbb, struct resource *res_base)
@@ -646,10 +723,20 @@ static int tegra234_cbb_probe(struct platform_device *pdev)
struct tegra_cbb_errmon_record *errmon = NULL;
const struct tegra_cbb_fabric_data *pdata = NULL;
struct resource *res_base = NULL;
+ struct device *dev = &pdev->dev;
+ struct acpi_device *device;
struct tegra_cbb *cbb;
int err = 0;
- pdata = of_device_get_match_data(&pdev->dev);
+ if (of_machine_is_compatible("nvidia,tegra23x") ||
+ of_machine_is_compatible("nvidia,tegra234")) {
+ pdata = of_device_get_match_data(&pdev->dev);
+ } else {
+ device = ACPI_COMPANION(dev);
+ if (!device)
+ return -ENODEV;
+ pdata = cbb_acpi_get_fab_data(device);
+ }
if (!pdata) {
dev_err(&pdev->dev, "No device match found\n");
return -EINVAL;
@@ -720,6 +807,7 @@ static struct platform_driver tegra234_cbb_driver = {
.owner = THIS_MODULE,
.name = "tegra234-cbb",
.of_match_table = of_match_ptr(tegra234_cbb_dt_ids),
+ .acpi_match_table = ACPI_PTR(tegra_grace_cbb_acpi_ids),
#ifdef CONFIG_PM_SLEEP
.pm = &tegra234_cbb_pm,
#endif
diff --git a/include/soc/tegra/tegra-grace-cbb.h b/include/soc/tegra/tegra-grace-cbb.h
new file mode 100644
index 000000000000..91cde80e773a
--- /dev/null
+++ b/include/soc/tegra/tegra-grace-cbb.h
@@ -0,0 +1,219 @@
+/* SPDX-License-Identifier: GPL-2.0 */
+/*
+ * Copyright (c) 2021, NVIDIA CORPORATION. All rights reserved
+ */
+
+static char *tegra_grace_master_id[] = {
+ "TZ", /* 0x0 */
+ "CCPLEX", /* 0x1 */
+ "CCPMU", /* 0x2 */
+ "BPMP_FW", /* 0x3 */
+ "PSC_FW_USER", /* 0x4 */
+ "PSC_FW_SUPERVISOR", /* 0x5 */
+ "PSC_FW_MACHINE", /* 0x6 */
+ "PSC_BOOT", /* 0x7 */
+ "BPMP_BOOT", /* 0x8 */
+ "JTAGM_DFT", /* 0x9 */
+ "CORESIGHT", /* 0xA */
+ "GPU", /* 0xB */
+ "PEATRANS", /* 0xC */
+ "RSVD" /* 0x3F */
+};
+
+/*
+ * Possible causes for Slave and Timeout errors.
+ * SLAVE_ERR:
+ * Slave being accessed responded with an error. Slave could return
+ * an error for various cases :
+ * Unsupported access, clamp setting when power gated, register
+ * level firewall(SCR), address hole within the slave, etc
+ *
+ * TIMEOUT_ERR:
+ * No response returned by slave. Can be due to slave being clock
+ * gated, under reset, powered down or slave inability to respond
+ * for an internal slave issue
+ */
+
+static struct tegra_noc_errors tegra_grace_errmon_errors[] = {
+ {.errcode = "SLAVE_ERR",
+ .type = "Slave being accessed responded with an error."
+ },
+ {.errcode = "DECODE_ERR",
+ .type = "Attempt to access an address hole or Reserved region of memory."
+ },
+ {.errcode = "FIREWALL_ERR",
+ .type = "Attempt to access a region which is firewalled."
+ },
+ {.errcode = "TIMEOUT_ERR",
+ .type = "No response returned by slave."
+ },
+ {.errcode = "PWRDOWN_ERR",
+ .type = "Attempt to access a portion of the fabric that is powered down."
+ },
+ {.errcode = "UNSUPPORTED_ERR",
+ .type = "Attempt to access a slave through an unsupported access."
+ },
+ {.errcode = "POISON_ERR",
+ .type = "Slave responds with poison error to indicate error in data."
+ },
+ {.errcode = "RSVD"},
+ {.errcode = "RSVD"},
+ {.errcode = "RSVD"},
+ {.errcode = "RSVD"},
+ {.errcode = "RSVD"},
+ {.errcode = "RSVD"},
+ {.errcode = "RSVD"},
+ {.errcode = "RSVD"},
+ {.errcode = "RSVD"},
+ {.errcode = "NO_SUCH_ADDRESS_ERR",
+ .type = "The address belongs to the pri_target range but there is no register implemented at the address."
+ },
+ {.errcode = "TASK_ERR",
+ .type = "Attempt to update a PRI task when the current task has still not completed."
+ },
+ {.errcode = "EXTERNAL_ERR",
+ .type = "Indicates that an external PRI register access met with an error due to any issue in the unit."
+ },
+ {.errcode = "INDEX_ERR",
+ .type = "Applicable to PRI index aperture pair, when the programmed index is outside the range defined in the manual."
+ },
+ {.errcode = "RESET_ERR",
+ .type = "Target in Reset Error: Attempt to access a SubPri or external PRI register but they are in reset."
+ },
+ {.errcode = "REGISTER_RST_ERR",
+ .type = "Attempt to access a PRI register but the register is partial or completely in reset."
+ },
+ {.errcode = "POWER_GATED_ERR",
+ .type = "Returned by external PRI client when the external access goes to a power gated domain."
+ },
+ {.errcode = "SUBPRI_FS_ERR",
+ .type = "Subpri is floorswept: Attempt to access a subpri through the main pri target but subPri logic is floorswept."
+ },
+ {.errcode = "SUBPRI_CLK_OFF_ERR",
+ .type = "Subpri clock is off: Attempt to access a subpri through the main pri target but subPris clock is gated/off."
+ }
+};
+
+#define REQ_SOCKET_ID GENMASK(27, 24)
+
+#define BPMP_SN_AXI2APB_1_BASE 0x00000
+#define BPMP_SN_CBB_T_BASE 0x15000
+#define BPMP_SN_CPU_T_BASE 0x16000
+#define BPMP_SN_DBB0_T_BASE 0x17000
+#define BPMP_SN_DBB1_T_BASE 0x18000
+
+#define CBB_SN_CCPLEX_SLAVE_BASE 0x50000
+#define CBB_SN_PCIE_C8_BASE 0x51000
+#define CBB_SN_PCIE_C9_BASE 0x52000
+#define CBB_SN_PCIE_C4_BASE 0x53000
+#define CBB_SN_PCIE_C5_BASE 0x54000
+#define CBB_SN_PCIE_C6_BASE 0x55000
+#define CBB_SN_PCIE_C7_BASE 0x56000
+#define CBB_SN_PCIE_C2_BASE 0x57000
+#define CBB_SN_PCIE_C3_BASE 0x58000
+#define CBB_SN_PCIE_C0_BASE 0x59000
+#define CBB_SN_PCIE_C1_BASE 0x5A000
+#define CBB_SN_AON_SLAVE_BASE 0x5B000
+#define CBB_SN_BPMP_SLAVE_BASE 0x5C000
+#define CBB_SN_PSC_SLAVE_BASE 0x5D000
+#define CBB_SN_STM_BASE 0x5E000
+#define CBB_SN_AXI2APB_1_BASE 0x70000
+#define CBB_SN_AXI2APB_10_BASE 0x71000
+#define CBB_SN_AXI2APB_11_BASE 0x72000
+#define CBB_SN_AXI2APB_12_BASE 0x73000
+#define CBB_SN_AXI2APB_13_BASE 0x74000
+#define CBB_SN_AXI2APB_14_BASE 0x75000
+#define CBB_SN_AXI2APB_15_BASE 0x76000
+#define CBB_SN_AXI2APB_16_BASE 0x77000
+#define CBB_SN_AXI2APB_17_BASE 0x78000
+#define CBB_SN_AXI2APB_18_BASE 0x79000
+#define CBB_SN_AXI2APB_19_BASE 0x7A000
+#define CBB_SN_AXI2APB_2_BASE 0x7B000
+#define CBB_SN_AXI2APB_20_BASE 0x7C000
+#define CBB_SN_AXI2APB_21_BASE 0x7D000
+#define CBB_SN_AXI2APB_22_BASE 0x7E000
+#define CBB_SN_AXI2APB_23_BASE 0x7F000
+#define CBB_SN_AXI2APB_24_BASE 0x80000
+#define CBB_SN_AXI2APB_25_BASE 0x81000
+#define CBB_SN_AXI2APB_26_BASE 0x82000
+#define CBB_SN_AXI2APB_27_BASE 0x83000
+#define CBB_SN_AXI2APB_28_BASE 0x84000
+#define CBB_SN_AXI2APB_29_BASE 0x85000
+#define CBB_SN_AXI2APB_30_BASE 0x86000
+#define CBB_SN_AXI2APB_4_BASE 0x87000
+#define CBB_SN_AXI2APB_5_BASE 0x88000
+#define CBB_SN_AXI2APB_6_BASE 0x89000
+#define CBB_SN_AXI2APB_7_BASE 0x8A000
+#define CBB_SN_AXI2APB_8_BASE 0x8B000
+#define CBB_SN_AXI2APB_9_BASE 0x8C000
+#define CBB_SN_AXI2APB_3_BASE 0x8D000
+
+#define SLAVE_LOOKUP(sn) #sn, sn
+
+static struct tegra_sn_addr_map tegra_grace_bpmp_sn_lookup[] = {
+ { "RSVD", 0 },
+ { "RSVD", 0 },
+ { SLAVE_LOOKUP(BPMP_SN_CBB_T_BASE) },
+ { SLAVE_LOOKUP(BPMP_SN_CPU_T_BASE) },
+ { SLAVE_LOOKUP(BPMP_SN_AXI2APB_1_BASE) },
+ { SLAVE_LOOKUP(BPMP_SN_DBB0_T_BASE) },
+ { SLAVE_LOOKUP(BPMP_SN_DBB1_T_BASE) }
+};
+
+static struct tegra_sn_addr_map tegra_grace_cbb_sn_lookup[] = {
+ { SLAVE_LOOKUP(CBB_SN_PCIE_C8_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_PCIE_C9_BASE) },
+ { "RSVD", 0 },
+ { "RSVD", 0 },
+ { "RSVD", 0 },
+ { "RSVD", 0 },
+ { "RSVD", 0 },
+ { "RSVD", 0 },
+ { "RSVD", 0 },
+ { "RSVD", 0 },
+ { SLAVE_LOOKUP(CBB_SN_AON_SLAVE_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_BPMP_SLAVE_BASE) },
+ { "RSVD", 0 },
+ { "RSVD", 0 },
+ { SLAVE_LOOKUP(CBB_SN_PSC_SLAVE_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_STM_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_1_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_10_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_11_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_12_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_13_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_14_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_15_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_16_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_17_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_18_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_19_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_2_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_20_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_4_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_5_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_6_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_7_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_8_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_9_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_3_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_21_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_22_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_23_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_24_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_25_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_26_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_27_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_28_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_PCIE_C4_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_PCIE_C5_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_PCIE_C6_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_PCIE_C7_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_PCIE_C2_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_PCIE_C3_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_PCIE_C0_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_PCIE_C1_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_CCPLEX_SLAVE_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_29_BASE) },
+ { SLAVE_LOOKUP(CBB_SN_AXI2APB_30_BASE) }
+};
--
2.17.1