162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * Copyright (C) 2014-2021 Broadcom 462306a36Sopenharmony_ci */ 562306a36Sopenharmony_ci 662306a36Sopenharmony_ci#include <linux/init.h> 762306a36Sopenharmony_ci#include <linux/types.h> 862306a36Sopenharmony_ci#include <linux/module.h> 962306a36Sopenharmony_ci#include <linux/panic_notifier.h> 1062306a36Sopenharmony_ci#include <linux/platform_device.h> 1162306a36Sopenharmony_ci#include <linux/interrupt.h> 1262306a36Sopenharmony_ci#include <linux/sysfs.h> 1362306a36Sopenharmony_ci#include <linux/io.h> 1462306a36Sopenharmony_ci#include <linux/string.h> 1562306a36Sopenharmony_ci#include <linux/device.h> 1662306a36Sopenharmony_ci#include <linux/list.h> 1762306a36Sopenharmony_ci#include <linux/of.h> 1862306a36Sopenharmony_ci#include <linux/bitops.h> 1962306a36Sopenharmony_ci#include <linux/pm.h> 2062306a36Sopenharmony_ci#include <linux/kernel.h> 2162306a36Sopenharmony_ci#include <linux/kdebug.h> 2262306a36Sopenharmony_ci#include <linux/notifier.h> 2362306a36Sopenharmony_ci 2462306a36Sopenharmony_ci#ifdef CONFIG_MIPS 2562306a36Sopenharmony_ci#include <asm/traps.h> 2662306a36Sopenharmony_ci#endif 2762306a36Sopenharmony_ci 2862306a36Sopenharmony_ci#define ARB_ERR_CAP_CLEAR (1 << 0) 2962306a36Sopenharmony_ci#define ARB_ERR_CAP_STATUS_TIMEOUT (1 << 12) 3062306a36Sopenharmony_ci#define ARB_ERR_CAP_STATUS_TEA (1 << 11) 3162306a36Sopenharmony_ci#define ARB_ERR_CAP_STATUS_WRITE (1 << 1) 3262306a36Sopenharmony_ci#define ARB_ERR_CAP_STATUS_VALID (1 << 0) 3362306a36Sopenharmony_ci 3462306a36Sopenharmony_ci#define ARB_BP_CAP_CLEAR (1 << 0) 3562306a36Sopenharmony_ci#define ARB_BP_CAP_STATUS_PROT_SHIFT 14 3662306a36Sopenharmony_ci#define ARB_BP_CAP_STATUS_TYPE (1 << 13) 3762306a36Sopenharmony_ci#define ARB_BP_CAP_STATUS_RSP_SHIFT 10 3862306a36Sopenharmony_ci#define ARB_BP_CAP_STATUS_MASK GENMASK(1, 0) 3962306a36Sopenharmony_ci#define ARB_BP_CAP_STATUS_BS_SHIFT 2 4062306a36Sopenharmony_ci#define ARB_BP_CAP_STATUS_WRITE (1 << 1) 4162306a36Sopenharmony_ci#define ARB_BP_CAP_STATUS_VALID (1 << 0) 4262306a36Sopenharmony_ci 4362306a36Sopenharmony_cienum { 4462306a36Sopenharmony_ci ARB_TIMER, 4562306a36Sopenharmony_ci ARB_BP_CAP_CLR, 4662306a36Sopenharmony_ci ARB_BP_CAP_HI_ADDR, 4762306a36Sopenharmony_ci ARB_BP_CAP_ADDR, 4862306a36Sopenharmony_ci ARB_BP_CAP_STATUS, 4962306a36Sopenharmony_ci ARB_BP_CAP_MASTER, 5062306a36Sopenharmony_ci ARB_ERR_CAP_CLR, 5162306a36Sopenharmony_ci ARB_ERR_CAP_HI_ADDR, 5262306a36Sopenharmony_ci ARB_ERR_CAP_ADDR, 5362306a36Sopenharmony_ci ARB_ERR_CAP_STATUS, 5462306a36Sopenharmony_ci ARB_ERR_CAP_MASTER, 5562306a36Sopenharmony_ci}; 5662306a36Sopenharmony_ci 5762306a36Sopenharmony_cistatic const int gisb_offsets_bcm7038[] = { 5862306a36Sopenharmony_ci [ARB_TIMER] = 0x00c, 5962306a36Sopenharmony_ci [ARB_BP_CAP_CLR] = 0x014, 6062306a36Sopenharmony_ci [ARB_BP_CAP_HI_ADDR] = -1, 6162306a36Sopenharmony_ci [ARB_BP_CAP_ADDR] = 0x0b8, 6262306a36Sopenharmony_ci [ARB_BP_CAP_STATUS] = 0x0c0, 6362306a36Sopenharmony_ci [ARB_BP_CAP_MASTER] = -1, 6462306a36Sopenharmony_ci [ARB_ERR_CAP_CLR] = 0x0c4, 6562306a36Sopenharmony_ci [ARB_ERR_CAP_HI_ADDR] = -1, 6662306a36Sopenharmony_ci [ARB_ERR_CAP_ADDR] = 0x0c8, 6762306a36Sopenharmony_ci [ARB_ERR_CAP_STATUS] = 0x0d0, 6862306a36Sopenharmony_ci [ARB_ERR_CAP_MASTER] = -1, 6962306a36Sopenharmony_ci}; 7062306a36Sopenharmony_ci 7162306a36Sopenharmony_cistatic const int gisb_offsets_bcm7278[] = { 7262306a36Sopenharmony_ci [ARB_TIMER] = 0x008, 7362306a36Sopenharmony_ci [ARB_BP_CAP_CLR] = 0x01c, 7462306a36Sopenharmony_ci [ARB_BP_CAP_HI_ADDR] = -1, 7562306a36Sopenharmony_ci [ARB_BP_CAP_ADDR] = 0x220, 7662306a36Sopenharmony_ci [ARB_BP_CAP_STATUS] = 0x230, 7762306a36Sopenharmony_ci [ARB_BP_CAP_MASTER] = 0x234, 7862306a36Sopenharmony_ci [ARB_ERR_CAP_CLR] = 0x7f8, 7962306a36Sopenharmony_ci [ARB_ERR_CAP_HI_ADDR] = -1, 8062306a36Sopenharmony_ci [ARB_ERR_CAP_ADDR] = 0x7e0, 8162306a36Sopenharmony_ci [ARB_ERR_CAP_STATUS] = 0x7f0, 8262306a36Sopenharmony_ci [ARB_ERR_CAP_MASTER] = 0x7f4, 8362306a36Sopenharmony_ci}; 8462306a36Sopenharmony_ci 8562306a36Sopenharmony_cistatic const int gisb_offsets_bcm7400[] = { 8662306a36Sopenharmony_ci [ARB_TIMER] = 0x00c, 8762306a36Sopenharmony_ci [ARB_BP_CAP_CLR] = 0x014, 8862306a36Sopenharmony_ci [ARB_BP_CAP_HI_ADDR] = -1, 8962306a36Sopenharmony_ci [ARB_BP_CAP_ADDR] = 0x0b8, 9062306a36Sopenharmony_ci [ARB_BP_CAP_STATUS] = 0x0c0, 9162306a36Sopenharmony_ci [ARB_BP_CAP_MASTER] = 0x0c4, 9262306a36Sopenharmony_ci [ARB_ERR_CAP_CLR] = 0x0c8, 9362306a36Sopenharmony_ci [ARB_ERR_CAP_HI_ADDR] = -1, 9462306a36Sopenharmony_ci [ARB_ERR_CAP_ADDR] = 0x0cc, 9562306a36Sopenharmony_ci [ARB_ERR_CAP_STATUS] = 0x0d4, 9662306a36Sopenharmony_ci [ARB_ERR_CAP_MASTER] = 0x0d8, 9762306a36Sopenharmony_ci}; 9862306a36Sopenharmony_ci 9962306a36Sopenharmony_cistatic const int gisb_offsets_bcm7435[] = { 10062306a36Sopenharmony_ci [ARB_TIMER] = 0x00c, 10162306a36Sopenharmony_ci [ARB_BP_CAP_CLR] = 0x014, 10262306a36Sopenharmony_ci [ARB_BP_CAP_HI_ADDR] = -1, 10362306a36Sopenharmony_ci [ARB_BP_CAP_ADDR] = 0x158, 10462306a36Sopenharmony_ci [ARB_BP_CAP_STATUS] = 0x160, 10562306a36Sopenharmony_ci [ARB_BP_CAP_MASTER] = 0x164, 10662306a36Sopenharmony_ci [ARB_ERR_CAP_CLR] = 0x168, 10762306a36Sopenharmony_ci [ARB_ERR_CAP_HI_ADDR] = -1, 10862306a36Sopenharmony_ci [ARB_ERR_CAP_ADDR] = 0x16c, 10962306a36Sopenharmony_ci [ARB_ERR_CAP_STATUS] = 0x174, 11062306a36Sopenharmony_ci [ARB_ERR_CAP_MASTER] = 0x178, 11162306a36Sopenharmony_ci}; 11262306a36Sopenharmony_ci 11362306a36Sopenharmony_cistatic const int gisb_offsets_bcm7445[] = { 11462306a36Sopenharmony_ci [ARB_TIMER] = 0x008, 11562306a36Sopenharmony_ci [ARB_BP_CAP_CLR] = 0x010, 11662306a36Sopenharmony_ci [ARB_BP_CAP_HI_ADDR] = -1, 11762306a36Sopenharmony_ci [ARB_BP_CAP_ADDR] = 0x1d8, 11862306a36Sopenharmony_ci [ARB_BP_CAP_STATUS] = 0x1e0, 11962306a36Sopenharmony_ci [ARB_BP_CAP_MASTER] = 0x1e4, 12062306a36Sopenharmony_ci [ARB_ERR_CAP_CLR] = 0x7e4, 12162306a36Sopenharmony_ci [ARB_ERR_CAP_HI_ADDR] = 0x7e8, 12262306a36Sopenharmony_ci [ARB_ERR_CAP_ADDR] = 0x7ec, 12362306a36Sopenharmony_ci [ARB_ERR_CAP_STATUS] = 0x7f4, 12462306a36Sopenharmony_ci [ARB_ERR_CAP_MASTER] = 0x7f8, 12562306a36Sopenharmony_ci}; 12662306a36Sopenharmony_ci 12762306a36Sopenharmony_cistruct brcmstb_gisb_arb_device { 12862306a36Sopenharmony_ci void __iomem *base; 12962306a36Sopenharmony_ci const int *gisb_offsets; 13062306a36Sopenharmony_ci bool big_endian; 13162306a36Sopenharmony_ci struct mutex lock; 13262306a36Sopenharmony_ci struct list_head next; 13362306a36Sopenharmony_ci u32 valid_mask; 13462306a36Sopenharmony_ci const char *master_names[sizeof(u32) * BITS_PER_BYTE]; 13562306a36Sopenharmony_ci u32 saved_timeout; 13662306a36Sopenharmony_ci}; 13762306a36Sopenharmony_ci 13862306a36Sopenharmony_cistatic LIST_HEAD(brcmstb_gisb_arb_device_list); 13962306a36Sopenharmony_ci 14062306a36Sopenharmony_cistatic u32 gisb_read(struct brcmstb_gisb_arb_device *gdev, int reg) 14162306a36Sopenharmony_ci{ 14262306a36Sopenharmony_ci int offset = gdev->gisb_offsets[reg]; 14362306a36Sopenharmony_ci 14462306a36Sopenharmony_ci if (offset < 0) { 14562306a36Sopenharmony_ci /* return 1 if the hardware doesn't have ARB_ERR_CAP_MASTER */ 14662306a36Sopenharmony_ci if (reg == ARB_ERR_CAP_MASTER) 14762306a36Sopenharmony_ci return 1; 14862306a36Sopenharmony_ci else 14962306a36Sopenharmony_ci return 0; 15062306a36Sopenharmony_ci } 15162306a36Sopenharmony_ci 15262306a36Sopenharmony_ci if (gdev->big_endian) 15362306a36Sopenharmony_ci return ioread32be(gdev->base + offset); 15462306a36Sopenharmony_ci else 15562306a36Sopenharmony_ci return ioread32(gdev->base + offset); 15662306a36Sopenharmony_ci} 15762306a36Sopenharmony_ci 15862306a36Sopenharmony_cistatic u64 gisb_read_address(struct brcmstb_gisb_arb_device *gdev) 15962306a36Sopenharmony_ci{ 16062306a36Sopenharmony_ci u64 value; 16162306a36Sopenharmony_ci 16262306a36Sopenharmony_ci value = gisb_read(gdev, ARB_ERR_CAP_ADDR); 16362306a36Sopenharmony_ci value |= (u64)gisb_read(gdev, ARB_ERR_CAP_HI_ADDR) << 32; 16462306a36Sopenharmony_ci 16562306a36Sopenharmony_ci return value; 16662306a36Sopenharmony_ci} 16762306a36Sopenharmony_ci 16862306a36Sopenharmony_cistatic u64 gisb_read_bp_address(struct brcmstb_gisb_arb_device *gdev) 16962306a36Sopenharmony_ci{ 17062306a36Sopenharmony_ci u64 value; 17162306a36Sopenharmony_ci 17262306a36Sopenharmony_ci value = gisb_read(gdev, ARB_BP_CAP_ADDR); 17362306a36Sopenharmony_ci value |= (u64)gisb_read(gdev, ARB_BP_CAP_HI_ADDR) << 32; 17462306a36Sopenharmony_ci 17562306a36Sopenharmony_ci return value; 17662306a36Sopenharmony_ci} 17762306a36Sopenharmony_ci 17862306a36Sopenharmony_cistatic void gisb_write(struct brcmstb_gisb_arb_device *gdev, u32 val, int reg) 17962306a36Sopenharmony_ci{ 18062306a36Sopenharmony_ci int offset = gdev->gisb_offsets[reg]; 18162306a36Sopenharmony_ci 18262306a36Sopenharmony_ci if (offset == -1) 18362306a36Sopenharmony_ci return; 18462306a36Sopenharmony_ci 18562306a36Sopenharmony_ci if (gdev->big_endian) 18662306a36Sopenharmony_ci iowrite32be(val, gdev->base + offset); 18762306a36Sopenharmony_ci else 18862306a36Sopenharmony_ci iowrite32(val, gdev->base + offset); 18962306a36Sopenharmony_ci} 19062306a36Sopenharmony_ci 19162306a36Sopenharmony_cistatic ssize_t gisb_arb_get_timeout(struct device *dev, 19262306a36Sopenharmony_ci struct device_attribute *attr, 19362306a36Sopenharmony_ci char *buf) 19462306a36Sopenharmony_ci{ 19562306a36Sopenharmony_ci struct brcmstb_gisb_arb_device *gdev = dev_get_drvdata(dev); 19662306a36Sopenharmony_ci u32 timeout; 19762306a36Sopenharmony_ci 19862306a36Sopenharmony_ci mutex_lock(&gdev->lock); 19962306a36Sopenharmony_ci timeout = gisb_read(gdev, ARB_TIMER); 20062306a36Sopenharmony_ci mutex_unlock(&gdev->lock); 20162306a36Sopenharmony_ci 20262306a36Sopenharmony_ci return sprintf(buf, "%d", timeout); 20362306a36Sopenharmony_ci} 20462306a36Sopenharmony_ci 20562306a36Sopenharmony_cistatic ssize_t gisb_arb_set_timeout(struct device *dev, 20662306a36Sopenharmony_ci struct device_attribute *attr, 20762306a36Sopenharmony_ci const char *buf, size_t count) 20862306a36Sopenharmony_ci{ 20962306a36Sopenharmony_ci struct brcmstb_gisb_arb_device *gdev = dev_get_drvdata(dev); 21062306a36Sopenharmony_ci int val, ret; 21162306a36Sopenharmony_ci 21262306a36Sopenharmony_ci ret = kstrtoint(buf, 10, &val); 21362306a36Sopenharmony_ci if (ret < 0) 21462306a36Sopenharmony_ci return ret; 21562306a36Sopenharmony_ci 21662306a36Sopenharmony_ci if (val == 0 || val >= 0xffffffff) 21762306a36Sopenharmony_ci return -EINVAL; 21862306a36Sopenharmony_ci 21962306a36Sopenharmony_ci mutex_lock(&gdev->lock); 22062306a36Sopenharmony_ci gisb_write(gdev, val, ARB_TIMER); 22162306a36Sopenharmony_ci mutex_unlock(&gdev->lock); 22262306a36Sopenharmony_ci 22362306a36Sopenharmony_ci return count; 22462306a36Sopenharmony_ci} 22562306a36Sopenharmony_ci 22662306a36Sopenharmony_cistatic const char * 22762306a36Sopenharmony_cibrcmstb_gisb_master_to_str(struct brcmstb_gisb_arb_device *gdev, 22862306a36Sopenharmony_ci u32 masters) 22962306a36Sopenharmony_ci{ 23062306a36Sopenharmony_ci u32 mask = gdev->valid_mask & masters; 23162306a36Sopenharmony_ci 23262306a36Sopenharmony_ci if (hweight_long(mask) != 1) 23362306a36Sopenharmony_ci return NULL; 23462306a36Sopenharmony_ci 23562306a36Sopenharmony_ci return gdev->master_names[ffs(mask) - 1]; 23662306a36Sopenharmony_ci} 23762306a36Sopenharmony_ci 23862306a36Sopenharmony_cistatic int brcmstb_gisb_arb_decode_addr(struct brcmstb_gisb_arb_device *gdev, 23962306a36Sopenharmony_ci const char *reason) 24062306a36Sopenharmony_ci{ 24162306a36Sopenharmony_ci u32 cap_status; 24262306a36Sopenharmony_ci u64 arb_addr; 24362306a36Sopenharmony_ci u32 master; 24462306a36Sopenharmony_ci const char *m_name; 24562306a36Sopenharmony_ci char m_fmt[11]; 24662306a36Sopenharmony_ci 24762306a36Sopenharmony_ci cap_status = gisb_read(gdev, ARB_ERR_CAP_STATUS); 24862306a36Sopenharmony_ci 24962306a36Sopenharmony_ci /* Invalid captured address, bail out */ 25062306a36Sopenharmony_ci if (!(cap_status & ARB_ERR_CAP_STATUS_VALID)) 25162306a36Sopenharmony_ci return 1; 25262306a36Sopenharmony_ci 25362306a36Sopenharmony_ci /* Read the address and master */ 25462306a36Sopenharmony_ci arb_addr = gisb_read_address(gdev); 25562306a36Sopenharmony_ci master = gisb_read(gdev, ARB_ERR_CAP_MASTER); 25662306a36Sopenharmony_ci 25762306a36Sopenharmony_ci m_name = brcmstb_gisb_master_to_str(gdev, master); 25862306a36Sopenharmony_ci if (!m_name) { 25962306a36Sopenharmony_ci snprintf(m_fmt, sizeof(m_fmt), "0x%08x", master); 26062306a36Sopenharmony_ci m_name = m_fmt; 26162306a36Sopenharmony_ci } 26262306a36Sopenharmony_ci 26362306a36Sopenharmony_ci pr_crit("GISB: %s at 0x%llx [%c %s], core: %s\n", 26462306a36Sopenharmony_ci reason, arb_addr, 26562306a36Sopenharmony_ci cap_status & ARB_ERR_CAP_STATUS_WRITE ? 'W' : 'R', 26662306a36Sopenharmony_ci cap_status & ARB_ERR_CAP_STATUS_TIMEOUT ? "timeout" : "", 26762306a36Sopenharmony_ci m_name); 26862306a36Sopenharmony_ci 26962306a36Sopenharmony_ci /* clear the GISB error */ 27062306a36Sopenharmony_ci gisb_write(gdev, ARB_ERR_CAP_CLEAR, ARB_ERR_CAP_CLR); 27162306a36Sopenharmony_ci 27262306a36Sopenharmony_ci return 0; 27362306a36Sopenharmony_ci} 27462306a36Sopenharmony_ci 27562306a36Sopenharmony_ci#ifdef CONFIG_MIPS 27662306a36Sopenharmony_cistatic int brcmstb_bus_error_handler(struct pt_regs *regs, int is_fixup) 27762306a36Sopenharmony_ci{ 27862306a36Sopenharmony_ci int ret = 0; 27962306a36Sopenharmony_ci struct brcmstb_gisb_arb_device *gdev; 28062306a36Sopenharmony_ci u32 cap_status; 28162306a36Sopenharmony_ci 28262306a36Sopenharmony_ci list_for_each_entry(gdev, &brcmstb_gisb_arb_device_list, next) { 28362306a36Sopenharmony_ci cap_status = gisb_read(gdev, ARB_ERR_CAP_STATUS); 28462306a36Sopenharmony_ci 28562306a36Sopenharmony_ci /* Invalid captured address, bail out */ 28662306a36Sopenharmony_ci if (!(cap_status & ARB_ERR_CAP_STATUS_VALID)) { 28762306a36Sopenharmony_ci is_fixup = 1; 28862306a36Sopenharmony_ci goto out; 28962306a36Sopenharmony_ci } 29062306a36Sopenharmony_ci 29162306a36Sopenharmony_ci ret |= brcmstb_gisb_arb_decode_addr(gdev, "bus error"); 29262306a36Sopenharmony_ci } 29362306a36Sopenharmony_ciout: 29462306a36Sopenharmony_ci return is_fixup ? MIPS_BE_FIXUP : MIPS_BE_FATAL; 29562306a36Sopenharmony_ci} 29662306a36Sopenharmony_ci#endif 29762306a36Sopenharmony_ci 29862306a36Sopenharmony_cistatic irqreturn_t brcmstb_gisb_timeout_handler(int irq, void *dev_id) 29962306a36Sopenharmony_ci{ 30062306a36Sopenharmony_ci brcmstb_gisb_arb_decode_addr(dev_id, "timeout"); 30162306a36Sopenharmony_ci 30262306a36Sopenharmony_ci return IRQ_HANDLED; 30362306a36Sopenharmony_ci} 30462306a36Sopenharmony_ci 30562306a36Sopenharmony_cistatic irqreturn_t brcmstb_gisb_tea_handler(int irq, void *dev_id) 30662306a36Sopenharmony_ci{ 30762306a36Sopenharmony_ci brcmstb_gisb_arb_decode_addr(dev_id, "target abort"); 30862306a36Sopenharmony_ci 30962306a36Sopenharmony_ci return IRQ_HANDLED; 31062306a36Sopenharmony_ci} 31162306a36Sopenharmony_ci 31262306a36Sopenharmony_cistatic irqreturn_t brcmstb_gisb_bp_handler(int irq, void *dev_id) 31362306a36Sopenharmony_ci{ 31462306a36Sopenharmony_ci struct brcmstb_gisb_arb_device *gdev = dev_id; 31562306a36Sopenharmony_ci const char *m_name; 31662306a36Sopenharmony_ci u32 bp_status; 31762306a36Sopenharmony_ci u64 arb_addr; 31862306a36Sopenharmony_ci u32 master; 31962306a36Sopenharmony_ci char m_fmt[11]; 32062306a36Sopenharmony_ci 32162306a36Sopenharmony_ci bp_status = gisb_read(gdev, ARB_BP_CAP_STATUS); 32262306a36Sopenharmony_ci 32362306a36Sopenharmony_ci /* Invalid captured address, bail out */ 32462306a36Sopenharmony_ci if (!(bp_status & ARB_BP_CAP_STATUS_VALID)) 32562306a36Sopenharmony_ci return IRQ_HANDLED; 32662306a36Sopenharmony_ci 32762306a36Sopenharmony_ci /* Read the address and master */ 32862306a36Sopenharmony_ci arb_addr = gisb_read_bp_address(gdev); 32962306a36Sopenharmony_ci master = gisb_read(gdev, ARB_BP_CAP_MASTER); 33062306a36Sopenharmony_ci 33162306a36Sopenharmony_ci m_name = brcmstb_gisb_master_to_str(gdev, master); 33262306a36Sopenharmony_ci if (!m_name) { 33362306a36Sopenharmony_ci snprintf(m_fmt, sizeof(m_fmt), "0x%08x", master); 33462306a36Sopenharmony_ci m_name = m_fmt; 33562306a36Sopenharmony_ci } 33662306a36Sopenharmony_ci 33762306a36Sopenharmony_ci pr_crit("GISB: breakpoint at 0x%llx [%c], core: %s\n", 33862306a36Sopenharmony_ci arb_addr, bp_status & ARB_BP_CAP_STATUS_WRITE ? 'W' : 'R', 33962306a36Sopenharmony_ci m_name); 34062306a36Sopenharmony_ci 34162306a36Sopenharmony_ci /* clear the GISB error */ 34262306a36Sopenharmony_ci gisb_write(gdev, ARB_ERR_CAP_CLEAR, ARB_ERR_CAP_CLR); 34362306a36Sopenharmony_ci 34462306a36Sopenharmony_ci return IRQ_HANDLED; 34562306a36Sopenharmony_ci} 34662306a36Sopenharmony_ci 34762306a36Sopenharmony_ci/* 34862306a36Sopenharmony_ci * Dump out gisb errors on die or panic. 34962306a36Sopenharmony_ci */ 35062306a36Sopenharmony_cistatic int dump_gisb_error(struct notifier_block *self, unsigned long v, 35162306a36Sopenharmony_ci void *p); 35262306a36Sopenharmony_ci 35362306a36Sopenharmony_cistatic struct notifier_block gisb_die_notifier = { 35462306a36Sopenharmony_ci .notifier_call = dump_gisb_error, 35562306a36Sopenharmony_ci}; 35662306a36Sopenharmony_ci 35762306a36Sopenharmony_cistatic struct notifier_block gisb_panic_notifier = { 35862306a36Sopenharmony_ci .notifier_call = dump_gisb_error, 35962306a36Sopenharmony_ci}; 36062306a36Sopenharmony_ci 36162306a36Sopenharmony_cistatic int dump_gisb_error(struct notifier_block *self, unsigned long v, 36262306a36Sopenharmony_ci void *p) 36362306a36Sopenharmony_ci{ 36462306a36Sopenharmony_ci struct brcmstb_gisb_arb_device *gdev; 36562306a36Sopenharmony_ci const char *reason = "panic"; 36662306a36Sopenharmony_ci 36762306a36Sopenharmony_ci if (self == &gisb_die_notifier) 36862306a36Sopenharmony_ci reason = "die"; 36962306a36Sopenharmony_ci 37062306a36Sopenharmony_ci /* iterate over each GISB arb registered handlers */ 37162306a36Sopenharmony_ci list_for_each_entry(gdev, &brcmstb_gisb_arb_device_list, next) 37262306a36Sopenharmony_ci brcmstb_gisb_arb_decode_addr(gdev, reason); 37362306a36Sopenharmony_ci 37462306a36Sopenharmony_ci return NOTIFY_DONE; 37562306a36Sopenharmony_ci} 37662306a36Sopenharmony_ci 37762306a36Sopenharmony_cistatic DEVICE_ATTR(gisb_arb_timeout, S_IWUSR | S_IRUGO, 37862306a36Sopenharmony_ci gisb_arb_get_timeout, gisb_arb_set_timeout); 37962306a36Sopenharmony_ci 38062306a36Sopenharmony_cistatic struct attribute *gisb_arb_sysfs_attrs[] = { 38162306a36Sopenharmony_ci &dev_attr_gisb_arb_timeout.attr, 38262306a36Sopenharmony_ci NULL, 38362306a36Sopenharmony_ci}; 38462306a36Sopenharmony_ci 38562306a36Sopenharmony_cistatic struct attribute_group gisb_arb_sysfs_attr_group = { 38662306a36Sopenharmony_ci .attrs = gisb_arb_sysfs_attrs, 38762306a36Sopenharmony_ci}; 38862306a36Sopenharmony_ci 38962306a36Sopenharmony_cistatic const struct of_device_id brcmstb_gisb_arb_of_match[] = { 39062306a36Sopenharmony_ci { .compatible = "brcm,gisb-arb", .data = gisb_offsets_bcm7445 }, 39162306a36Sopenharmony_ci { .compatible = "brcm,bcm7445-gisb-arb", .data = gisb_offsets_bcm7445 }, 39262306a36Sopenharmony_ci { .compatible = "brcm,bcm7435-gisb-arb", .data = gisb_offsets_bcm7435 }, 39362306a36Sopenharmony_ci { .compatible = "brcm,bcm7400-gisb-arb", .data = gisb_offsets_bcm7400 }, 39462306a36Sopenharmony_ci { .compatible = "brcm,bcm7278-gisb-arb", .data = gisb_offsets_bcm7278 }, 39562306a36Sopenharmony_ci { .compatible = "brcm,bcm7038-gisb-arb", .data = gisb_offsets_bcm7038 }, 39662306a36Sopenharmony_ci { }, 39762306a36Sopenharmony_ci}; 39862306a36Sopenharmony_ci 39962306a36Sopenharmony_cistatic int __init brcmstb_gisb_arb_probe(struct platform_device *pdev) 40062306a36Sopenharmony_ci{ 40162306a36Sopenharmony_ci struct device_node *dn = pdev->dev.of_node; 40262306a36Sopenharmony_ci struct brcmstb_gisb_arb_device *gdev; 40362306a36Sopenharmony_ci const struct of_device_id *of_id; 40462306a36Sopenharmony_ci int err, timeout_irq, tea_irq, bp_irq; 40562306a36Sopenharmony_ci unsigned int num_masters, j = 0; 40662306a36Sopenharmony_ci int i, first, last; 40762306a36Sopenharmony_ci 40862306a36Sopenharmony_ci timeout_irq = platform_get_irq(pdev, 0); 40962306a36Sopenharmony_ci tea_irq = platform_get_irq(pdev, 1); 41062306a36Sopenharmony_ci bp_irq = platform_get_irq(pdev, 2); 41162306a36Sopenharmony_ci 41262306a36Sopenharmony_ci gdev = devm_kzalloc(&pdev->dev, sizeof(*gdev), GFP_KERNEL); 41362306a36Sopenharmony_ci if (!gdev) 41462306a36Sopenharmony_ci return -ENOMEM; 41562306a36Sopenharmony_ci 41662306a36Sopenharmony_ci mutex_init(&gdev->lock); 41762306a36Sopenharmony_ci INIT_LIST_HEAD(&gdev->next); 41862306a36Sopenharmony_ci 41962306a36Sopenharmony_ci gdev->base = devm_platform_get_and_ioremap_resource(pdev, 0, NULL); 42062306a36Sopenharmony_ci if (IS_ERR(gdev->base)) 42162306a36Sopenharmony_ci return PTR_ERR(gdev->base); 42262306a36Sopenharmony_ci 42362306a36Sopenharmony_ci of_id = of_match_node(brcmstb_gisb_arb_of_match, dn); 42462306a36Sopenharmony_ci if (!of_id) { 42562306a36Sopenharmony_ci pr_err("failed to look up compatible string\n"); 42662306a36Sopenharmony_ci return -EINVAL; 42762306a36Sopenharmony_ci } 42862306a36Sopenharmony_ci gdev->gisb_offsets = of_id->data; 42962306a36Sopenharmony_ci gdev->big_endian = of_device_is_big_endian(dn); 43062306a36Sopenharmony_ci 43162306a36Sopenharmony_ci err = devm_request_irq(&pdev->dev, timeout_irq, 43262306a36Sopenharmony_ci brcmstb_gisb_timeout_handler, 0, pdev->name, 43362306a36Sopenharmony_ci gdev); 43462306a36Sopenharmony_ci if (err < 0) 43562306a36Sopenharmony_ci return err; 43662306a36Sopenharmony_ci 43762306a36Sopenharmony_ci err = devm_request_irq(&pdev->dev, tea_irq, 43862306a36Sopenharmony_ci brcmstb_gisb_tea_handler, 0, pdev->name, 43962306a36Sopenharmony_ci gdev); 44062306a36Sopenharmony_ci if (err < 0) 44162306a36Sopenharmony_ci return err; 44262306a36Sopenharmony_ci 44362306a36Sopenharmony_ci /* Interrupt is optional */ 44462306a36Sopenharmony_ci if (bp_irq > 0) { 44562306a36Sopenharmony_ci err = devm_request_irq(&pdev->dev, bp_irq, 44662306a36Sopenharmony_ci brcmstb_gisb_bp_handler, 0, pdev->name, 44762306a36Sopenharmony_ci gdev); 44862306a36Sopenharmony_ci if (err < 0) 44962306a36Sopenharmony_ci return err; 45062306a36Sopenharmony_ci } 45162306a36Sopenharmony_ci 45262306a36Sopenharmony_ci /* If we do not have a valid mask, assume all masters are enabled */ 45362306a36Sopenharmony_ci if (of_property_read_u32(dn, "brcm,gisb-arb-master-mask", 45462306a36Sopenharmony_ci &gdev->valid_mask)) 45562306a36Sopenharmony_ci gdev->valid_mask = 0xffffffff; 45662306a36Sopenharmony_ci 45762306a36Sopenharmony_ci /* Proceed with reading the litteral names if we agree on the 45862306a36Sopenharmony_ci * number of masters 45962306a36Sopenharmony_ci */ 46062306a36Sopenharmony_ci num_masters = of_property_count_strings(dn, 46162306a36Sopenharmony_ci "brcm,gisb-arb-master-names"); 46262306a36Sopenharmony_ci if (hweight_long(gdev->valid_mask) == num_masters) { 46362306a36Sopenharmony_ci first = ffs(gdev->valid_mask) - 1; 46462306a36Sopenharmony_ci last = fls(gdev->valid_mask) - 1; 46562306a36Sopenharmony_ci 46662306a36Sopenharmony_ci for (i = first; i < last; i++) { 46762306a36Sopenharmony_ci if (!(gdev->valid_mask & BIT(i))) 46862306a36Sopenharmony_ci continue; 46962306a36Sopenharmony_ci 47062306a36Sopenharmony_ci of_property_read_string_index(dn, 47162306a36Sopenharmony_ci "brcm,gisb-arb-master-names", j, 47262306a36Sopenharmony_ci &gdev->master_names[i]); 47362306a36Sopenharmony_ci j++; 47462306a36Sopenharmony_ci } 47562306a36Sopenharmony_ci } 47662306a36Sopenharmony_ci 47762306a36Sopenharmony_ci err = sysfs_create_group(&pdev->dev.kobj, &gisb_arb_sysfs_attr_group); 47862306a36Sopenharmony_ci if (err) 47962306a36Sopenharmony_ci return err; 48062306a36Sopenharmony_ci 48162306a36Sopenharmony_ci platform_set_drvdata(pdev, gdev); 48262306a36Sopenharmony_ci 48362306a36Sopenharmony_ci list_add_tail(&gdev->next, &brcmstb_gisb_arb_device_list); 48462306a36Sopenharmony_ci 48562306a36Sopenharmony_ci#ifdef CONFIG_MIPS 48662306a36Sopenharmony_ci mips_set_be_handler(brcmstb_bus_error_handler); 48762306a36Sopenharmony_ci#endif 48862306a36Sopenharmony_ci 48962306a36Sopenharmony_ci if (list_is_singular(&brcmstb_gisb_arb_device_list)) { 49062306a36Sopenharmony_ci register_die_notifier(&gisb_die_notifier); 49162306a36Sopenharmony_ci atomic_notifier_chain_register(&panic_notifier_list, 49262306a36Sopenharmony_ci &gisb_panic_notifier); 49362306a36Sopenharmony_ci } 49462306a36Sopenharmony_ci 49562306a36Sopenharmony_ci dev_info(&pdev->dev, "registered irqs: %d, %d\n", 49662306a36Sopenharmony_ci timeout_irq, tea_irq); 49762306a36Sopenharmony_ci 49862306a36Sopenharmony_ci return 0; 49962306a36Sopenharmony_ci} 50062306a36Sopenharmony_ci 50162306a36Sopenharmony_ci#ifdef CONFIG_PM_SLEEP 50262306a36Sopenharmony_cistatic int brcmstb_gisb_arb_suspend(struct device *dev) 50362306a36Sopenharmony_ci{ 50462306a36Sopenharmony_ci struct brcmstb_gisb_arb_device *gdev = dev_get_drvdata(dev); 50562306a36Sopenharmony_ci 50662306a36Sopenharmony_ci gdev->saved_timeout = gisb_read(gdev, ARB_TIMER); 50762306a36Sopenharmony_ci 50862306a36Sopenharmony_ci return 0; 50962306a36Sopenharmony_ci} 51062306a36Sopenharmony_ci 51162306a36Sopenharmony_ci/* Make sure we provide the same timeout value that was configured before, and 51262306a36Sopenharmony_ci * do this before the GISB timeout interrupt handler has any chance to run. 51362306a36Sopenharmony_ci */ 51462306a36Sopenharmony_cistatic int brcmstb_gisb_arb_resume_noirq(struct device *dev) 51562306a36Sopenharmony_ci{ 51662306a36Sopenharmony_ci struct brcmstb_gisb_arb_device *gdev = dev_get_drvdata(dev); 51762306a36Sopenharmony_ci 51862306a36Sopenharmony_ci gisb_write(gdev, gdev->saved_timeout, ARB_TIMER); 51962306a36Sopenharmony_ci 52062306a36Sopenharmony_ci return 0; 52162306a36Sopenharmony_ci} 52262306a36Sopenharmony_ci#else 52362306a36Sopenharmony_ci#define brcmstb_gisb_arb_suspend NULL 52462306a36Sopenharmony_ci#define brcmstb_gisb_arb_resume_noirq NULL 52562306a36Sopenharmony_ci#endif 52662306a36Sopenharmony_ci 52762306a36Sopenharmony_cistatic const struct dev_pm_ops brcmstb_gisb_arb_pm_ops = { 52862306a36Sopenharmony_ci .suspend = brcmstb_gisb_arb_suspend, 52962306a36Sopenharmony_ci .resume_noirq = brcmstb_gisb_arb_resume_noirq, 53062306a36Sopenharmony_ci}; 53162306a36Sopenharmony_ci 53262306a36Sopenharmony_cistatic struct platform_driver brcmstb_gisb_arb_driver = { 53362306a36Sopenharmony_ci .driver = { 53462306a36Sopenharmony_ci .name = "brcm-gisb-arb", 53562306a36Sopenharmony_ci .of_match_table = brcmstb_gisb_arb_of_match, 53662306a36Sopenharmony_ci .pm = &brcmstb_gisb_arb_pm_ops, 53762306a36Sopenharmony_ci }, 53862306a36Sopenharmony_ci}; 53962306a36Sopenharmony_ci 54062306a36Sopenharmony_cistatic int __init brcm_gisb_driver_init(void) 54162306a36Sopenharmony_ci{ 54262306a36Sopenharmony_ci return platform_driver_probe(&brcmstb_gisb_arb_driver, 54362306a36Sopenharmony_ci brcmstb_gisb_arb_probe); 54462306a36Sopenharmony_ci} 54562306a36Sopenharmony_ci 54662306a36Sopenharmony_cimodule_init(brcm_gisb_driver_init); 54762306a36Sopenharmony_ci 54862306a36Sopenharmony_ciMODULE_AUTHOR("Broadcom"); 54962306a36Sopenharmony_ciMODULE_DESCRIPTION("Broadcom STB GISB arbiter driver"); 55062306a36Sopenharmony_ciMODULE_LICENSE("GPL v2"); 551