mv64x60_edac.c 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891
  1. /*
  2. * Marvell MV64x60 Memory Controller kernel module for PPC platforms
  3. *
  4. * Author: Dave Jiang <djiang@mvista.com>
  5. *
  6. * 2006-2007 (c) MontaVista Software, Inc. This file is licensed under
  7. * the terms of the GNU General Public License version 2. This program
  8. * is licensed "as is" without any warranty of any kind, whether express
  9. * or implied.
  10. *
  11. */
  12. #include <linux/module.h>
  13. #include <linux/init.h>
  14. #include <linux/interrupt.h>
  15. #include <linux/io.h>
  16. #include <linux/edac.h>
  17. #include <linux/gfp.h>
  18. #include "edac_core.h"
  19. #include "edac_module.h"
  20. #include "mv64x60_edac.h"
  21. static const char *mv64x60_ctl_name = "MV64x60";
  22. static int edac_dev_idx;
  23. static int edac_pci_idx;
  24. static int edac_mc_idx;
  25. /*********************** PCI err device **********************************/
  26. #ifdef CONFIG_PCI
  27. static void mv64x60_pci_check(struct edac_pci_ctl_info *pci)
  28. {
  29. struct mv64x60_pci_pdata *pdata = pci->pvt_info;
  30. u32 cause;
  31. cause = in_le32(pdata->pci_vbase + MV64X60_PCI_ERROR_CAUSE);
  32. if (!cause)
  33. return;
  34. printk(KERN_ERR "Error in PCI %d Interface\n", pdata->pci_hose);
  35. printk(KERN_ERR "Cause register: 0x%08x\n", cause);
  36. printk(KERN_ERR "Address Low: 0x%08x\n",
  37. in_le32(pdata->pci_vbase + MV64X60_PCI_ERROR_ADDR_LO));
  38. printk(KERN_ERR "Address High: 0x%08x\n",
  39. in_le32(pdata->pci_vbase + MV64X60_PCI_ERROR_ADDR_HI));
  40. printk(KERN_ERR "Attribute: 0x%08x\n",
  41. in_le32(pdata->pci_vbase + MV64X60_PCI_ERROR_ATTR));
  42. printk(KERN_ERR "Command: 0x%08x\n",
  43. in_le32(pdata->pci_vbase + MV64X60_PCI_ERROR_CMD));
  44. out_le32(pdata->pci_vbase + MV64X60_PCI_ERROR_CAUSE, ~cause);
  45. if (cause & MV64X60_PCI_PE_MASK)
  46. edac_pci_handle_pe(pci, pci->ctl_name);
  47. if (!(cause & MV64X60_PCI_PE_MASK))
  48. edac_pci_handle_npe(pci, pci->ctl_name);
  49. }
  50. static irqreturn_t mv64x60_pci_isr(int irq, void *dev_id)
  51. {
  52. struct edac_pci_ctl_info *pci = dev_id;
  53. struct mv64x60_pci_pdata *pdata = pci->pvt_info;
  54. u32 val;
  55. val = in_le32(pdata->pci_vbase + MV64X60_PCI_ERROR_CAUSE);
  56. if (!val)
  57. return IRQ_NONE;
  58. mv64x60_pci_check(pci);
  59. return IRQ_HANDLED;
  60. }
  61. /*
  62. * Bit 0 of MV64x60_PCIx_ERR_MASK does not exist on the 64360 and because of
  63. * errata FEr-#11 and FEr-##16 for the 64460, it should be 0 on that chip as
  64. * well. IOW, don't set bit 0.
  65. */
  66. /* Erratum FEr PCI-#16: clear bit 0 of PCI SERRn Mask reg. */
  67. static int __init mv64x60_pci_fixup(struct platform_device *pdev)
  68. {
  69. struct resource *r;
  70. void __iomem *pci_serr;
  71. r = platform_get_resource(pdev, IORESOURCE_MEM, 1);
  72. if (!r) {
  73. printk(KERN_ERR "%s: Unable to get resource for "
  74. "PCI err regs\n", __func__);
  75. return -ENOENT;
  76. }
  77. pci_serr = ioremap(r->start, resource_size(r));
  78. if (!pci_serr)
  79. return -ENOMEM;
  80. out_le32(pci_serr, in_le32(pci_serr) & ~0x1);
  81. iounmap(pci_serr);
  82. return 0;
  83. }
  84. static int __devinit mv64x60_pci_err_probe(struct platform_device *pdev)
  85. {
  86. struct edac_pci_ctl_info *pci;
  87. struct mv64x60_pci_pdata *pdata;
  88. struct resource *r;
  89. int res = 0;
  90. if (!devres_open_group(&pdev->dev, mv64x60_pci_err_probe, GFP_KERNEL))
  91. return -ENOMEM;
  92. pci = edac_pci_alloc_ctl_info(sizeof(*pdata), "mv64x60_pci_err");
  93. if (!pci)
  94. return -ENOMEM;
  95. pdata = pci->pvt_info;
  96. pdata->pci_hose = pdev->id;
  97. pdata->name = "mpc85xx_pci_err";
  98. pdata->irq = NO_IRQ;
  99. platform_set_drvdata(pdev, pci);
  100. pci->dev = &pdev->dev;
  101. pci->dev_name = dev_name(&pdev->dev);
  102. pci->mod_name = EDAC_MOD_STR;
  103. pci->ctl_name = pdata->name;
  104. if (edac_op_state == EDAC_OPSTATE_POLL)
  105. pci->edac_check = mv64x60_pci_check;
  106. pdata->edac_idx = edac_pci_idx++;
  107. r = platform_get_resource(pdev, IORESOURCE_MEM, 0);
  108. if (!r) {
  109. printk(KERN_ERR "%s: Unable to get resource for "
  110. "PCI err regs\n", __func__);
  111. res = -ENOENT;
  112. goto err;
  113. }
  114. if (!devm_request_mem_region(&pdev->dev,
  115. r->start,
  116. resource_size(r),
  117. pdata->name)) {
  118. printk(KERN_ERR "%s: Error while requesting mem region\n",
  119. __func__);
  120. res = -EBUSY;
  121. goto err;
  122. }
  123. pdata->pci_vbase = devm_ioremap(&pdev->dev,
  124. r->start,
  125. resource_size(r));
  126. if (!pdata->pci_vbase) {
  127. printk(KERN_ERR "%s: Unable to setup PCI err regs\n", __func__);
  128. res = -ENOMEM;
  129. goto err;
  130. }
  131. res = mv64x60_pci_fixup(pdev);
  132. if (res < 0) {
  133. printk(KERN_ERR "%s: PCI fixup failed\n", __func__);
  134. goto err;
  135. }
  136. out_le32(pdata->pci_vbase + MV64X60_PCI_ERROR_CAUSE, 0);
  137. out_le32(pdata->pci_vbase + MV64X60_PCI_ERROR_MASK, 0);
  138. out_le32(pdata->pci_vbase + MV64X60_PCI_ERROR_MASK,
  139. MV64X60_PCIx_ERR_MASK_VAL);
  140. if (edac_pci_add_device(pci, pdata->edac_idx) > 0) {
  141. debugf3("%s(): failed edac_pci_add_device()\n", __func__);
  142. goto err;
  143. }
  144. if (edac_op_state == EDAC_OPSTATE_INT) {
  145. pdata->irq = platform_get_irq(pdev, 0);
  146. res = devm_request_irq(&pdev->dev,
  147. pdata->irq,
  148. mv64x60_pci_isr,
  149. IRQF_DISABLED,
  150. "[EDAC] PCI err",
  151. pci);
  152. if (res < 0) {
  153. printk(KERN_ERR "%s: Unable to request irq %d for "
  154. "MV64x60 PCI ERR\n", __func__, pdata->irq);
  155. res = -ENODEV;
  156. goto err2;
  157. }
  158. printk(KERN_INFO EDAC_MOD_STR " acquired irq %d for PCI Err\n",
  159. pdata->irq);
  160. }
  161. devres_remove_group(&pdev->dev, mv64x60_pci_err_probe);
  162. /* get this far and it's successful */
  163. debugf3("%s(): success\n", __func__);
  164. return 0;
  165. err2:
  166. edac_pci_del_device(&pdev->dev);
  167. err:
  168. edac_pci_free_ctl_info(pci);
  169. devres_release_group(&pdev->dev, mv64x60_pci_err_probe);
  170. return res;
  171. }
  172. static int mv64x60_pci_err_remove(struct platform_device *pdev)
  173. {
  174. struct edac_pci_ctl_info *pci = platform_get_drvdata(pdev);
  175. debugf0("%s()\n", __func__);
  176. edac_pci_del_device(&pdev->dev);
  177. edac_pci_free_ctl_info(pci);
  178. return 0;
  179. }
  180. static struct platform_driver mv64x60_pci_err_driver = {
  181. .probe = mv64x60_pci_err_probe,
  182. .remove = __devexit_p(mv64x60_pci_err_remove),
  183. .driver = {
  184. .name = "mv64x60_pci_err",
  185. }
  186. };
  187. #endif /* CONFIG_PCI */
  188. /*********************** SRAM err device **********************************/
  189. static void mv64x60_sram_check(struct edac_device_ctl_info *edac_dev)
  190. {
  191. struct mv64x60_sram_pdata *pdata = edac_dev->pvt_info;
  192. u32 cause;
  193. cause = in_le32(pdata->sram_vbase + MV64X60_SRAM_ERR_CAUSE);
  194. if (!cause)
  195. return;
  196. printk(KERN_ERR "Error in internal SRAM\n");
  197. printk(KERN_ERR "Cause register: 0x%08x\n", cause);
  198. printk(KERN_ERR "Address Low: 0x%08x\n",
  199. in_le32(pdata->sram_vbase + MV64X60_SRAM_ERR_ADDR_LO));
  200. printk(KERN_ERR "Address High: 0x%08x\n",
  201. in_le32(pdata->sram_vbase + MV64X60_SRAM_ERR_ADDR_HI));
  202. printk(KERN_ERR "Data Low: 0x%08x\n",
  203. in_le32(pdata->sram_vbase + MV64X60_SRAM_ERR_DATA_LO));
  204. printk(KERN_ERR "Data High: 0x%08x\n",
  205. in_le32(pdata->sram_vbase + MV64X60_SRAM_ERR_DATA_HI));
  206. printk(KERN_ERR "Parity: 0x%08x\n",
  207. in_le32(pdata->sram_vbase + MV64X60_SRAM_ERR_PARITY));
  208. out_le32(pdata->sram_vbase + MV64X60_SRAM_ERR_CAUSE, 0);
  209. edac_device_handle_ue(edac_dev, 0, 0, edac_dev->ctl_name);
  210. }
  211. static irqreturn_t mv64x60_sram_isr(int irq, void *dev_id)
  212. {
  213. struct edac_device_ctl_info *edac_dev = dev_id;
  214. struct mv64x60_sram_pdata *pdata = edac_dev->pvt_info;
  215. u32 cause;
  216. cause = in_le32(pdata->sram_vbase + MV64X60_SRAM_ERR_CAUSE);
  217. if (!cause)
  218. return IRQ_NONE;
  219. mv64x60_sram_check(edac_dev);
  220. return IRQ_HANDLED;
  221. }
  222. static int __devinit mv64x60_sram_err_probe(struct platform_device *pdev)
  223. {
  224. struct edac_device_ctl_info *edac_dev;
  225. struct mv64x60_sram_pdata *pdata;
  226. struct resource *r;
  227. int res = 0;
  228. if (!devres_open_group(&pdev->dev, mv64x60_sram_err_probe, GFP_KERNEL))
  229. return -ENOMEM;
  230. edac_dev = edac_device_alloc_ctl_info(sizeof(*pdata),
  231. "sram", 1, NULL, 0, 0, NULL, 0,
  232. edac_dev_idx);
  233. if (!edac_dev) {
  234. devres_release_group(&pdev->dev, mv64x60_sram_err_probe);
  235. return -ENOMEM;
  236. }
  237. pdata = edac_dev->pvt_info;
  238. pdata->name = "mv64x60_sram_err";
  239. pdata->irq = NO_IRQ;
  240. edac_dev->dev = &pdev->dev;
  241. platform_set_drvdata(pdev, edac_dev);
  242. edac_dev->dev_name = dev_name(&pdev->dev);
  243. r = platform_get_resource(pdev, IORESOURCE_MEM, 0);
  244. if (!r) {
  245. printk(KERN_ERR "%s: Unable to get resource for "
  246. "SRAM err regs\n", __func__);
  247. res = -ENOENT;
  248. goto err;
  249. }
  250. if (!devm_request_mem_region(&pdev->dev,
  251. r->start,
  252. resource_size(r),
  253. pdata->name)) {
  254. printk(KERN_ERR "%s: Error while request mem region\n",
  255. __func__);
  256. res = -EBUSY;
  257. goto err;
  258. }
  259. pdata->sram_vbase = devm_ioremap(&pdev->dev,
  260. r->start,
  261. resource_size(r));
  262. if (!pdata->sram_vbase) {
  263. printk(KERN_ERR "%s: Unable to setup SRAM err regs\n",
  264. __func__);
  265. res = -ENOMEM;
  266. goto err;
  267. }
  268. /* setup SRAM err registers */
  269. out_le32(pdata->sram_vbase + MV64X60_SRAM_ERR_CAUSE, 0);
  270. edac_dev->mod_name = EDAC_MOD_STR;
  271. edac_dev->ctl_name = pdata->name;
  272. if (edac_op_state == EDAC_OPSTATE_POLL)
  273. edac_dev->edac_check = mv64x60_sram_check;
  274. pdata->edac_idx = edac_dev_idx++;
  275. if (edac_device_add_device(edac_dev) > 0) {
  276. debugf3("%s(): failed edac_device_add_device()\n", __func__);
  277. goto err;
  278. }
  279. if (edac_op_state == EDAC_OPSTATE_INT) {
  280. pdata->irq = platform_get_irq(pdev, 0);
  281. res = devm_request_irq(&pdev->dev,
  282. pdata->irq,
  283. mv64x60_sram_isr,
  284. IRQF_DISABLED,
  285. "[EDAC] SRAM err",
  286. edac_dev);
  287. if (res < 0) {
  288. printk(KERN_ERR
  289. "%s: Unable to request irq %d for "
  290. "MV64x60 SRAM ERR\n", __func__, pdata->irq);
  291. res = -ENODEV;
  292. goto err2;
  293. }
  294. printk(KERN_INFO EDAC_MOD_STR " acquired irq %d for SRAM Err\n",
  295. pdata->irq);
  296. }
  297. devres_remove_group(&pdev->dev, mv64x60_sram_err_probe);
  298. /* get this far and it's successful */
  299. debugf3("%s(): success\n", __func__);
  300. return 0;
  301. err2:
  302. edac_device_del_device(&pdev->dev);
  303. err:
  304. devres_release_group(&pdev->dev, mv64x60_sram_err_probe);
  305. edac_device_free_ctl_info(edac_dev);
  306. return res;
  307. }
  308. static int mv64x60_sram_err_remove(struct platform_device *pdev)
  309. {
  310. struct edac_device_ctl_info *edac_dev = platform_get_drvdata(pdev);
  311. debugf0("%s()\n", __func__);
  312. edac_device_del_device(&pdev->dev);
  313. edac_device_free_ctl_info(edac_dev);
  314. return 0;
  315. }
  316. static struct platform_driver mv64x60_sram_err_driver = {
  317. .probe = mv64x60_sram_err_probe,
  318. .remove = mv64x60_sram_err_remove,
  319. .driver = {
  320. .name = "mv64x60_sram_err",
  321. }
  322. };
  323. /*********************** CPU err device **********************************/
  324. static void mv64x60_cpu_check(struct edac_device_ctl_info *edac_dev)
  325. {
  326. struct mv64x60_cpu_pdata *pdata = edac_dev->pvt_info;
  327. u32 cause;
  328. cause = in_le32(pdata->cpu_vbase[1] + MV64x60_CPU_ERR_CAUSE) &
  329. MV64x60_CPU_CAUSE_MASK;
  330. if (!cause)
  331. return;
  332. printk(KERN_ERR "Error on CPU interface\n");
  333. printk(KERN_ERR "Cause register: 0x%08x\n", cause);
  334. printk(KERN_ERR "Address Low: 0x%08x\n",
  335. in_le32(pdata->cpu_vbase[0] + MV64x60_CPU_ERR_ADDR_LO));
  336. printk(KERN_ERR "Address High: 0x%08x\n",
  337. in_le32(pdata->cpu_vbase[0] + MV64x60_CPU_ERR_ADDR_HI));
  338. printk(KERN_ERR "Data Low: 0x%08x\n",
  339. in_le32(pdata->cpu_vbase[1] + MV64x60_CPU_ERR_DATA_LO));
  340. printk(KERN_ERR "Data High: 0x%08x\n",
  341. in_le32(pdata->cpu_vbase[1] + MV64x60_CPU_ERR_DATA_HI));
  342. printk(KERN_ERR "Parity: 0x%08x\n",
  343. in_le32(pdata->cpu_vbase[1] + MV64x60_CPU_ERR_PARITY));
  344. out_le32(pdata->cpu_vbase[1] + MV64x60_CPU_ERR_CAUSE, 0);
  345. edac_device_handle_ue(edac_dev, 0, 0, edac_dev->ctl_name);
  346. }
  347. static irqreturn_t mv64x60_cpu_isr(int irq, void *dev_id)
  348. {
  349. struct edac_device_ctl_info *edac_dev = dev_id;
  350. struct mv64x60_cpu_pdata *pdata = edac_dev->pvt_info;
  351. u32 cause;
  352. cause = in_le32(pdata->cpu_vbase[1] + MV64x60_CPU_ERR_CAUSE) &
  353. MV64x60_CPU_CAUSE_MASK;
  354. if (!cause)
  355. return IRQ_NONE;
  356. mv64x60_cpu_check(edac_dev);
  357. return IRQ_HANDLED;
  358. }
  359. static int __devinit mv64x60_cpu_err_probe(struct platform_device *pdev)
  360. {
  361. struct edac_device_ctl_info *edac_dev;
  362. struct resource *r;
  363. struct mv64x60_cpu_pdata *pdata;
  364. int res = 0;
  365. if (!devres_open_group(&pdev->dev, mv64x60_cpu_err_probe, GFP_KERNEL))
  366. return -ENOMEM;
  367. edac_dev = edac_device_alloc_ctl_info(sizeof(*pdata),
  368. "cpu", 1, NULL, 0, 0, NULL, 0,
  369. edac_dev_idx);
  370. if (!edac_dev) {
  371. devres_release_group(&pdev->dev, mv64x60_cpu_err_probe);
  372. return -ENOMEM;
  373. }
  374. pdata = edac_dev->pvt_info;
  375. pdata->name = "mv64x60_cpu_err";
  376. pdata->irq = NO_IRQ;
  377. edac_dev->dev = &pdev->dev;
  378. platform_set_drvdata(pdev, edac_dev);
  379. edac_dev->dev_name = dev_name(&pdev->dev);
  380. r = platform_get_resource(pdev, IORESOURCE_MEM, 0);
  381. if (!r) {
  382. printk(KERN_ERR "%s: Unable to get resource for "
  383. "CPU err regs\n", __func__);
  384. res = -ENOENT;
  385. goto err;
  386. }
  387. if (!devm_request_mem_region(&pdev->dev,
  388. r->start,
  389. resource_size(r),
  390. pdata->name)) {
  391. printk(KERN_ERR "%s: Error while requesting mem region\n",
  392. __func__);
  393. res = -EBUSY;
  394. goto err;
  395. }
  396. pdata->cpu_vbase[0] = devm_ioremap(&pdev->dev,
  397. r->start,
  398. resource_size(r));
  399. if (!pdata->cpu_vbase[0]) {
  400. printk(KERN_ERR "%s: Unable to setup CPU err regs\n", __func__);
  401. res = -ENOMEM;
  402. goto err;
  403. }
  404. r = platform_get_resource(pdev, IORESOURCE_MEM, 1);
  405. if (!r) {
  406. printk(KERN_ERR "%s: Unable to get resource for "
  407. "CPU err regs\n", __func__);
  408. res = -ENOENT;
  409. goto err;
  410. }
  411. if (!devm_request_mem_region(&pdev->dev,
  412. r->start,
  413. resource_size(r),
  414. pdata->name)) {
  415. printk(KERN_ERR "%s: Error while requesting mem region\n",
  416. __func__);
  417. res = -EBUSY;
  418. goto err;
  419. }
  420. pdata->cpu_vbase[1] = devm_ioremap(&pdev->dev,
  421. r->start,
  422. resource_size(r));
  423. if (!pdata->cpu_vbase[1]) {
  424. printk(KERN_ERR "%s: Unable to setup CPU err regs\n", __func__);
  425. res = -ENOMEM;
  426. goto err;
  427. }
  428. /* setup CPU err registers */
  429. out_le32(pdata->cpu_vbase[1] + MV64x60_CPU_ERR_CAUSE, 0);
  430. out_le32(pdata->cpu_vbase[1] + MV64x60_CPU_ERR_MASK, 0);
  431. out_le32(pdata->cpu_vbase[1] + MV64x60_CPU_ERR_MASK, 0x000000ff);
  432. edac_dev->mod_name = EDAC_MOD_STR;
  433. edac_dev->ctl_name = pdata->name;
  434. if (edac_op_state == EDAC_OPSTATE_POLL)
  435. edac_dev->edac_check = mv64x60_cpu_check;
  436. pdata->edac_idx = edac_dev_idx++;
  437. if (edac_device_add_device(edac_dev) > 0) {
  438. debugf3("%s(): failed edac_device_add_device()\n", __func__);
  439. goto err;
  440. }
  441. if (edac_op_state == EDAC_OPSTATE_INT) {
  442. pdata->irq = platform_get_irq(pdev, 0);
  443. res = devm_request_irq(&pdev->dev,
  444. pdata->irq,
  445. mv64x60_cpu_isr,
  446. IRQF_DISABLED,
  447. "[EDAC] CPU err",
  448. edac_dev);
  449. if (res < 0) {
  450. printk(KERN_ERR
  451. "%s: Unable to request irq %d for MV64x60 "
  452. "CPU ERR\n", __func__, pdata->irq);
  453. res = -ENODEV;
  454. goto err2;
  455. }
  456. printk(KERN_INFO EDAC_MOD_STR
  457. " acquired irq %d for CPU Err\n", pdata->irq);
  458. }
  459. devres_remove_group(&pdev->dev, mv64x60_cpu_err_probe);
  460. /* get this far and it's successful */
  461. debugf3("%s(): success\n", __func__);
  462. return 0;
  463. err2:
  464. edac_device_del_device(&pdev->dev);
  465. err:
  466. devres_release_group(&pdev->dev, mv64x60_cpu_err_probe);
  467. edac_device_free_ctl_info(edac_dev);
  468. return res;
  469. }
  470. static int mv64x60_cpu_err_remove(struct platform_device *pdev)
  471. {
  472. struct edac_device_ctl_info *edac_dev = platform_get_drvdata(pdev);
  473. debugf0("%s()\n", __func__);
  474. edac_device_del_device(&pdev->dev);
  475. edac_device_free_ctl_info(edac_dev);
  476. return 0;
  477. }
  478. static struct platform_driver mv64x60_cpu_err_driver = {
  479. .probe = mv64x60_cpu_err_probe,
  480. .remove = mv64x60_cpu_err_remove,
  481. .driver = {
  482. .name = "mv64x60_cpu_err",
  483. }
  484. };
  485. /*********************** DRAM err device **********************************/
  486. static void mv64x60_mc_check(struct mem_ctl_info *mci)
  487. {
  488. struct mv64x60_mc_pdata *pdata = mci->pvt_info;
  489. u32 reg;
  490. u32 err_addr;
  491. u32 sdram_ecc;
  492. u32 comp_ecc;
  493. u32 syndrome;
  494. reg = in_le32(pdata->mc_vbase + MV64X60_SDRAM_ERR_ADDR);
  495. if (!reg)
  496. return;
  497. err_addr = reg & ~0x3;
  498. sdram_ecc = in_le32(pdata->mc_vbase + MV64X60_SDRAM_ERR_ECC_RCVD);
  499. comp_ecc = in_le32(pdata->mc_vbase + MV64X60_SDRAM_ERR_ECC_CALC);
  500. syndrome = sdram_ecc ^ comp_ecc;
  501. /* first bit clear in ECC Err Reg, 1 bit error, correctable by HW */
  502. if (!(reg & 0x1))
  503. edac_mc_handle_ce(mci, err_addr >> PAGE_SHIFT,
  504. err_addr & PAGE_MASK, syndrome, 0, 0,
  505. mci->ctl_name);
  506. else /* 2 bit error, UE */
  507. edac_mc_handle_ue(mci, err_addr >> PAGE_SHIFT,
  508. err_addr & PAGE_MASK, 0, mci->ctl_name);
  509. /* clear the error */
  510. out_le32(pdata->mc_vbase + MV64X60_SDRAM_ERR_ADDR, 0);
  511. }
  512. static irqreturn_t mv64x60_mc_isr(int irq, void *dev_id)
  513. {
  514. struct mem_ctl_info *mci = dev_id;
  515. struct mv64x60_mc_pdata *pdata = mci->pvt_info;
  516. u32 reg;
  517. reg = in_le32(pdata->mc_vbase + MV64X60_SDRAM_ERR_ADDR);
  518. if (!reg)
  519. return IRQ_NONE;
  520. /* writing 0's to the ECC err addr in check function clears irq */
  521. mv64x60_mc_check(mci);
  522. return IRQ_HANDLED;
  523. }
  524. static void get_total_mem(struct mv64x60_mc_pdata *pdata)
  525. {
  526. struct device_node *np = NULL;
  527. const unsigned int *reg;
  528. np = of_find_node_by_type(NULL, "memory");
  529. if (!np)
  530. return;
  531. reg = of_get_property(np, "reg", NULL);
  532. pdata->total_mem = reg[1];
  533. }
  534. static void mv64x60_init_csrows(struct mem_ctl_info *mci,
  535. struct mv64x60_mc_pdata *pdata)
  536. {
  537. struct csrow_info *csrow;
  538. u32 devtype;
  539. u32 ctl;
  540. get_total_mem(pdata);
  541. ctl = in_le32(pdata->mc_vbase + MV64X60_SDRAM_CONFIG);
  542. csrow = &mci->csrows[0];
  543. csrow->first_page = 0;
  544. csrow->nr_pages = pdata->total_mem >> PAGE_SHIFT;
  545. csrow->last_page = csrow->first_page + csrow->nr_pages - 1;
  546. csrow->grain = 8;
  547. csrow->mtype = (ctl & MV64X60_SDRAM_REGISTERED) ? MEM_RDDR : MEM_DDR;
  548. devtype = (ctl >> 20) & 0x3;
  549. switch (devtype) {
  550. case 0x0:
  551. csrow->dtype = DEV_X32;
  552. break;
  553. case 0x2: /* could be X8 too, but no way to tell */
  554. csrow->dtype = DEV_X16;
  555. break;
  556. case 0x3:
  557. csrow->dtype = DEV_X4;
  558. break;
  559. default:
  560. csrow->dtype = DEV_UNKNOWN;
  561. break;
  562. }
  563. csrow->edac_mode = EDAC_SECDED;
  564. }
  565. static int __devinit mv64x60_mc_err_probe(struct platform_device *pdev)
  566. {
  567. struct mem_ctl_info *mci;
  568. struct mv64x60_mc_pdata *pdata;
  569. struct resource *r;
  570. u32 ctl;
  571. int res = 0;
  572. if (!devres_open_group(&pdev->dev, mv64x60_mc_err_probe, GFP_KERNEL))
  573. return -ENOMEM;
  574. mci = edac_mc_alloc(sizeof(struct mv64x60_mc_pdata), 1, 1, edac_mc_idx);
  575. if (!mci) {
  576. printk(KERN_ERR "%s: No memory for CPU err\n", __func__);
  577. devres_release_group(&pdev->dev, mv64x60_mc_err_probe);
  578. return -ENOMEM;
  579. }
  580. pdata = mci->pvt_info;
  581. mci->dev = &pdev->dev;
  582. platform_set_drvdata(pdev, mci);
  583. pdata->name = "mv64x60_mc_err";
  584. pdata->irq = NO_IRQ;
  585. mci->dev_name = dev_name(&pdev->dev);
  586. pdata->edac_idx = edac_mc_idx++;
  587. r = platform_get_resource(pdev, IORESOURCE_MEM, 0);
  588. if (!r) {
  589. printk(KERN_ERR "%s: Unable to get resource for "
  590. "MC err regs\n", __func__);
  591. res = -ENOENT;
  592. goto err;
  593. }
  594. if (!devm_request_mem_region(&pdev->dev,
  595. r->start,
  596. resource_size(r),
  597. pdata->name)) {
  598. printk(KERN_ERR "%s: Error while requesting mem region\n",
  599. __func__);
  600. res = -EBUSY;
  601. goto err;
  602. }
  603. pdata->mc_vbase = devm_ioremap(&pdev->dev,
  604. r->start,
  605. resource_size(r));
  606. if (!pdata->mc_vbase) {
  607. printk(KERN_ERR "%s: Unable to setup MC err regs\n", __func__);
  608. res = -ENOMEM;
  609. goto err;
  610. }
  611. ctl = in_le32(pdata->mc_vbase + MV64X60_SDRAM_CONFIG);
  612. if (!(ctl & MV64X60_SDRAM_ECC)) {
  613. /* Non-ECC RAM? */
  614. printk(KERN_WARNING "%s: No ECC DIMMs discovered\n", __func__);
  615. res = -ENODEV;
  616. goto err2;
  617. }
  618. debugf3("%s(): init mci\n", __func__);
  619. mci->mtype_cap = MEM_FLAG_RDDR | MEM_FLAG_DDR;
  620. mci->edac_ctl_cap = EDAC_FLAG_NONE | EDAC_FLAG_SECDED;
  621. mci->edac_cap = EDAC_FLAG_SECDED;
  622. mci->mod_name = EDAC_MOD_STR;
  623. mci->mod_ver = MV64x60_REVISION;
  624. mci->ctl_name = mv64x60_ctl_name;
  625. if (edac_op_state == EDAC_OPSTATE_POLL)
  626. mci->edac_check = mv64x60_mc_check;
  627. mci->ctl_page_to_phys = NULL;
  628. mci->scrub_mode = SCRUB_SW_SRC;
  629. mv64x60_init_csrows(mci, pdata);
  630. /* setup MC registers */
  631. out_le32(pdata->mc_vbase + MV64X60_SDRAM_ERR_ADDR, 0);
  632. ctl = in_le32(pdata->mc_vbase + MV64X60_SDRAM_ERR_ECC_CNTL);
  633. ctl = (ctl & 0xff00ffff) | 0x10000;
  634. out_le32(pdata->mc_vbase + MV64X60_SDRAM_ERR_ECC_CNTL, ctl);
  635. if (edac_mc_add_mc(mci)) {
  636. debugf3("%s(): failed edac_mc_add_mc()\n", __func__);
  637. goto err;
  638. }
  639. if (edac_op_state == EDAC_OPSTATE_INT) {
  640. /* acquire interrupt that reports errors */
  641. pdata->irq = platform_get_irq(pdev, 0);
  642. res = devm_request_irq(&pdev->dev,
  643. pdata->irq,
  644. mv64x60_mc_isr,
  645. IRQF_DISABLED,
  646. "[EDAC] MC err",
  647. mci);
  648. if (res < 0) {
  649. printk(KERN_ERR "%s: Unable to request irq %d for "
  650. "MV64x60 DRAM ERR\n", __func__, pdata->irq);
  651. res = -ENODEV;
  652. goto err2;
  653. }
  654. printk(KERN_INFO EDAC_MOD_STR " acquired irq %d for MC Err\n",
  655. pdata->irq);
  656. }
  657. /* get this far and it's successful */
  658. debugf3("%s(): success\n", __func__);
  659. return 0;
  660. err2:
  661. edac_mc_del_mc(&pdev->dev);
  662. err:
  663. devres_release_group(&pdev->dev, mv64x60_mc_err_probe);
  664. edac_mc_free(mci);
  665. return res;
  666. }
  667. static int mv64x60_mc_err_remove(struct platform_device *pdev)
  668. {
  669. struct mem_ctl_info *mci = platform_get_drvdata(pdev);
  670. debugf0("%s()\n", __func__);
  671. edac_mc_del_mc(&pdev->dev);
  672. edac_mc_free(mci);
  673. return 0;
  674. }
  675. static struct platform_driver mv64x60_mc_err_driver = {
  676. .probe = mv64x60_mc_err_probe,
  677. .remove = mv64x60_mc_err_remove,
  678. .driver = {
  679. .name = "mv64x60_mc_err",
  680. }
  681. };
  682. static int __init mv64x60_edac_init(void)
  683. {
  684. int ret = 0;
  685. printk(KERN_INFO "Marvell MV64x60 EDAC driver " MV64x60_REVISION "\n");
  686. printk(KERN_INFO "\t(C) 2006-2007 MontaVista Software\n");
  687. /* make sure error reporting method is sane */
  688. switch (edac_op_state) {
  689. case EDAC_OPSTATE_POLL:
  690. case EDAC_OPSTATE_INT:
  691. break;
  692. default:
  693. edac_op_state = EDAC_OPSTATE_INT;
  694. break;
  695. }
  696. ret = platform_driver_register(&mv64x60_mc_err_driver);
  697. if (ret)
  698. printk(KERN_WARNING EDAC_MOD_STR "MC err failed to register\n");
  699. ret = platform_driver_register(&mv64x60_cpu_err_driver);
  700. if (ret)
  701. printk(KERN_WARNING EDAC_MOD_STR
  702. "CPU err failed to register\n");
  703. ret = platform_driver_register(&mv64x60_sram_err_driver);
  704. if (ret)
  705. printk(KERN_WARNING EDAC_MOD_STR
  706. "SRAM err failed to register\n");
  707. #ifdef CONFIG_PCI
  708. ret = platform_driver_register(&mv64x60_pci_err_driver);
  709. if (ret)
  710. printk(KERN_WARNING EDAC_MOD_STR
  711. "PCI err failed to register\n");
  712. #endif
  713. return ret;
  714. }
  715. module_init(mv64x60_edac_init);
  716. static void __exit mv64x60_edac_exit(void)
  717. {
  718. #ifdef CONFIG_PCI
  719. platform_driver_unregister(&mv64x60_pci_err_driver);
  720. #endif
  721. platform_driver_unregister(&mv64x60_sram_err_driver);
  722. platform_driver_unregister(&mv64x60_cpu_err_driver);
  723. platform_driver_unregister(&mv64x60_mc_err_driver);
  724. }
  725. module_exit(mv64x60_edac_exit);
  726. MODULE_LICENSE("GPL");
  727. MODULE_AUTHOR("Montavista Software, Inc.");
  728. module_param(edac_op_state, int, 0444);
  729. MODULE_PARM_DESC(edac_op_state,
  730. "EDAC Error Reporting state: 0=Poll, 2=Interrupt");