fsl-corenet-cf.c 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251
  1. /*
  2. * CoreNet Coherency Fabric error reporting
  3. *
  4. * Copyright 2014 Freescale Semiconductor Inc.
  5. *
  6. * This program is free software; you can redistribute it and/or modify it
  7. * under the terms of the GNU General Public License as published by the
  8. * Free Software Foundation; either version 2 of the License, or (at your
  9. * option) any later version.
  10. */
  11. #include <linux/interrupt.h>
  12. #include <linux/io.h>
  13. #include <linux/irq.h>
  14. #include <linux/module.h>
  15. #include <linux/of.h>
  16. #include <linux/of_address.h>
  17. #include <linux/of_device.h>
  18. #include <linux/of_irq.h>
  19. #include <linux/platform_device.h>
  20. enum ccf_version {
  21. CCF1,
  22. CCF2,
  23. };
  24. struct ccf_info {
  25. enum ccf_version version;
  26. int err_reg_offs;
  27. };
  28. static const struct ccf_info ccf1_info = {
  29. .version = CCF1,
  30. .err_reg_offs = 0xa00,
  31. };
  32. static const struct ccf_info ccf2_info = {
  33. .version = CCF2,
  34. .err_reg_offs = 0xe40,
  35. };
  36. static const struct of_device_id ccf_matches[] = {
  37. {
  38. .compatible = "fsl,corenet1-cf",
  39. .data = &ccf1_info,
  40. },
  41. {
  42. .compatible = "fsl,corenet2-cf",
  43. .data = &ccf2_info,
  44. },
  45. {}
  46. };
  47. struct ccf_err_regs {
  48. u32 errdet; /* 0x00 Error Detect Register */
  49. /* 0x04 Error Enable (ccf1)/Disable (ccf2) Register */
  50. u32 errdis;
  51. /* 0x08 Error Interrupt Enable Register (ccf2 only) */
  52. u32 errinten;
  53. u32 cecar; /* 0x0c Error Capture Attribute Register */
  54. u32 cecaddrh; /* 0x10 Error Capture Address High */
  55. u32 cecaddrl; /* 0x14 Error Capture Address Low */
  56. u32 cecar2; /* 0x18 Error Capture Attribute Register 2 */
  57. };
  58. /* LAE/CV also valid for errdis and errinten */
  59. #define ERRDET_LAE (1 << 0) /* Local Access Error */
  60. #define ERRDET_CV (1 << 1) /* Coherency Violation */
  61. #define ERRDET_CTYPE_SHIFT 26 /* Capture Type (ccf2 only) */
  62. #define ERRDET_CTYPE_MASK (0x1f << ERRDET_CTYPE_SHIFT)
  63. #define ERRDET_CAP (1 << 31) /* Capture Valid (ccf2 only) */
  64. #define CECAR_VAL (1 << 0) /* Valid (ccf1 only) */
  65. #define CECAR_UVT (1 << 15) /* Unavailable target ID (ccf1) */
  66. #define CECAR_SRCID_SHIFT_CCF1 24
  67. #define CECAR_SRCID_MASK_CCF1 (0xff << CECAR_SRCID_SHIFT_CCF1)
  68. #define CECAR_SRCID_SHIFT_CCF2 18
  69. #define CECAR_SRCID_MASK_CCF2 (0xff << CECAR_SRCID_SHIFT_CCF2)
  70. #define CECADDRH_ADDRH 0xff
  71. struct ccf_private {
  72. const struct ccf_info *info;
  73. struct device *dev;
  74. void __iomem *regs;
  75. struct ccf_err_regs __iomem *err_regs;
  76. };
  77. static irqreturn_t ccf_irq(int irq, void *dev_id)
  78. {
  79. struct ccf_private *ccf = dev_id;
  80. static DEFINE_RATELIMIT_STATE(ratelimit, DEFAULT_RATELIMIT_INTERVAL,
  81. DEFAULT_RATELIMIT_BURST);
  82. u32 errdet, cecar, cecar2;
  83. u64 addr;
  84. u32 src_id;
  85. bool uvt = false;
  86. bool cap_valid = false;
  87. errdet = ioread32be(&ccf->err_regs->errdet);
  88. cecar = ioread32be(&ccf->err_regs->cecar);
  89. cecar2 = ioread32be(&ccf->err_regs->cecar2);
  90. addr = ioread32be(&ccf->err_regs->cecaddrl);
  91. addr |= ((u64)(ioread32be(&ccf->err_regs->cecaddrh) &
  92. CECADDRH_ADDRH)) << 32;
  93. if (!__ratelimit(&ratelimit))
  94. goto out;
  95. switch (ccf->info->version) {
  96. case CCF1:
  97. if (cecar & CECAR_VAL) {
  98. if (cecar & CECAR_UVT)
  99. uvt = true;
  100. src_id = (cecar & CECAR_SRCID_MASK_CCF1) >>
  101. CECAR_SRCID_SHIFT_CCF1;
  102. cap_valid = true;
  103. }
  104. break;
  105. case CCF2:
  106. if (errdet & ERRDET_CAP) {
  107. src_id = (cecar & CECAR_SRCID_MASK_CCF2) >>
  108. CECAR_SRCID_SHIFT_CCF2;
  109. cap_valid = true;
  110. }
  111. break;
  112. }
  113. dev_crit(ccf->dev, "errdet 0x%08x cecar 0x%08x cecar2 0x%08x\n",
  114. errdet, cecar, cecar2);
  115. if (errdet & ERRDET_LAE) {
  116. if (uvt)
  117. dev_crit(ccf->dev, "LAW Unavailable Target ID\n");
  118. else
  119. dev_crit(ccf->dev, "Local Access Window Error\n");
  120. }
  121. if (errdet & ERRDET_CV)
  122. dev_crit(ccf->dev, "Coherency Violation\n");
  123. if (cap_valid) {
  124. dev_crit(ccf->dev, "address 0x%09llx, src id 0x%x\n",
  125. addr, src_id);
  126. }
  127. out:
  128. iowrite32be(errdet, &ccf->err_regs->errdet);
  129. return errdet ? IRQ_HANDLED : IRQ_NONE;
  130. }
  131. static int ccf_probe(struct platform_device *pdev)
  132. {
  133. struct ccf_private *ccf;
  134. struct resource *r;
  135. const struct of_device_id *match;
  136. int ret, irq;
  137. match = of_match_device(ccf_matches, &pdev->dev);
  138. if (WARN_ON(!match))
  139. return -ENODEV;
  140. ccf = devm_kzalloc(&pdev->dev, sizeof(*ccf), GFP_KERNEL);
  141. if (!ccf)
  142. return -ENOMEM;
  143. r = platform_get_resource(pdev, IORESOURCE_MEM, 0);
  144. if (!r) {
  145. dev_err(&pdev->dev, "%s: no mem resource\n", __func__);
  146. return -ENXIO;
  147. }
  148. ccf->regs = devm_ioremap_resource(&pdev->dev, r);
  149. if (IS_ERR(ccf->regs)) {
  150. dev_err(&pdev->dev, "%s: can't map mem resource\n", __func__);
  151. return PTR_ERR(ccf->regs);
  152. }
  153. ccf->dev = &pdev->dev;
  154. ccf->info = match->data;
  155. ccf->err_regs = ccf->regs + ccf->info->err_reg_offs;
  156. dev_set_drvdata(&pdev->dev, ccf);
  157. irq = platform_get_irq(pdev, 0);
  158. if (!irq) {
  159. dev_err(&pdev->dev, "%s: no irq\n", __func__);
  160. return -ENXIO;
  161. }
  162. ret = devm_request_irq(&pdev->dev, irq, ccf_irq, 0, pdev->name, ccf);
  163. if (ret) {
  164. dev_err(&pdev->dev, "%s: can't request irq\n", __func__);
  165. return ret;
  166. }
  167. switch (ccf->info->version) {
  168. case CCF1:
  169. /* On CCF1 this register enables rather than disables. */
  170. iowrite32be(ERRDET_LAE | ERRDET_CV, &ccf->err_regs->errdis);
  171. break;
  172. case CCF2:
  173. iowrite32be(0, &ccf->err_regs->errdis);
  174. iowrite32be(ERRDET_LAE | ERRDET_CV, &ccf->err_regs->errinten);
  175. break;
  176. }
  177. return 0;
  178. }
  179. static int ccf_remove(struct platform_device *pdev)
  180. {
  181. struct ccf_private *ccf = dev_get_drvdata(&pdev->dev);
  182. switch (ccf->info->version) {
  183. case CCF1:
  184. iowrite32be(0, &ccf->err_regs->errdis);
  185. break;
  186. case CCF2:
  187. /*
  188. * We clear errdis on ccf1 because that's the only way to
  189. * disable interrupts, but on ccf2 there's no need to disable
  190. * detection.
  191. */
  192. iowrite32be(0, &ccf->err_regs->errinten);
  193. break;
  194. }
  195. return 0;
  196. }
  197. static struct platform_driver ccf_driver = {
  198. .driver = {
  199. .name = KBUILD_MODNAME,
  200. .owner = THIS_MODULE,
  201. .of_match_table = ccf_matches,
  202. },
  203. .probe = ccf_probe,
  204. .remove = ccf_remove,
  205. };
  206. module_platform_driver(ccf_driver);
  207. MODULE_LICENSE("GPL");
  208. MODULE_AUTHOR("Freescale Semiconductor");
  209. MODULE_DESCRIPTION("Freescale CoreNet Coherency Fabric error reporting");