Commit 2927cf85 authored by Sumit Gupta's avatar Sumit Gupta Committed by Thierry Reding

soc/tegra: cbb: Check firewall before enabling error reporting

To enable error reporting for a fabric to CCPLEX, we need to write its
register for enabling error interrupt to CCPLEX during boot and later
clear the error status register after error occurs. If a fabric's
registers are protected and not accessible from CCPLEX, then accessing
the registers will cause CBB firewall error.

Add support to check whether write access from CCPLEX to the registers
of a fabric is not blocked by it's firewall before enabling error
reporting to CCPLEX for that fabric.

Fixes: fc2f151d ("soc/tegra: cbb: Add driver for Tegra234 CBB 2.0")
Signed-off-by: default avatarSumit Gupta <sumitg@nvidia.com>
Signed-off-by: default avatarThierry Reding <treding@nvidia.com>
parent 55084947
...@@ -72,6 +72,11 @@ ...@@ -72,6 +72,11 @@
#define REQ_SOCKET_ID GENMASK(27, 24) #define REQ_SOCKET_ID GENMASK(27, 24)
#define CCPLEX_MSTRID 0x1
#define FIREWALL_APERTURE_SZ 0x10000
/* Write firewall check enable */
#define WEN 0x20000
enum tegra234_cbb_fabric_ids { enum tegra234_cbb_fabric_ids {
CBB_FAB_ID, CBB_FAB_ID,
SCE_FAB_ID, SCE_FAB_ID,
...@@ -92,6 +97,9 @@ struct tegra234_slave_lookup { ...@@ -92,6 +97,9 @@ struct tegra234_slave_lookup {
struct tegra234_cbb_fabric { struct tegra234_cbb_fabric {
const char *name; const char *name;
phys_addr_t off_mask_erd; phys_addr_t off_mask_erd;
phys_addr_t firewall_base;
unsigned int firewall_ctl;
unsigned int firewall_wr_ctl;
const char * const *master_id; const char * const *master_id;
unsigned int notifier_offset; unsigned int notifier_offset;
const struct tegra_cbb_error *errors; const struct tegra_cbb_error *errors;
...@@ -129,6 +137,44 @@ static inline struct tegra234_cbb *to_tegra234_cbb(struct tegra_cbb *cbb) ...@@ -129,6 +137,44 @@ static inline struct tegra234_cbb *to_tegra234_cbb(struct tegra_cbb *cbb)
static LIST_HEAD(cbb_list); static LIST_HEAD(cbb_list);
static DEFINE_SPINLOCK(cbb_lock); static DEFINE_SPINLOCK(cbb_lock);
static bool
tegra234_cbb_write_access_allowed(struct platform_device *pdev, struct tegra234_cbb *cbb)
{
u32 val;
if (!cbb->fabric->firewall_base ||
!cbb->fabric->firewall_ctl ||
!cbb->fabric->firewall_wr_ctl) {
dev_info(&pdev->dev, "SoC data missing for firewall\n");
return false;
}
if ((cbb->fabric->firewall_ctl > FIREWALL_APERTURE_SZ) ||
(cbb->fabric->firewall_wr_ctl > FIREWALL_APERTURE_SZ)) {
dev_err(&pdev->dev, "wrong firewall offset value\n");
return false;
}
val = readl(cbb->regs + cbb->fabric->firewall_base + cbb->fabric->firewall_ctl);
/*
* If the firewall check feature for allowing or blocking the
* write accesses through the firewall of a fabric is disabled
* then CCPLEX can write to the registers of that fabric.
*/
if (!(val & WEN))
return true;
/*
* If the firewall check is enabled then check whether CCPLEX
* has write access to the fabric's error notifier registers
*/
val = readl(cbb->regs + cbb->fabric->firewall_base + cbb->fabric->firewall_wr_ctl);
if (val & (BIT(CCPLEX_MSTRID)))
return true;
return false;
}
static void tegra234_cbb_fault_enable(struct tegra_cbb *cbb) static void tegra234_cbb_fault_enable(struct tegra_cbb *cbb)
{ {
struct tegra234_cbb *priv = to_tegra234_cbb(cbb); struct tegra234_cbb *priv = to_tegra234_cbb(cbb);
...@@ -551,7 +597,7 @@ static irqreturn_t tegra234_cbb_isr(int irq, void *data) ...@@ -551,7 +597,7 @@ static irqreturn_t tegra234_cbb_isr(int irq, void *data)
*/ */
if (priv->fabric->off_mask_erd) { if (priv->fabric->off_mask_erd) {
mstr_id = FIELD_GET(USRBITS_MSTR_ID, priv->mn_user_bits); mstr_id = FIELD_GET(USRBITS_MSTR_ID, priv->mn_user_bits);
if (mstr_id == 0x1) if (mstr_id == CCPLEX_MSTRID)
is_inband_err = 1; is_inband_err = 1;
} }
} }
...@@ -665,6 +711,9 @@ static const struct tegra234_cbb_fabric tegra234_aon_fabric = { ...@@ -665,6 +711,9 @@ static const struct tegra234_cbb_fabric tegra234_aon_fabric = {
.errors = tegra234_cbb_errors, .errors = tegra234_cbb_errors,
.max_errors = ARRAY_SIZE(tegra234_cbb_errors), .max_errors = ARRAY_SIZE(tegra234_cbb_errors),
.notifier_offset = 0x17000, .notifier_offset = 0x17000,
.firewall_base = 0x30000,
.firewall_ctl = 0x8d0,
.firewall_wr_ctl = 0x8c8,
}; };
static const struct tegra234_slave_lookup tegra234_bpmp_slave_map[] = { static const struct tegra234_slave_lookup tegra234_bpmp_slave_map[] = {
...@@ -683,6 +732,9 @@ static const struct tegra234_cbb_fabric tegra234_bpmp_fabric = { ...@@ -683,6 +732,9 @@ static const struct tegra234_cbb_fabric tegra234_bpmp_fabric = {
.errors = tegra234_cbb_errors, .errors = tegra234_cbb_errors,
.max_errors = ARRAY_SIZE(tegra234_cbb_errors), .max_errors = ARRAY_SIZE(tegra234_cbb_errors),
.notifier_offset = 0x19000, .notifier_offset = 0x19000,
.firewall_base = 0x30000,
.firewall_ctl = 0x8f0,
.firewall_wr_ctl = 0x8e8,
}; };
static const struct tegra234_slave_lookup tegra234_cbb_slave_map[] = { static const struct tegra234_slave_lookup tegra234_cbb_slave_map[] = {
...@@ -757,7 +809,10 @@ static const struct tegra234_cbb_fabric tegra234_cbb_fabric = { ...@@ -757,7 +809,10 @@ static const struct tegra234_cbb_fabric tegra234_cbb_fabric = {
.errors = tegra234_cbb_errors, .errors = tegra234_cbb_errors,
.max_errors = ARRAY_SIZE(tegra234_cbb_errors), .max_errors = ARRAY_SIZE(tegra234_cbb_errors),
.notifier_offset = 0x60000, .notifier_offset = 0x60000,
.off_mask_erd = 0x3a004 .off_mask_erd = 0x3a004,
.firewall_base = 0x10000,
.firewall_ctl = 0x23f0,
.firewall_wr_ctl = 0x23e8,
}; };
static const struct tegra234_slave_lookup tegra234_common_slave_map[] = { static const struct tegra234_slave_lookup tegra234_common_slave_map[] = {
...@@ -777,6 +832,9 @@ static const struct tegra234_cbb_fabric tegra234_dce_fabric = { ...@@ -777,6 +832,9 @@ static const struct tegra234_cbb_fabric tegra234_dce_fabric = {
.errors = tegra234_cbb_errors, .errors = tegra234_cbb_errors,
.max_errors = ARRAY_SIZE(tegra234_cbb_errors), .max_errors = ARRAY_SIZE(tegra234_cbb_errors),
.notifier_offset = 0x19000, .notifier_offset = 0x19000,
.firewall_base = 0x30000,
.firewall_ctl = 0x290,
.firewall_wr_ctl = 0x288,
}; };
static const struct tegra234_cbb_fabric tegra234_rce_fabric = { static const struct tegra234_cbb_fabric tegra234_rce_fabric = {
...@@ -787,6 +845,9 @@ static const struct tegra234_cbb_fabric tegra234_rce_fabric = { ...@@ -787,6 +845,9 @@ static const struct tegra234_cbb_fabric tegra234_rce_fabric = {
.errors = tegra234_cbb_errors, .errors = tegra234_cbb_errors,
.max_errors = ARRAY_SIZE(tegra234_cbb_errors), .max_errors = ARRAY_SIZE(tegra234_cbb_errors),
.notifier_offset = 0x19000, .notifier_offset = 0x19000,
.firewall_base = 0x30000,
.firewall_ctl = 0x290,
.firewall_wr_ctl = 0x288,
}; };
static const struct tegra234_cbb_fabric tegra234_sce_fabric = { static const struct tegra234_cbb_fabric tegra234_sce_fabric = {
...@@ -797,6 +858,9 @@ static const struct tegra234_cbb_fabric tegra234_sce_fabric = { ...@@ -797,6 +858,9 @@ static const struct tegra234_cbb_fabric tegra234_sce_fabric = {
.errors = tegra234_cbb_errors, .errors = tegra234_cbb_errors,
.max_errors = ARRAY_SIZE(tegra234_cbb_errors), .max_errors = ARRAY_SIZE(tegra234_cbb_errors),
.notifier_offset = 0x19000, .notifier_offset = 0x19000,
.firewall_base = 0x30000,
.firewall_ctl = 0x290,
.firewall_wr_ctl = 0x288,
}; };
static const char * const tegra241_master_id[] = { static const char * const tegra241_master_id[] = {
...@@ -979,6 +1043,9 @@ static const struct tegra234_cbb_fabric tegra241_cbb_fabric = { ...@@ -979,6 +1043,9 @@ static const struct tegra234_cbb_fabric tegra241_cbb_fabric = {
.max_errors = ARRAY_SIZE(tegra241_cbb_errors), .max_errors = ARRAY_SIZE(tegra241_cbb_errors),
.notifier_offset = 0x60000, .notifier_offset = 0x60000,
.off_mask_erd = 0x40004, .off_mask_erd = 0x40004,
.firewall_base = 0x20000,
.firewall_ctl = 0x2370,
.firewall_wr_ctl = 0x2368,
}; };
static const struct tegra234_slave_lookup tegra241_bpmp_slave_map[] = { static const struct tegra234_slave_lookup tegra241_bpmp_slave_map[] = {
...@@ -1000,6 +1067,9 @@ static const struct tegra234_cbb_fabric tegra241_bpmp_fabric = { ...@@ -1000,6 +1067,9 @@ static const struct tegra234_cbb_fabric tegra241_bpmp_fabric = {
.errors = tegra241_cbb_errors, .errors = tegra241_cbb_errors,
.max_errors = ARRAY_SIZE(tegra241_cbb_errors), .max_errors = ARRAY_SIZE(tegra241_cbb_errors),
.notifier_offset = 0x19000, .notifier_offset = 0x19000,
.firewall_base = 0x30000,
.firewall_ctl = 0x8f0,
.firewall_wr_ctl = 0x8e8,
}; };
static const struct of_device_id tegra234_cbb_dt_ids[] = { static const struct of_device_id tegra234_cbb_dt_ids[] = {
...@@ -1084,6 +1154,15 @@ static int tegra234_cbb_probe(struct platform_device *pdev) ...@@ -1084,6 +1154,15 @@ static int tegra234_cbb_probe(struct platform_device *pdev)
platform_set_drvdata(pdev, cbb); platform_set_drvdata(pdev, cbb);
/*
* Don't enable error reporting for a Fabric if write to it's registers
* is blocked by CBB firewall.
*/
if (!tegra234_cbb_write_access_allowed(pdev, cbb)) {
dev_info(&pdev->dev, "error reporting not enabled due to firewall\n");
return 0;
}
spin_lock_irqsave(&cbb_lock, flags); spin_lock_irqsave(&cbb_lock, flags);
list_add(&cbb->base.node, &cbb_list); list_add(&cbb->base.node, &cbb_list);
spin_unlock_irqrestore(&cbb_lock, flags); spin_unlock_irqrestore(&cbb_lock, flags);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment