f5f7e1a049
Reading the debugfs files under /sys/kernel/debug/ccree/ can be done by the user at any time. On R-Car SoCs, the CCREE device is power-managed using a moduile clock, and if this clock is not running, bogus register values may be read. Fix this by filling in the debugfs_regset32.dev field, so debugfs will make sure the device is resumed while its registers are being read. This fixes the bogus values (0x00000260) in the register dumps on R-Car H3 ES1.0: -e6601000.crypto/regs:HOST_IRR = 0x00000260 -e6601000.crypto/regs:HOST_POWER_DOWN_EN = 0x00000260 +e6601000.crypto/regs:HOST_IRR = 0x00000038 +e6601000.crypto/regs:HOST_POWER_DOWN_EN = 0x00000038 e6601000.crypto/regs:AXIM_MON_ERR = 0x00000000 e6601000.crypto/regs:DSCRPTR_QUEUE_CONTENT = 0x000002aa -e6601000.crypto/regs:HOST_IMR = 0x00000260 +e6601000.crypto/regs:HOST_IMR = 0x017ffeff e6601000.crypto/regs:AXIM_CFG = 0x001f0007 e6601000.crypto/regs:AXIM_CACHE_PARAMS = 0x00000000 -e6601000.crypto/regs:GPR_HOST = 0x00000260 +e6601000.crypto/regs:GPR_HOST = 0x017ffeff e6601000.crypto/regs:AXIM_MON_COMP = 0x00000000 -e6601000.crypto/version:SIGNATURE = 0x00000260 -e6601000.crypto/version:VERSION = 0x00000260 +e6601000.crypto/version:SIGNATURE = 0xdcc63000 +e6601000.crypto/version:VERSION = 0xaf400001 Note that this behavior is system-dependent, and the issue does not show up on all R-Car Gen3 SoCs and boards. Even when the device is suspended, the module clock may be left enabled, if configured by the firmware for Secure Mode, or when controlled by the Real-Time Core. Signed-off-by: Geert Uytterhoeven <geert+renesas@glider.be> Acked-by: Gilad Ben-Yossef <gilad@benyossef.com> Acked-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com> Reviewed-by: Niklas Söderlund <niklas.soderlund@ragnatech.se> Signed-off-by: Herbert Xu <herbert@gondor.apana.org.au>
121 lines
2.9 KiB
C
121 lines
2.9 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/* Copyright (C) 2012-2019 ARM Limited or its affiliates. */
|
|
|
|
#include <linux/kernel.h>
|
|
#include <linux/debugfs.h>
|
|
#include <linux/stringify.h>
|
|
#include "cc_driver.h"
|
|
#include "cc_crypto_ctx.h"
|
|
#include "cc_debugfs.h"
|
|
|
|
struct cc_debugfs_ctx {
|
|
struct dentry *dir;
|
|
};
|
|
|
|
#define CC_DEBUG_REG(_X) { \
|
|
.name = __stringify(_X),\
|
|
.offset = CC_REG(_X) \
|
|
}
|
|
|
|
/*
|
|
* This is a global var for the dentry of the
|
|
* debugfs ccree/ dir. It is not tied down to
|
|
* a specific instance of ccree, hence it is
|
|
* global.
|
|
*/
|
|
static struct dentry *cc_debugfs_dir;
|
|
|
|
static struct debugfs_reg32 ver_sig_regs[] = {
|
|
{ .name = "SIGNATURE" }, /* Must be 0th */
|
|
{ .name = "VERSION" }, /* Must be 1st */
|
|
};
|
|
|
|
static struct debugfs_reg32 pid_cid_regs[] = {
|
|
CC_DEBUG_REG(PERIPHERAL_ID_0),
|
|
CC_DEBUG_REG(PERIPHERAL_ID_1),
|
|
CC_DEBUG_REG(PERIPHERAL_ID_2),
|
|
CC_DEBUG_REG(PERIPHERAL_ID_3),
|
|
CC_DEBUG_REG(PERIPHERAL_ID_4),
|
|
CC_DEBUG_REG(COMPONENT_ID_0),
|
|
CC_DEBUG_REG(COMPONENT_ID_1),
|
|
CC_DEBUG_REG(COMPONENT_ID_2),
|
|
CC_DEBUG_REG(COMPONENT_ID_3),
|
|
};
|
|
|
|
static struct debugfs_reg32 debug_regs[] = {
|
|
CC_DEBUG_REG(HOST_IRR),
|
|
CC_DEBUG_REG(HOST_POWER_DOWN_EN),
|
|
CC_DEBUG_REG(AXIM_MON_ERR),
|
|
CC_DEBUG_REG(DSCRPTR_QUEUE_CONTENT),
|
|
CC_DEBUG_REG(HOST_IMR),
|
|
CC_DEBUG_REG(AXIM_CFG),
|
|
CC_DEBUG_REG(AXIM_CACHE_PARAMS),
|
|
CC_DEBUG_REG(GPR_HOST),
|
|
CC_DEBUG_REG(AXIM_MON_COMP),
|
|
};
|
|
|
|
void __init cc_debugfs_global_init(void)
|
|
{
|
|
cc_debugfs_dir = debugfs_create_dir("ccree", NULL);
|
|
}
|
|
|
|
void __exit cc_debugfs_global_fini(void)
|
|
{
|
|
debugfs_remove(cc_debugfs_dir);
|
|
}
|
|
|
|
int cc_debugfs_init(struct cc_drvdata *drvdata)
|
|
{
|
|
struct device *dev = drvdata_to_dev(drvdata);
|
|
struct cc_debugfs_ctx *ctx;
|
|
struct debugfs_regset32 *regset, *verset;
|
|
|
|
ctx = devm_kzalloc(dev, sizeof(*ctx), GFP_KERNEL);
|
|
if (!ctx)
|
|
return -ENOMEM;
|
|
|
|
regset = devm_kzalloc(dev, sizeof(*regset), GFP_KERNEL);
|
|
if (!regset)
|
|
return -ENOMEM;
|
|
|
|
regset->regs = debug_regs;
|
|
regset->nregs = ARRAY_SIZE(debug_regs);
|
|
regset->base = drvdata->cc_base;
|
|
regset->dev = dev;
|
|
|
|
ctx->dir = debugfs_create_dir(drvdata->plat_dev->name, cc_debugfs_dir);
|
|
|
|
debugfs_create_regset32("regs", 0400, ctx->dir, regset);
|
|
debugfs_create_bool("coherent", 0400, ctx->dir, &drvdata->coherent);
|
|
|
|
verset = devm_kzalloc(dev, sizeof(*verset), GFP_KERNEL);
|
|
/* Failing here is not important enough to fail the module load */
|
|
if (!verset)
|
|
goto out;
|
|
|
|
if (drvdata->hw_rev <= CC_HW_REV_712) {
|
|
ver_sig_regs[0].offset = drvdata->sig_offset;
|
|
ver_sig_regs[1].offset = drvdata->ver_offset;
|
|
verset->regs = ver_sig_regs;
|
|
verset->nregs = ARRAY_SIZE(ver_sig_regs);
|
|
} else {
|
|
verset->regs = pid_cid_regs;
|
|
verset->nregs = ARRAY_SIZE(pid_cid_regs);
|
|
}
|
|
verset->base = drvdata->cc_base;
|
|
verset->dev = dev;
|
|
|
|
debugfs_create_regset32("version", 0400, ctx->dir, verset);
|
|
|
|
out:
|
|
drvdata->debugfs = ctx;
|
|
return 0;
|
|
}
|
|
|
|
void cc_debugfs_fini(struct cc_drvdata *drvdata)
|
|
{
|
|
struct cc_debugfs_ctx *ctx = (struct cc_debugfs_ctx *)drvdata->debugfs;
|
|
|
|
debugfs_remove_recursive(ctx->dir);
|
|
}
|