aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/edac
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/edac')
-rw-r--r--drivers/edac/highbank_mc_edac.c55
1 files changed, 20 insertions, 35 deletions
diff --git a/drivers/edac/highbank_mc_edac.c b/drivers/edac/highbank_mc_edac.c
index 82211915e3b0..f784de1dc793 100644
--- a/drivers/edac/highbank_mc_edac.c
+++ b/drivers/edac/highbank_mc_edac.c
@@ -95,50 +95,34 @@ static irqreturn_t highbank_mc_err_handler(int irq, void *dev_id)
95 return IRQ_HANDLED; 95 return IRQ_HANDLED;
96} 96}
97 97
98#ifdef CONFIG_EDAC_DEBUG 98static void highbank_mc_err_inject(struct mem_ctl_info *mci, u8 synd)
99static ssize_t highbank_mc_err_inject_write(struct file *file,
100 const char __user *data,
101 size_t count, loff_t *ppos)
102{ 99{
103 struct mem_ctl_info *mci = file->private_data;
104 struct hb_mc_drvdata *pdata = mci->pvt_info; 100 struct hb_mc_drvdata *pdata = mci->pvt_info;
105 char buf[32];
106 size_t buf_size;
107 u32 reg; 101 u32 reg;
102
103 reg = readl(pdata->mc_err_base + HB_DDR_ECC_OPT);
104 reg &= HB_DDR_ECC_OPT_MODE_MASK;
105 reg |= (synd << HB_DDR_ECC_OPT_XOR_SHIFT) | HB_DDR_ECC_OPT_FWC;
106 writel(reg, pdata->mc_err_base + HB_DDR_ECC_OPT);
107}
108
109#define to_mci(k) container_of(k, struct mem_ctl_info, dev)
110
111static ssize_t highbank_mc_inject_ctrl(struct device *dev,
112 struct device_attribute *attr, const char *buf, size_t count)
113{
114 struct mem_ctl_info *mci = to_mci(dev);
108 u8 synd; 115 u8 synd;
109 116
110 buf_size = min(count, (sizeof(buf)-1)); 117 if (kstrtou8(buf, 16, &synd))
111 if (copy_from_user(buf, data, buf_size)) 118 return -EINVAL;
112 return -EFAULT;
113 buf[buf_size] = 0;
114 119
115 if (!kstrtou8(buf, 16, &synd)) { 120 highbank_mc_err_inject(mci, synd);
116 reg = readl(pdata->mc_err_base + HB_DDR_ECC_OPT);
117 reg &= HB_DDR_ECC_OPT_MODE_MASK;
118 reg |= (synd << HB_DDR_ECC_OPT_XOR_SHIFT) | HB_DDR_ECC_OPT_FWC;
119 writel(reg, pdata->mc_err_base + HB_DDR_ECC_OPT);
120 }
121 121
122 return count; 122 return count;
123} 123}
124 124
125static const struct file_operations highbank_mc_debug_inject_fops = { 125static DEVICE_ATTR(inject_ctrl, S_IWUSR, NULL, highbank_mc_inject_ctrl);
126 .open = simple_open,
127 .write = highbank_mc_err_inject_write,
128 .llseek = generic_file_llseek,
129};
130
131static void highbank_mc_create_debugfs_nodes(struct mem_ctl_info *mci)
132{
133 if (mci->debugfs)
134 debugfs_create_file("inject_ctrl", S_IWUSR, mci->debugfs, mci,
135 &highbank_mc_debug_inject_fops);
136;
137}
138#else
139static void highbank_mc_create_debugfs_nodes(struct mem_ctl_info *mci)
140{}
141#endif
142 126
143struct hb_mc_settings { 127struct hb_mc_settings {
144 int err_offset; 128 int err_offset;
@@ -259,7 +243,7 @@ static int highbank_mc_probe(struct platform_device *pdev)
259 goto err2; 243 goto err2;
260 } 244 }
261 245
262 highbank_mc_create_debugfs_nodes(mci); 246 device_create_file(&mci->dev, &dev_attr_inject_ctrl);
263 247
264 devres_close_group(&pdev->dev, NULL); 248 devres_close_group(&pdev->dev, NULL);
265 return 0; 249 return 0;
@@ -275,6 +259,7 @@ static int highbank_mc_remove(struct platform_device *pdev)
275{ 259{
276 struct mem_ctl_info *mci = platform_get_drvdata(pdev); 260 struct mem_ctl_info *mci = platform_get_drvdata(pdev);
277 261
262 device_remove_file(&mci->dev, &dev_attr_inject_ctrl);
278 edac_mc_del_mc(&pdev->dev); 263 edac_mc_del_mc(&pdev->dev);
279 edac_mc_free(mci); 264 edac_mc_free(mci);
280 return 0; 265 return 0;