2 * Cell MIC driver for ECC counting
4 * Copyright 2007 Benjamin Herrenschmidt, IBM Corp.
5 * <benh@kernel.crashing.org>
7 * This file may be distributed under the terms of the
8 * GNU General Public License.
12 #include <linux/edac.h>
13 #include <linux/module.h>
14 #include <linux/init.h>
15 #include <linux/platform_device.h>
16 #include <linux/stop_machine.h>
18 #include <asm/machdep.h>
19 #include <asm/cell-regs.h>
21 #include "edac_core.h"
25 struct cbe_mic_tm_regs __iomem
*regs
;
33 static void cell_edac_count_ce(struct mem_ctl_info
*mci
, int chan
, u64 ar
)
35 struct cell_edac_priv
*priv
= mci
->pvt_info
;
36 struct csrow_info
*csrow
= mci
->csrows
[0];
37 unsigned long address
, pfn
, offset
, syndrome
;
39 dev_dbg(mci
->pdev
, "ECC CE err on node %d, channel %d, ar = 0x%016llx\n",
40 priv
->node
, chan
, ar
);
42 /* Address decoding is likely a bit bogus, to dbl check */
43 address
= (ar
& 0xffffffffe0000000ul
) >> 29;
44 if (priv
->chanmask
== 0x3)
45 address
= (address
<< 1) | chan
;
46 pfn
= address
>> PAGE_SHIFT
;
47 offset
= address
& ~PAGE_MASK
;
48 syndrome
= (ar
& 0x000000001fe00000ul
) >> 21;
50 /* TODO: Decoding of the error address */
51 edac_mc_handle_error(HW_EVENT_ERR_CORRECTED
, mci
, 1,
52 csrow
->first_page
+ pfn
, offset
, syndrome
,
56 static void cell_edac_count_ue(struct mem_ctl_info
*mci
, int chan
, u64 ar
)
58 struct cell_edac_priv
*priv
= mci
->pvt_info
;
59 struct csrow_info
*csrow
= mci
->csrows
[0];
60 unsigned long address
, pfn
, offset
;
62 dev_dbg(mci
->pdev
, "ECC UE err on node %d, channel %d, ar = 0x%016llx\n",
63 priv
->node
, chan
, ar
);
65 /* Address decoding is likely a bit bogus, to dbl check */
66 address
= (ar
& 0xffffffffe0000000ul
) >> 29;
67 if (priv
->chanmask
== 0x3)
68 address
= (address
<< 1) | chan
;
69 pfn
= address
>> PAGE_SHIFT
;
70 offset
= address
& ~PAGE_MASK
;
72 /* TODO: Decoding of the error address */
73 edac_mc_handle_error(HW_EVENT_ERR_UNCORRECTED
, mci
, 1,
74 csrow
->first_page
+ pfn
, offset
, 0,
78 static void cell_edac_check(struct mem_ctl_info
*mci
)
80 struct cell_edac_priv
*priv
= mci
->pvt_info
;
81 u64 fir
, addreg
, clear
= 0;
83 fir
= in_be64(&priv
->regs
->mic_fir
);
85 if (fir
!= priv
->prev_fir
) {
86 dev_dbg(mci
->pdev
, "fir change : 0x%016lx\n", fir
);
90 if ((priv
->chanmask
& 0x1) && (fir
& CBE_MIC_FIR_ECC_SINGLE_0_ERR
)) {
91 addreg
= in_be64(&priv
->regs
->mic_df_ecc_address_0
);
92 clear
|= CBE_MIC_FIR_ECC_SINGLE_0_RESET
;
93 cell_edac_count_ce(mci
, 0, addreg
);
95 if ((priv
->chanmask
& 0x2) && (fir
& CBE_MIC_FIR_ECC_SINGLE_1_ERR
)) {
96 addreg
= in_be64(&priv
->regs
->mic_df_ecc_address_1
);
97 clear
|= CBE_MIC_FIR_ECC_SINGLE_1_RESET
;
98 cell_edac_count_ce(mci
, 1, addreg
);
100 if ((priv
->chanmask
& 0x1) && (fir
& CBE_MIC_FIR_ECC_MULTI_0_ERR
)) {
101 addreg
= in_be64(&priv
->regs
->mic_df_ecc_address_0
);
102 clear
|= CBE_MIC_FIR_ECC_MULTI_0_RESET
;
103 cell_edac_count_ue(mci
, 0, addreg
);
105 if ((priv
->chanmask
& 0x2) && (fir
& CBE_MIC_FIR_ECC_MULTI_1_ERR
)) {
106 addreg
= in_be64(&priv
->regs
->mic_df_ecc_address_1
);
107 clear
|= CBE_MIC_FIR_ECC_MULTI_1_RESET
;
108 cell_edac_count_ue(mci
, 1, addreg
);
111 /* The procedure for clearing FIR bits is a bit ... weird */
113 fir
&= ~(CBE_MIC_FIR_ECC_ERR_MASK
| CBE_MIC_FIR_ECC_SET_MASK
);
114 fir
|= CBE_MIC_FIR_ECC_RESET_MASK
;
116 out_be64(&priv
->regs
->mic_fir
, fir
);
117 (void)in_be64(&priv
->regs
->mic_fir
);
121 fir
= in_be64(&priv
->regs
->mic_fir
);
122 dev_dbg(mci
->pdev
, "fir clear : 0x%016lx\n", fir
);
127 static void cell_edac_init_csrows(struct mem_ctl_info
*mci
)
129 struct csrow_info
*csrow
= mci
->csrows
[0];
130 struct dimm_info
*dimm
;
131 struct cell_edac_priv
*priv
= mci
->pvt_info
;
132 struct device_node
*np
;
137 (np
= of_find_node_by_name(np
, "memory")) != NULL
;) {
140 /* We "know" that the Cell firmware only creates one entry
141 * in the "memory" nodes. If that changes, this code will
142 * need to be adapted.
144 if (of_address_to_resource(np
, 0, &r
))
146 if (of_node_to_nid(np
) != priv
->node
)
148 csrow
->first_page
= r
.start
>> PAGE_SHIFT
;
149 nr_pages
= resource_size(&r
) >> PAGE_SHIFT
;
150 csrow
->last_page
= csrow
->first_page
+ nr_pages
- 1;
152 for (j
= 0; j
< csrow
->nr_channels
; j
++) {
153 dimm
= csrow
->channels
[j
]->dimm
;
154 dimm
->mtype
= MEM_XDR
;
155 dimm
->edac_mode
= EDAC_SECDED
;
156 dimm
->nr_pages
= nr_pages
/ csrow
->nr_channels
;
159 "Initialized on node %d, chanmask=0x%x,"
160 " first_page=0x%lx, nr_pages=0x%x\n",
161 priv
->node
, priv
->chanmask
,
162 csrow
->first_page
, nr_pages
);
167 static int cell_edac_probe(struct platform_device
*pdev
)
169 struct cbe_mic_tm_regs __iomem
*regs
;
170 struct mem_ctl_info
*mci
;
171 struct edac_mc_layer layers
[2];
172 struct cell_edac_priv
*priv
;
174 int rc
, chanmask
, num_chans
;
176 regs
= cbe_get_cpu_mic_tm_regs(cbe_node_to_cpu(pdev
->id
));
180 edac_op_state
= EDAC_OPSTATE_POLL
;
182 /* Get channel population */
183 reg
= in_be64(®s
->mic_mnt_cfg
);
184 dev_dbg(&pdev
->dev
, "MIC_MNT_CFG = 0x%016llx\n", reg
);
186 if (reg
& CBE_MIC_MNT_CFG_CHAN_0_POP
)
188 if (reg
& CBE_MIC_MNT_CFG_CHAN_1_POP
)
192 "Yuck ! No channel populated ? Aborting !\n");
195 dev_dbg(&pdev
->dev
, "Initial FIR = 0x%016llx\n",
196 in_be64(®s
->mic_fir
));
198 /* Allocate & init EDAC MC data structure */
199 num_chans
= chanmask
== 3 ? 2 : 1;
201 layers
[0].type
= EDAC_MC_LAYER_CHIP_SELECT
;
203 layers
[0].is_virt_csrow
= true;
204 layers
[1].type
= EDAC_MC_LAYER_CHANNEL
;
205 layers
[1].size
= num_chans
;
206 layers
[1].is_virt_csrow
= false;
207 mci
= edac_mc_alloc(pdev
->id
, ARRAY_SIZE(layers
), layers
,
208 sizeof(struct cell_edac_priv
));
211 priv
= mci
->pvt_info
;
213 priv
->node
= pdev
->id
;
214 priv
->chanmask
= chanmask
;
215 mci
->pdev
= &pdev
->dev
;
216 mci
->mtype_cap
= MEM_FLAG_XDR
;
217 mci
->edac_ctl_cap
= EDAC_FLAG_NONE
| EDAC_FLAG_EC
| EDAC_FLAG_SECDED
;
218 mci
->edac_cap
= EDAC_FLAG_EC
| EDAC_FLAG_SECDED
;
219 mci
->mod_name
= "cell_edac";
220 mci
->ctl_name
= "MIC";
221 mci
->dev_name
= dev_name(&pdev
->dev
);
222 mci
->edac_check
= cell_edac_check
;
223 cell_edac_init_csrows(mci
);
225 /* Register with EDAC core */
226 rc
= edac_mc_add_mc(mci
);
228 dev_err(&pdev
->dev
, "failed to register with EDAC core\n");
236 static int cell_edac_remove(struct platform_device
*pdev
)
238 struct mem_ctl_info
*mci
= edac_mc_del_mc(&pdev
->dev
);
244 static struct platform_driver cell_edac_driver
= {
247 .owner
= THIS_MODULE
,
249 .probe
= cell_edac_probe
,
250 .remove
= cell_edac_remove
,
253 static int __init
cell_edac_init(void)
255 /* Sanity check registers data structure */
256 BUILD_BUG_ON(offsetof(struct cbe_mic_tm_regs
,
257 mic_df_ecc_address_0
) != 0xf8);
258 BUILD_BUG_ON(offsetof(struct cbe_mic_tm_regs
,
259 mic_df_ecc_address_1
) != 0x1b8);
260 BUILD_BUG_ON(offsetof(struct cbe_mic_tm_regs
,
261 mic_df_config
) != 0x218);
262 BUILD_BUG_ON(offsetof(struct cbe_mic_tm_regs
,
264 BUILD_BUG_ON(offsetof(struct cbe_mic_tm_regs
,
265 mic_mnt_cfg
) != 0x210);
266 BUILD_BUG_ON(offsetof(struct cbe_mic_tm_regs
,
269 return platform_driver_register(&cell_edac_driver
);
272 static void __exit
cell_edac_exit(void)
274 platform_driver_unregister(&cell_edac_driver
);
277 module_init(cell_edac_init
);
278 module_exit(cell_edac_exit
);
280 MODULE_LICENSE("GPL");
281 MODULE_AUTHOR("Benjamin Herrenschmidt <benh@kernel.crashing.org>");
282 MODULE_DESCRIPTION("ECC counting for Cell MIC");