edac.h 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393
  1. /*
  2. * Copyright (c) 2020 Intel Corporation
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief EDAC API header file
  9. */
  10. #ifndef ZEPHYR_INCLUDE_DRIVERS_EDAC_H_
  11. #define ZEPHYR_INCLUDE_DRIVERS_EDAC_H_
  12. #include <sys/types.h>
  13. typedef void (*edac_notify_callback_f)(const struct device *dev, void *data);
  14. /**
  15. * @defgroup edac EDAC API
  16. * @ingroup io_interfaces
  17. * @{
  18. */
  19. /**
  20. * @brief EDAC error type
  21. */
  22. enum edac_error_type {
  23. /** Correctable error type */
  24. EDAC_ERROR_TYPE_DRAM_COR = BIT(0),
  25. /** Uncorrectable error type */
  26. EDAC_ERROR_TYPE_DRAM_UC = BIT(1)
  27. };
  28. /**
  29. * @brief EDAC driver API
  30. *
  31. * This is the mandatory API any EDAC driver needs to expose.
  32. */
  33. __subsystem struct edac_driver_api {
  34. /* Error Injection API is disabled by default */
  35. int (*inject_set_param1)(const struct device *dev, uint64_t value);
  36. int (*inject_get_param1)(const struct device *dev, uint64_t *value);
  37. int (*inject_set_param2)(const struct device *dev, uint64_t value);
  38. int (*inject_get_param2)(const struct device *dev, uint64_t *value);
  39. int (*inject_set_error_type)(const struct device *dev, uint32_t value);
  40. int (*inject_get_error_type)(const struct device *dev, uint32_t *value);
  41. int (*inject_error_trigger)(const struct device *dev);
  42. /* Error Logging API */
  43. int (*ecc_error_log_get)(const struct device *dev, uint64_t *value);
  44. int (*ecc_error_log_clear)(const struct device *dev);
  45. int (*parity_error_log_get)(const struct device *dev, uint64_t *value);
  46. int (*parity_error_log_clear)(const struct device *dev);
  47. /* Error stats API */
  48. int (*errors_cor_get)(const struct device *dev);
  49. int (*errors_uc_get)(const struct device *dev);
  50. /* Notification callback API */
  51. int (*notify_cb_set)(const struct device *dev,
  52. edac_notify_callback_f cb);
  53. };
  54. /* Optional interfaces */
  55. /**
  56. * @brief Set injection parameter param1
  57. *
  58. * Set first error injection parameter value.
  59. *
  60. * @param dev Pointer to the device structure
  61. * @param value First injection parameter
  62. *
  63. * @retval -ENOSYS if the optional interface is not implemented
  64. * @retval 0 on success, other error code otherwise
  65. */
  66. static inline int edac_inject_set_param1(const struct device *dev,
  67. uint64_t value)
  68. {
  69. const struct edac_driver_api *api =
  70. (const struct edac_driver_api *)dev->api;
  71. if (api->inject_set_param1 == NULL) {
  72. return -ENOSYS;
  73. }
  74. return api->inject_set_param1(dev, value);
  75. }
  76. /**
  77. * @brief Get injection parameter param1
  78. *
  79. * Get first error injection parameter value.
  80. *
  81. * @param dev Pointer to the device structure
  82. * @param value Pointer to the first injection parameter
  83. *
  84. * @retval -ENOSYS if the optional interface is not implemented
  85. * @retval 0 on success, error code otherwise
  86. */
  87. static inline int edac_inject_get_param1(const struct device *dev,
  88. uint64_t *value)
  89. {
  90. const struct edac_driver_api *api =
  91. (const struct edac_driver_api *)dev->api;
  92. if (api->inject_get_param1 == NULL) {
  93. return -ENOSYS;
  94. }
  95. return api->inject_get_param1(dev, value);
  96. }
  97. /**
  98. * @brief Set injection parameter param2
  99. *
  100. * Set second error injection parameter value.
  101. *
  102. * @param dev Pointer to the device structure
  103. * @param value Second injection parameter
  104. *
  105. * @retval -ENOSYS if the optional interface is not implemented
  106. * @retval 0 on success, error code otherwise
  107. */
  108. static inline int edac_inject_set_param2(const struct device *dev,
  109. uint64_t value)
  110. {
  111. const struct edac_driver_api *api =
  112. (const struct edac_driver_api *)dev->api;
  113. if (api->inject_set_param2 == NULL) {
  114. return -ENOSYS;
  115. }
  116. return api->inject_set_param2(dev, value);
  117. }
  118. /**
  119. * @brief Get injection parameter param2
  120. *
  121. * @param dev Pointer to the device structure
  122. * @param value Pointer to the second injection parameter
  123. *
  124. * @retval -ENOSYS if the optional interface is not implemented
  125. * @retval 0 on success, error code otherwise
  126. */
  127. static inline int edac_inject_get_param2(const struct device *dev,
  128. uint64_t *value)
  129. {
  130. const struct edac_driver_api *api =
  131. (const struct edac_driver_api *)dev->api;
  132. if (api->inject_get_param2 == NULL) {
  133. return -ENOSYS;
  134. }
  135. return api->inject_get_param2(dev, value);
  136. }
  137. /**
  138. * @brief Set error type value
  139. *
  140. * Set the value of error type to be injected
  141. *
  142. * @param dev Pointer to the device structure
  143. * @param error_type Error type value
  144. *
  145. * @retval -ENOSYS if the optional interface is not implemented
  146. * @retval 0 on success, error code otherwise
  147. */
  148. static inline int edac_inject_set_error_type(const struct device *dev,
  149. uint32_t error_type)
  150. {
  151. const struct edac_driver_api *api =
  152. (const struct edac_driver_api *)dev->api;
  153. if (api->inject_set_error_type == NULL) {
  154. return -ENOSYS;
  155. }
  156. return api->inject_set_error_type(dev, error_type);
  157. }
  158. /**
  159. * @brief Get error type value
  160. *
  161. * Get the value of error type to be injected
  162. *
  163. * @param dev Pointer to the device structure
  164. * @param error_type Pointer to error type value
  165. *
  166. * @retval -ENOSYS if the optional interface is not implemented
  167. * @retval 0 on success, error code otherwise
  168. */
  169. static inline int edac_inject_get_error_type(const struct device *dev,
  170. uint32_t *error_type)
  171. {
  172. const struct edac_driver_api *api =
  173. (const struct edac_driver_api *)dev->api;
  174. if (api->inject_get_error_type == NULL) {
  175. return -ENOSYS;
  176. }
  177. return api->inject_get_error_type(dev, error_type);
  178. }
  179. /**
  180. * @brief Set injection control
  181. *
  182. * Trigger error injection.
  183. *
  184. * @param dev Pointer to the device structure
  185. *
  186. * @retval -ENOSYS if the optional interface is not implemented
  187. * @retval 0 on success, error code otherwise
  188. */
  189. static inline int edac_inject_error_trigger(const struct device *dev)
  190. {
  191. const struct edac_driver_api *api =
  192. (const struct edac_driver_api *)dev->api;
  193. if (api->inject_error_trigger == NULL) {
  194. return -ENOSYS;
  195. }
  196. return api->inject_error_trigger(dev);
  197. }
  198. /* Mandatory interfaces */
  199. /**
  200. * @brief Get ECC Error Log
  201. *
  202. * Read value of ECC Error Log.
  203. *
  204. * @param dev Pointer to the device structure
  205. * @param value Pointer to the ECC Error Log value
  206. *
  207. * @retval 0 on success, error code otherwise
  208. * @retval -ENOSYS if the mandatory interface is not implemented
  209. */
  210. static inline int edac_ecc_error_log_get(const struct device *dev,
  211. uint64_t *value)
  212. {
  213. const struct edac_driver_api *api =
  214. (const struct edac_driver_api *)dev->api;
  215. if (api->ecc_error_log_get == NULL) {
  216. return -ENOSYS;
  217. }
  218. return api->ecc_error_log_get(dev, value);
  219. }
  220. /**
  221. * @brief Clear ECC Error Log
  222. *
  223. * Clear value of ECC Error Log.
  224. *
  225. * @param dev Pointer to the device structure
  226. *
  227. * @retval 0 on success, error code otherwise
  228. * @retval -ENOSYS if the mandatory interface is not implemented
  229. */
  230. static inline int edac_ecc_error_log_clear(const struct device *dev)
  231. {
  232. const struct edac_driver_api *api =
  233. (const struct edac_driver_api *)dev->api;
  234. if (api->ecc_error_log_clear == NULL) {
  235. return -ENOSYS;
  236. }
  237. return api->ecc_error_log_clear(dev);
  238. }
  239. /**
  240. * @brief Get Parity Error Log
  241. *
  242. * Read value of Parity Error Log.
  243. *
  244. * @param dev Pointer to the device structure
  245. * @param value Pointer to the parity Error Log value
  246. *
  247. * @retval 0 on success, error code otherwise
  248. * @retval -ENOSYS if the mandatory interface is not implemented
  249. */
  250. static inline int edac_parity_error_log_get(const struct device *dev,
  251. uint64_t *value)
  252. {
  253. const struct edac_driver_api *api =
  254. (const struct edac_driver_api *)dev->api;
  255. if (api->parity_error_log_get == NULL) {
  256. return -ENOSYS;
  257. }
  258. return api->parity_error_log_get(dev, value);
  259. }
  260. /**
  261. * @brief Clear Parity Error Log
  262. *
  263. * Clear value of Parity Error Log.
  264. *
  265. * @param dev Pointer to the device structure
  266. *
  267. * @retval 0 on success, error code otherwise
  268. * @retval -ENOSYS if the mandatory interface is not implemented
  269. */
  270. static inline int edac_parity_error_log_clear(const struct device *dev)
  271. {
  272. const struct edac_driver_api *api =
  273. (const struct edac_driver_api *)dev->api;
  274. if (api->parity_error_log_clear == NULL) {
  275. return -ENOSYS;
  276. }
  277. return api->parity_error_log_clear(dev);
  278. }
  279. /**
  280. * @brief Get number of correctable errors
  281. *
  282. * @param dev Pointer to the device structure
  283. *
  284. * @retval num Number of correctable errors
  285. * @retval -ENOSYS if the mandatory interface is not implemented
  286. */
  287. static inline int edac_errors_cor_get(const struct device *dev)
  288. {
  289. const struct edac_driver_api *api =
  290. (const struct edac_driver_api *)dev->api;
  291. if (api->errors_cor_get == NULL) {
  292. return -ENOSYS;
  293. }
  294. return api->errors_cor_get(dev);
  295. }
  296. /**
  297. * @brief Get number of uncorrectable errors
  298. *
  299. * @param dev Pointer to the device structure
  300. *
  301. * @retval num Number of uncorrectable errors
  302. * @retval -ENOSYS if the mandatory interface is not implemented
  303. */
  304. static inline int edac_errors_uc_get(const struct device *dev)
  305. {
  306. const struct edac_driver_api *api =
  307. (const struct edac_driver_api *)dev->api;
  308. if (api->errors_uc_get == NULL) {
  309. return -ENOSYS;
  310. }
  311. return api->errors_uc_get(dev);
  312. }
  313. /**
  314. * Register callback function for memory error exception
  315. *
  316. * This callback runs in interrupt context
  317. *
  318. * @param dev EDAC driver device to install callback
  319. * @param cb Callback function pointer
  320. *
  321. * @retval 0 on success, error code otherwise
  322. * @retval -ENOSYS if the mandatory interface is not implemented
  323. */
  324. static inline int edac_notify_callback_set(const struct device *dev,
  325. edac_notify_callback_f cb)
  326. {
  327. const struct edac_driver_api *api = dev->api;
  328. if (api->notify_cb_set == NULL) {
  329. return -ENOSYS;
  330. }
  331. return api->notify_cb_set(dev, cb);
  332. }
  333. /**
  334. * @}
  335. */
  336. #endif /* ZEPHYR_INCLUDE_DRIVERS_EDAC_H_ */