ghes.h 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123
  1. #ifndef GHES_H
  2. #define GHES_H
  3. #include <acpi/apei.h>
  4. #include <acpi/hed.h>
  5. /*
  6. * One struct ghes is created for each generic hardware error source.
  7. * It provides the context for APEI hardware error timer/IRQ/SCI/NMI
  8. * handler.
  9. *
  10. * estatus: memory buffer for error status block, allocated during
  11. * HEST parsing.
  12. */
  13. #define GHES_TO_CLEAR 0x0001
  14. #define GHES_EXITING 0x0002
  15. struct ghes {
  16. union {
  17. struct acpi_hest_generic *generic;
  18. struct acpi_hest_generic_v2 *generic_v2;
  19. };
  20. struct acpi_hest_generic_status *estatus;
  21. u64 buffer_paddr;
  22. unsigned long flags;
  23. union {
  24. struct list_head list;
  25. struct timer_list timer;
  26. unsigned int irq;
  27. };
  28. };
  29. struct ghes_estatus_node {
  30. struct llist_node llnode;
  31. struct acpi_hest_generic *generic;
  32. struct ghes *ghes;
  33. };
  34. struct ghes_estatus_cache {
  35. u32 estatus_len;
  36. atomic_t count;
  37. struct acpi_hest_generic *generic;
  38. unsigned long long time_in;
  39. struct rcu_head rcu;
  40. };
  41. enum {
  42. GHES_SEV_NO = 0x0,
  43. GHES_SEV_CORRECTED = 0x1,
  44. GHES_SEV_RECOVERABLE = 0x2,
  45. GHES_SEV_PANIC = 0x3,
  46. };
  47. /* From drivers/edac/ghes_edac.c */
  48. #ifdef CONFIG_EDAC_GHES
  49. void ghes_edac_report_mem_error(struct ghes *ghes, int sev,
  50. struct cper_sec_mem_err *mem_err);
  51. int ghes_edac_register(struct ghes *ghes, struct device *dev);
  52. void ghes_edac_unregister(struct ghes *ghes);
  53. #else
  54. static inline void ghes_edac_report_mem_error(struct ghes *ghes, int sev,
  55. struct cper_sec_mem_err *mem_err)
  56. {
  57. }
  58. static inline int ghes_edac_register(struct ghes *ghes, struct device *dev)
  59. {
  60. return 0;
  61. }
  62. static inline void ghes_edac_unregister(struct ghes *ghes)
  63. {
  64. }
  65. #endif
  66. static inline int acpi_hest_get_version(struct acpi_hest_generic_data *gdata)
  67. {
  68. return gdata->revision >> 8;
  69. }
  70. static inline void *acpi_hest_get_payload(struct acpi_hest_generic_data *gdata)
  71. {
  72. if (acpi_hest_get_version(gdata) >= 3)
  73. return (void *)(((struct acpi_hest_generic_data_v300 *)(gdata)) + 1);
  74. return gdata + 1;
  75. }
  76. static inline int acpi_hest_get_error_length(struct acpi_hest_generic_data *gdata)
  77. {
  78. return ((struct acpi_hest_generic_data *)(gdata))->error_data_length;
  79. }
  80. static inline int acpi_hest_get_size(struct acpi_hest_generic_data *gdata)
  81. {
  82. if (acpi_hest_get_version(gdata) >= 3)
  83. return sizeof(struct acpi_hest_generic_data_v300);
  84. return sizeof(struct acpi_hest_generic_data);
  85. }
  86. static inline int acpi_hest_get_record_size(struct acpi_hest_generic_data *gdata)
  87. {
  88. return (acpi_hest_get_size(gdata) + acpi_hest_get_error_length(gdata));
  89. }
  90. static inline void *acpi_hest_get_next(struct acpi_hest_generic_data *gdata)
  91. {
  92. return (void *)(gdata) + acpi_hest_get_record_size(gdata);
  93. }
  94. #define apei_estatus_for_each_section(estatus, section) \
  95. for (section = (struct acpi_hest_generic_data *)(estatus + 1); \
  96. (void *)section - (void *)(estatus + 1) < estatus->data_length; \
  97. section = acpi_hest_get_next(section))
  98. int ghes_notify_sea(void);
  99. #endif /* GHES_H */