Jetpack/kernel/nvidia/drivers/platform/tegra/tegra18_serr.c

137 lines
3.8 KiB
C

/*
* Copyright (c) 2015-2017, NVIDIA CORPORATION. All rights reserved.
*
* This program is free software; you can redistribute it and/or modify it
* under the terms and conditions of the GNU General Public License,
* version 2, as published by the Free Software Foundation.
*
* This program is distributed in the hope it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
* more details.
*/
#include <asm/traps.h>
#include <linux/debugfs.h>
#include <linux/debugfs.h>
#include <linux/cpu_pm.h>
#include <linux/cpu.h>
#include <linux/module.h>
#include <linux/of.h>
#include <linux/of_platform.h>
#include <linux/of_device.h>
#include <linux/platform_device.h>
#include <linux/platform/tegra/denver_mca.h>
#include <linux/tegra-mce.h>
#include <linux/t18x_ari.h>
#include <soc/tegra/chip-id.h>
/* Denver MCA */
/* Instantiate all accessors for a bank */
DEFINE_DENVER_MCA_OPS(0, 4, 0, 4, 1, 4, 2, 4, 3, 4, 4)
DEFINE_DENVER_MCA_OPS(1, 4, 6, 4, 7, 5, 0, 5, 1, 5, 2)
DEFINE_DENVER_MCA_OPS(2, 5, 4, 5, 5, 5, 6, 5, 7, 6, 0)
DEFINE_DENVER_MCA_OPS(3, 6, 2, 6, 3, 6, 4, 6, 5, 6, 6)
DEFINE_DENVER_MCA_OPS(4, 7, 0, 7, 1, 7, 2, 7, 3, 7, 4)
DEFINE_DENVER_MCA_OPS(5, 7, 6, 7, 7, 8, 0, 8, 1, 8, 2)
DEFINE_DENVER_MCA_OPS(6, 8, 4, 8, 5, 8, 6, 8, 7, 9, 0)
DEFINE_DENVER_MCA_OPS(7, 9, 2, 9, 3, 9, 4, 9, 5, 9, 6)
static struct denver_mca_error jsr_ret_errors[] = {
{.name = "Internal timer error",
.error_code = 0},
{}
};
static struct denver_mca_error jsr_mts_errors[] = {
{.name = "Internal unclasified error",
.error_code = 4},
{}
};
static struct denver_mca_bank denver_mca_banks[] = {
{.name = "JSR:RET",
DENVER_MCA_OP_ENTRY(0),
.errors = jsr_ret_errors},
{.name = "JSR:MTS",
DENVER_MCA_OP_ENTRY(1),
.errors = jsr_mts_errors},
{SIMPLE_DENVER_MCA_OP_ENTRY("DCC:1", 2)},
{SIMPLE_DENVER_MCA_OP_ENTRY("DCC:2", 3)},
{SIMPLE_DENVER_MCA_OP_ENTRY("LVB:3", 4)},
{SIMPLE_DENVER_MCA_OP_ENTRY("MM", 5)},
{SIMPLE_DENVER_MCA_OP_ENTRY("L2", 6)},
{SIMPLE_DENVER_MCA_OP_ENTRY("IFU", 7)},
{}
};
static void tegra18_register_denver_mca_banks(void)
{
int i;
for (i = 0; denver_mca_banks[i].name; i++)
register_denver_mca_bank(&denver_mca_banks[i]);
}
static void tegra18_unregister_denver_mca_banks(void)
{
int i;
for (i = 0; denver_mca_banks[i].name; i++)
unregister_denver_mca_bank(&denver_mca_banks[i]);
}
/*
* Without this enable set, Denver cores will halt on serrors
* instead of continuing (if possible) to the kernel serror
* handler
*/
static void tegra18_denver_serr_enable(void)
{
mca_cmd_t cmd;
u32 error;
u64 data;
cmd.data = 0;
cmd.cmd = TEGRA_ARI_MCA_WRITE_SERR;
cmd.idx = TEGRA_ARI_MCA_RD_WR_GLOBAL_CONFIG_REGISTER;
if (tegra_mce_write_uncore_mca(cmd, 1, &error))
pr_err("%s:mce write failed: error=0x%x\n", __func__, error);
cmd.cmd = TEGRA_ARI_MCA_READ_SERR;
cmd.idx = TEGRA_ARI_MCA_RD_WR_CCE;
cmd.subidx = TEGRA_ARI_MCA_RD_WR_ASERRX_MISC1;
if (tegra_mce_read_uncore_mca(cmd, &data, &error))
pr_err("%s:mce write failed: error=0x%x\n", __func__, error);
cmd.cmd = TEGRA_ARI_MCA_WRITE_SERR;
/* Disable Error Response (Data abort) on VPR reads.
* set PSN_ERR_CORR_MASK[7] = 1.
*/
data |= 1 << 19;
if (tegra_mce_write_uncore_mca(cmd, data, &error))
pr_err("%s:mce write failed: error=0x%x\n", __func__, error);
}
static int __init tegra18_serr_init(void)
{
if (tegra_get_chip_id() != TEGRA186)
return 0;
tegra18_denver_serr_enable();
tegra18_register_denver_mca_banks();
return 0;
}
module_init(tegra18_serr_init);
static void __exit tegra18_serr_exit(void)
{
tegra18_unregister_denver_mca_banks();
}
module_exit(tegra18_serr_exit);
MODULE_LICENSE("GPL v2");
MODULE_DESCRIPTION("Tegra T18x SError handler");