162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * SiFive composable cache controller Driver 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * Copyright (C) 2018-2022 SiFive, Inc. 662306a36Sopenharmony_ci * 762306a36Sopenharmony_ci */ 862306a36Sopenharmony_ci 962306a36Sopenharmony_ci#define pr_fmt(fmt) "CCACHE: " fmt 1062306a36Sopenharmony_ci 1162306a36Sopenharmony_ci#include <linux/debugfs.h> 1262306a36Sopenharmony_ci#include <linux/interrupt.h> 1362306a36Sopenharmony_ci#include <linux/of_irq.h> 1462306a36Sopenharmony_ci#include <linux/of_address.h> 1562306a36Sopenharmony_ci#include <linux/device.h> 1662306a36Sopenharmony_ci#include <linux/bitfield.h> 1762306a36Sopenharmony_ci#include <asm/cacheinfo.h> 1862306a36Sopenharmony_ci#include <soc/sifive/sifive_ccache.h> 1962306a36Sopenharmony_ci 2062306a36Sopenharmony_ci#define SIFIVE_CCACHE_DIRECCFIX_LOW 0x100 2162306a36Sopenharmony_ci#define SIFIVE_CCACHE_DIRECCFIX_HIGH 0x104 2262306a36Sopenharmony_ci#define SIFIVE_CCACHE_DIRECCFIX_COUNT 0x108 2362306a36Sopenharmony_ci 2462306a36Sopenharmony_ci#define SIFIVE_CCACHE_DIRECCFAIL_LOW 0x120 2562306a36Sopenharmony_ci#define SIFIVE_CCACHE_DIRECCFAIL_HIGH 0x124 2662306a36Sopenharmony_ci#define SIFIVE_CCACHE_DIRECCFAIL_COUNT 0x128 2762306a36Sopenharmony_ci 2862306a36Sopenharmony_ci#define SIFIVE_CCACHE_DATECCFIX_LOW 0x140 2962306a36Sopenharmony_ci#define SIFIVE_CCACHE_DATECCFIX_HIGH 0x144 3062306a36Sopenharmony_ci#define SIFIVE_CCACHE_DATECCFIX_COUNT 0x148 3162306a36Sopenharmony_ci 3262306a36Sopenharmony_ci#define SIFIVE_CCACHE_DATECCFAIL_LOW 0x160 3362306a36Sopenharmony_ci#define SIFIVE_CCACHE_DATECCFAIL_HIGH 0x164 3462306a36Sopenharmony_ci#define SIFIVE_CCACHE_DATECCFAIL_COUNT 0x168 3562306a36Sopenharmony_ci 3662306a36Sopenharmony_ci#define SIFIVE_CCACHE_CONFIG 0x00 3762306a36Sopenharmony_ci#define SIFIVE_CCACHE_CONFIG_BANK_MASK GENMASK_ULL(7, 0) 3862306a36Sopenharmony_ci#define SIFIVE_CCACHE_CONFIG_WAYS_MASK GENMASK_ULL(15, 8) 3962306a36Sopenharmony_ci#define SIFIVE_CCACHE_CONFIG_SETS_MASK GENMASK_ULL(23, 16) 4062306a36Sopenharmony_ci#define SIFIVE_CCACHE_CONFIG_BLKS_MASK GENMASK_ULL(31, 24) 4162306a36Sopenharmony_ci 4262306a36Sopenharmony_ci#define SIFIVE_CCACHE_WAYENABLE 0x08 4362306a36Sopenharmony_ci#define SIFIVE_CCACHE_ECCINJECTERR 0x40 4462306a36Sopenharmony_ci 4562306a36Sopenharmony_ci#define SIFIVE_CCACHE_MAX_ECCINTR 4 4662306a36Sopenharmony_ci 4762306a36Sopenharmony_cistatic void __iomem *ccache_base; 4862306a36Sopenharmony_cistatic int g_irq[SIFIVE_CCACHE_MAX_ECCINTR]; 4962306a36Sopenharmony_cistatic struct riscv_cacheinfo_ops ccache_cache_ops; 5062306a36Sopenharmony_cistatic int level; 5162306a36Sopenharmony_ci 5262306a36Sopenharmony_cienum { 5362306a36Sopenharmony_ci DIR_CORR = 0, 5462306a36Sopenharmony_ci DATA_CORR, 5562306a36Sopenharmony_ci DATA_UNCORR, 5662306a36Sopenharmony_ci DIR_UNCORR, 5762306a36Sopenharmony_ci}; 5862306a36Sopenharmony_ci 5962306a36Sopenharmony_ci#ifdef CONFIG_DEBUG_FS 6062306a36Sopenharmony_cistatic struct dentry *sifive_test; 6162306a36Sopenharmony_ci 6262306a36Sopenharmony_cistatic ssize_t ccache_write(struct file *file, const char __user *data, 6362306a36Sopenharmony_ci size_t count, loff_t *ppos) 6462306a36Sopenharmony_ci{ 6562306a36Sopenharmony_ci unsigned int val; 6662306a36Sopenharmony_ci 6762306a36Sopenharmony_ci if (kstrtouint_from_user(data, count, 0, &val)) 6862306a36Sopenharmony_ci return -EINVAL; 6962306a36Sopenharmony_ci if ((val < 0xFF) || (val >= 0x10000 && val < 0x100FF)) 7062306a36Sopenharmony_ci writel(val, ccache_base + SIFIVE_CCACHE_ECCINJECTERR); 7162306a36Sopenharmony_ci else 7262306a36Sopenharmony_ci return -EINVAL; 7362306a36Sopenharmony_ci return count; 7462306a36Sopenharmony_ci} 7562306a36Sopenharmony_ci 7662306a36Sopenharmony_cistatic const struct file_operations ccache_fops = { 7762306a36Sopenharmony_ci .owner = THIS_MODULE, 7862306a36Sopenharmony_ci .open = simple_open, 7962306a36Sopenharmony_ci .write = ccache_write 8062306a36Sopenharmony_ci}; 8162306a36Sopenharmony_ci 8262306a36Sopenharmony_cistatic void setup_sifive_debug(void) 8362306a36Sopenharmony_ci{ 8462306a36Sopenharmony_ci sifive_test = debugfs_create_dir("sifive_ccache_cache", NULL); 8562306a36Sopenharmony_ci 8662306a36Sopenharmony_ci debugfs_create_file("sifive_debug_inject_error", 0200, 8762306a36Sopenharmony_ci sifive_test, NULL, &ccache_fops); 8862306a36Sopenharmony_ci} 8962306a36Sopenharmony_ci#endif 9062306a36Sopenharmony_ci 9162306a36Sopenharmony_cistatic void ccache_config_read(void) 9262306a36Sopenharmony_ci{ 9362306a36Sopenharmony_ci u32 cfg; 9462306a36Sopenharmony_ci 9562306a36Sopenharmony_ci cfg = readl(ccache_base + SIFIVE_CCACHE_CONFIG); 9662306a36Sopenharmony_ci pr_info("%llu banks, %llu ways, sets/bank=%llu, bytes/block=%llu\n", 9762306a36Sopenharmony_ci FIELD_GET(SIFIVE_CCACHE_CONFIG_BANK_MASK, cfg), 9862306a36Sopenharmony_ci FIELD_GET(SIFIVE_CCACHE_CONFIG_WAYS_MASK, cfg), 9962306a36Sopenharmony_ci BIT_ULL(FIELD_GET(SIFIVE_CCACHE_CONFIG_SETS_MASK, cfg)), 10062306a36Sopenharmony_ci BIT_ULL(FIELD_GET(SIFIVE_CCACHE_CONFIG_BLKS_MASK, cfg))); 10162306a36Sopenharmony_ci 10262306a36Sopenharmony_ci cfg = readl(ccache_base + SIFIVE_CCACHE_WAYENABLE); 10362306a36Sopenharmony_ci pr_info("Index of the largest way enabled: %u\n", cfg); 10462306a36Sopenharmony_ci} 10562306a36Sopenharmony_ci 10662306a36Sopenharmony_cistatic const struct of_device_id sifive_ccache_ids[] = { 10762306a36Sopenharmony_ci { .compatible = "sifive,fu540-c000-ccache" }, 10862306a36Sopenharmony_ci { .compatible = "sifive,fu740-c000-ccache" }, 10962306a36Sopenharmony_ci { .compatible = "sifive,ccache0" }, 11062306a36Sopenharmony_ci { /* end of table */ } 11162306a36Sopenharmony_ci}; 11262306a36Sopenharmony_ci 11362306a36Sopenharmony_cistatic ATOMIC_NOTIFIER_HEAD(ccache_err_chain); 11462306a36Sopenharmony_ci 11562306a36Sopenharmony_ciint register_sifive_ccache_error_notifier(struct notifier_block *nb) 11662306a36Sopenharmony_ci{ 11762306a36Sopenharmony_ci return atomic_notifier_chain_register(&ccache_err_chain, nb); 11862306a36Sopenharmony_ci} 11962306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(register_sifive_ccache_error_notifier); 12062306a36Sopenharmony_ci 12162306a36Sopenharmony_ciint unregister_sifive_ccache_error_notifier(struct notifier_block *nb) 12262306a36Sopenharmony_ci{ 12362306a36Sopenharmony_ci return atomic_notifier_chain_unregister(&ccache_err_chain, nb); 12462306a36Sopenharmony_ci} 12562306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(unregister_sifive_ccache_error_notifier); 12662306a36Sopenharmony_ci 12762306a36Sopenharmony_cistatic int ccache_largest_wayenabled(void) 12862306a36Sopenharmony_ci{ 12962306a36Sopenharmony_ci return readl(ccache_base + SIFIVE_CCACHE_WAYENABLE) & 0xFF; 13062306a36Sopenharmony_ci} 13162306a36Sopenharmony_ci 13262306a36Sopenharmony_cistatic ssize_t number_of_ways_enabled_show(struct device *dev, 13362306a36Sopenharmony_ci struct device_attribute *attr, 13462306a36Sopenharmony_ci char *buf) 13562306a36Sopenharmony_ci{ 13662306a36Sopenharmony_ci return sprintf(buf, "%u\n", ccache_largest_wayenabled()); 13762306a36Sopenharmony_ci} 13862306a36Sopenharmony_ci 13962306a36Sopenharmony_cistatic DEVICE_ATTR_RO(number_of_ways_enabled); 14062306a36Sopenharmony_ci 14162306a36Sopenharmony_cistatic struct attribute *priv_attrs[] = { 14262306a36Sopenharmony_ci &dev_attr_number_of_ways_enabled.attr, 14362306a36Sopenharmony_ci NULL, 14462306a36Sopenharmony_ci}; 14562306a36Sopenharmony_ci 14662306a36Sopenharmony_cistatic const struct attribute_group priv_attr_group = { 14762306a36Sopenharmony_ci .attrs = priv_attrs, 14862306a36Sopenharmony_ci}; 14962306a36Sopenharmony_ci 15062306a36Sopenharmony_cistatic const struct attribute_group *ccache_get_priv_group(struct cacheinfo 15162306a36Sopenharmony_ci *this_leaf) 15262306a36Sopenharmony_ci{ 15362306a36Sopenharmony_ci /* We want to use private group for composable cache only */ 15462306a36Sopenharmony_ci if (this_leaf->level == level) 15562306a36Sopenharmony_ci return &priv_attr_group; 15662306a36Sopenharmony_ci else 15762306a36Sopenharmony_ci return NULL; 15862306a36Sopenharmony_ci} 15962306a36Sopenharmony_ci 16062306a36Sopenharmony_cistatic irqreturn_t ccache_int_handler(int irq, void *device) 16162306a36Sopenharmony_ci{ 16262306a36Sopenharmony_ci unsigned int add_h, add_l; 16362306a36Sopenharmony_ci 16462306a36Sopenharmony_ci if (irq == g_irq[DIR_CORR]) { 16562306a36Sopenharmony_ci add_h = readl(ccache_base + SIFIVE_CCACHE_DIRECCFIX_HIGH); 16662306a36Sopenharmony_ci add_l = readl(ccache_base + SIFIVE_CCACHE_DIRECCFIX_LOW); 16762306a36Sopenharmony_ci pr_err("DirError @ 0x%08X.%08X\n", add_h, add_l); 16862306a36Sopenharmony_ci /* Reading this register clears the DirError interrupt sig */ 16962306a36Sopenharmony_ci readl(ccache_base + SIFIVE_CCACHE_DIRECCFIX_COUNT); 17062306a36Sopenharmony_ci atomic_notifier_call_chain(&ccache_err_chain, 17162306a36Sopenharmony_ci SIFIVE_CCACHE_ERR_TYPE_CE, 17262306a36Sopenharmony_ci "DirECCFix"); 17362306a36Sopenharmony_ci } 17462306a36Sopenharmony_ci if (irq == g_irq[DIR_UNCORR]) { 17562306a36Sopenharmony_ci add_h = readl(ccache_base + SIFIVE_CCACHE_DIRECCFAIL_HIGH); 17662306a36Sopenharmony_ci add_l = readl(ccache_base + SIFIVE_CCACHE_DIRECCFAIL_LOW); 17762306a36Sopenharmony_ci /* Reading this register clears the DirFail interrupt sig */ 17862306a36Sopenharmony_ci readl(ccache_base + SIFIVE_CCACHE_DIRECCFAIL_COUNT); 17962306a36Sopenharmony_ci atomic_notifier_call_chain(&ccache_err_chain, 18062306a36Sopenharmony_ci SIFIVE_CCACHE_ERR_TYPE_UE, 18162306a36Sopenharmony_ci "DirECCFail"); 18262306a36Sopenharmony_ci panic("CCACHE: DirFail @ 0x%08X.%08X\n", add_h, add_l); 18362306a36Sopenharmony_ci } 18462306a36Sopenharmony_ci if (irq == g_irq[DATA_CORR]) { 18562306a36Sopenharmony_ci add_h = readl(ccache_base + SIFIVE_CCACHE_DATECCFIX_HIGH); 18662306a36Sopenharmony_ci add_l = readl(ccache_base + SIFIVE_CCACHE_DATECCFIX_LOW); 18762306a36Sopenharmony_ci pr_err("DataError @ 0x%08X.%08X\n", add_h, add_l); 18862306a36Sopenharmony_ci /* Reading this register clears the DataError interrupt sig */ 18962306a36Sopenharmony_ci readl(ccache_base + SIFIVE_CCACHE_DATECCFIX_COUNT); 19062306a36Sopenharmony_ci atomic_notifier_call_chain(&ccache_err_chain, 19162306a36Sopenharmony_ci SIFIVE_CCACHE_ERR_TYPE_CE, 19262306a36Sopenharmony_ci "DatECCFix"); 19362306a36Sopenharmony_ci } 19462306a36Sopenharmony_ci if (irq == g_irq[DATA_UNCORR]) { 19562306a36Sopenharmony_ci add_h = readl(ccache_base + SIFIVE_CCACHE_DATECCFAIL_HIGH); 19662306a36Sopenharmony_ci add_l = readl(ccache_base + SIFIVE_CCACHE_DATECCFAIL_LOW); 19762306a36Sopenharmony_ci pr_err("DataFail @ 0x%08X.%08X\n", add_h, add_l); 19862306a36Sopenharmony_ci /* Reading this register clears the DataFail interrupt sig */ 19962306a36Sopenharmony_ci readl(ccache_base + SIFIVE_CCACHE_DATECCFAIL_COUNT); 20062306a36Sopenharmony_ci atomic_notifier_call_chain(&ccache_err_chain, 20162306a36Sopenharmony_ci SIFIVE_CCACHE_ERR_TYPE_UE, 20262306a36Sopenharmony_ci "DatECCFail"); 20362306a36Sopenharmony_ci } 20462306a36Sopenharmony_ci 20562306a36Sopenharmony_ci return IRQ_HANDLED; 20662306a36Sopenharmony_ci} 20762306a36Sopenharmony_ci 20862306a36Sopenharmony_cistatic int __init sifive_ccache_init(void) 20962306a36Sopenharmony_ci{ 21062306a36Sopenharmony_ci struct device_node *np; 21162306a36Sopenharmony_ci struct resource res; 21262306a36Sopenharmony_ci int i, rc, intr_num; 21362306a36Sopenharmony_ci 21462306a36Sopenharmony_ci np = of_find_matching_node(NULL, sifive_ccache_ids); 21562306a36Sopenharmony_ci if (!np) 21662306a36Sopenharmony_ci return -ENODEV; 21762306a36Sopenharmony_ci 21862306a36Sopenharmony_ci if (of_address_to_resource(np, 0, &res)) { 21962306a36Sopenharmony_ci rc = -ENODEV; 22062306a36Sopenharmony_ci goto err_node_put; 22162306a36Sopenharmony_ci } 22262306a36Sopenharmony_ci 22362306a36Sopenharmony_ci ccache_base = ioremap(res.start, resource_size(&res)); 22462306a36Sopenharmony_ci if (!ccache_base) { 22562306a36Sopenharmony_ci rc = -ENOMEM; 22662306a36Sopenharmony_ci goto err_node_put; 22762306a36Sopenharmony_ci } 22862306a36Sopenharmony_ci 22962306a36Sopenharmony_ci if (of_property_read_u32(np, "cache-level", &level)) { 23062306a36Sopenharmony_ci rc = -ENOENT; 23162306a36Sopenharmony_ci goto err_unmap; 23262306a36Sopenharmony_ci } 23362306a36Sopenharmony_ci 23462306a36Sopenharmony_ci intr_num = of_property_count_u32_elems(np, "interrupts"); 23562306a36Sopenharmony_ci if (!intr_num) { 23662306a36Sopenharmony_ci pr_err("No interrupts property\n"); 23762306a36Sopenharmony_ci rc = -ENODEV; 23862306a36Sopenharmony_ci goto err_unmap; 23962306a36Sopenharmony_ci } 24062306a36Sopenharmony_ci 24162306a36Sopenharmony_ci for (i = 0; i < intr_num; i++) { 24262306a36Sopenharmony_ci g_irq[i] = irq_of_parse_and_map(np, i); 24362306a36Sopenharmony_ci rc = request_irq(g_irq[i], ccache_int_handler, 0, "ccache_ecc", 24462306a36Sopenharmony_ci NULL); 24562306a36Sopenharmony_ci if (rc) { 24662306a36Sopenharmony_ci pr_err("Could not request IRQ %d\n", g_irq[i]); 24762306a36Sopenharmony_ci goto err_free_irq; 24862306a36Sopenharmony_ci } 24962306a36Sopenharmony_ci } 25062306a36Sopenharmony_ci of_node_put(np); 25162306a36Sopenharmony_ci 25262306a36Sopenharmony_ci ccache_config_read(); 25362306a36Sopenharmony_ci 25462306a36Sopenharmony_ci ccache_cache_ops.get_priv_group = ccache_get_priv_group; 25562306a36Sopenharmony_ci riscv_set_cacheinfo_ops(&ccache_cache_ops); 25662306a36Sopenharmony_ci 25762306a36Sopenharmony_ci#ifdef CONFIG_DEBUG_FS 25862306a36Sopenharmony_ci setup_sifive_debug(); 25962306a36Sopenharmony_ci#endif 26062306a36Sopenharmony_ci return 0; 26162306a36Sopenharmony_ci 26262306a36Sopenharmony_cierr_free_irq: 26362306a36Sopenharmony_ci while (--i >= 0) 26462306a36Sopenharmony_ci free_irq(g_irq[i], NULL); 26562306a36Sopenharmony_cierr_unmap: 26662306a36Sopenharmony_ci iounmap(ccache_base); 26762306a36Sopenharmony_cierr_node_put: 26862306a36Sopenharmony_ci of_node_put(np); 26962306a36Sopenharmony_ci return rc; 27062306a36Sopenharmony_ci} 27162306a36Sopenharmony_ci 27262306a36Sopenharmony_cidevice_initcall(sifive_ccache_init); 273