mmu2_error_converter.cpp 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128
  1. #include "mmu2_error_converter.h"
  2. #include "mmu2/error_codes.h"
  3. #include "mmu2/errors_list.h"
  4. #include "language.h"
  5. #include <stdio.h>
  6. namespace MMU2 {
  7. // we don't have a constexpr find_if in C++17/STL yet
  8. template <class InputIt, class UnaryPredicate>
  9. constexpr InputIt find_if_cx(InputIt first, InputIt last, UnaryPredicate p) {
  10. for (; first != last; ++first) {
  11. if (p(*first)) {
  12. return first;
  13. }
  14. }
  15. return last;
  16. }
  17. // Making a constexpr FindError should instruct the compiler to optimize the ConvertMMUErrorCode
  18. // in such a way that no searching will ever be done at runtime.
  19. // A call to FindError then compiles to a single instruction even on the AVR.
  20. static constexpr uint16_t FindErrorIndex(uint32_t pec) {
  21. constexpr uint32_t errorCodesSize = sizeof(errorCodes) / sizeof(errorCodes[0]);
  22. constexpr auto errorCodesEnd = errorCodes + errorCodesSize;
  23. auto i = find_if_cx(errorCodes, errorCodesEnd, [pec](uint16_t ed) -> bool {
  24. return ed == pec;
  25. });
  26. return i != errorCodesEnd ? *i : errorCodes[errorCodesSize - 1];
  27. }
  28. const uint16_t MMUErrorCodeIndex(uint16_t ec) {
  29. switch (ec) {
  30. case (uint16_t)ErrorCode::FINDA_DIDNT_SWITCH_ON:
  31. return FindErrorIndex(ERR_MECHANICAL_FINDA_DIDNT_TRIGGER);
  32. case (uint16_t)ErrorCode::FINDA_DIDNT_SWITCH_OFF:
  33. return FindErrorIndex(ERR_MECHANICAL_FINDA_DIDNT_GO_OFF);
  34. case (uint16_t)ErrorCode::FSENSOR_DIDNT_SWITCH_ON:
  35. return FindErrorIndex(ERR_MECHANICAL_FSENSOR_DIDNT_TRIGGER);
  36. case (uint16_t)ErrorCode::FSENSOR_DIDNT_SWITCH_OFF:
  37. return FindErrorIndex(ERR_MECHANICAL_FSENSOR_DIDNT_GO_OFF);
  38. case (uint16_t)ErrorCode::STALLED_PULLEY:
  39. case (uint16_t)ErrorCode::MOVE_PULLEY_FAILED:
  40. return FindErrorIndex(ERR_MECHANICAL_PULLEY_CANNOT_MOVE);
  41. case (uint16_t)ErrorCode::HOMING_SELECTOR_FAILED:
  42. return FindErrorIndex(ERR_MECHANICAL_SELECTOR_CANNOT_HOME);
  43. case (uint16_t)ErrorCode::MOVE_SELECTOR_FAILED:
  44. return FindErrorIndex(ERR_MECHANICAL_SELECTOR_CANNOT_MOVE);
  45. case (uint16_t)ErrorCode::HOMING_IDLER_FAILED:
  46. return FindErrorIndex(ERR_MECHANICAL_IDLER_CANNOT_HOME);
  47. case (uint16_t)ErrorCode::MMU_NOT_RESPONDING:
  48. return FindErrorIndex(ERR_MECHANICAL_IDLER_CANNOT_MOVE);
  49. case (uint16_t)ErrorCode::PROTOCOL_ERROR:
  50. return FindErrorIndex(ERR_CONNECT_COMMUNICATION_ERROR);
  51. case (uint16_t)ErrorCode::FILAMENT_ALREADY_LOADED:
  52. return FindErrorIndex(ERR_SYSTEM_FILAMENT_ALREADY_LOADED);
  53. case (uint16_t)ErrorCode::INVALID_TOOL:
  54. return FindErrorIndex(ERR_SYSTEM_INVALID_TOOL);
  55. case (uint16_t)ErrorCode::QUEUE_FULL:
  56. return FindErrorIndex(ERR_SYSTEM_QUEUE_FULL);
  57. case (uint16_t)ErrorCode::VERSION_MISMATCH:
  58. return FindErrorIndex(ERR_SYSTEM_FW_UPDATE_NEEDED);
  59. case (uint16_t)ErrorCode::INTERNAL:
  60. return FindErrorIndex(ERR_SYSTEM_FW_RUNTIME_ERROR);
  61. case (uint16_t)ErrorCode::FINDA_VS_EEPROM_DISREPANCY:
  62. return FindErrorIndex(ERR_SYSTEM_UNLOAD_MANUALLY);
  63. }
  64. // // TMC-related errors - multiple of these can occur at once
  65. // // - in such a case we report the first which gets found/converted into Prusa-Error-Codes (usually the fact, that one TMC has an issue is serious enough)
  66. // // By carefully ordering the checks here we can prioritize the errors being reported to the user.
  67. if (ec & (uint16_t)ErrorCode::TMC_PULLEY_BIT) {
  68. if (ec & (uint16_t)ErrorCode::TMC_IOIN_MISMATCH)
  69. return FindErrorIndex(ERR_ELECTRICAL_PULLEY_TMC_DRIVER_ERROR);
  70. if (ec & (uint16_t)ErrorCode::TMC_RESET)
  71. return FindErrorIndex(ERR_ELECTRICAL_PULLEY_TMC_DRIVER_RESET);
  72. if (ec & (uint16_t)ErrorCode::TMC_UNDERVOLTAGE_ON_CHARGE_PUMP)
  73. return FindErrorIndex(ERR_ELECTRICAL_PULLEY_TMC_UNDERVOLTAGE_ERROR);
  74. if (ec & (uint16_t)ErrorCode::TMC_SHORT_TO_GROUND)
  75. return FindErrorIndex(ERR_ELECTRICAL_PULLEY_TMC_DRIVER_SHORTED);
  76. if (ec & (uint16_t)ErrorCode::TMC_OVER_TEMPERATURE_WARN)
  77. return FindErrorIndex(ERR_TEMPERATURE_PULLEY_WARNING_TMC_TOO_HOT);
  78. if (ec & (uint16_t)ErrorCode::TMC_OVER_TEMPERATURE_ERROR)
  79. return FindErrorIndex(ERR_TEMPERATURE_PULLEY_TMC_OVERHEAT_ERROR);
  80. } else if (ec & (uint16_t)ErrorCode::TMC_SELECTOR_BIT) {
  81. if (ec & (uint16_t)ErrorCode::TMC_IOIN_MISMATCH)
  82. return FindErrorIndex(ERR_ELECTRICAL_SELECTOR_TMC_DRIVER_ERROR);
  83. if (ec & (uint16_t)ErrorCode::TMC_RESET)
  84. return FindErrorIndex(ERR_ELECTRICAL_SELECTOR_TMC_DRIVER_RESET);
  85. if (ec & (uint16_t)ErrorCode::TMC_UNDERVOLTAGE_ON_CHARGE_PUMP)
  86. return FindErrorIndex(ERR_ELECTRICAL_SELECTOR_TMC_UNDERVOLTAGE_ERROR);
  87. if (ec & (uint16_t)ErrorCode::TMC_SHORT_TO_GROUND)
  88. return FindErrorIndex(ERR_ELECTRICAL_SELECTOR_TMC_DRIVER_SHORTED);
  89. if (ec & (uint16_t)ErrorCode::TMC_OVER_TEMPERATURE_WARN)
  90. return FindErrorIndex(ERR_TEMPERATURE_SELECTOR_WARNING_TMC_TOO_HOT);
  91. if (ec & (uint16_t)ErrorCode::TMC_OVER_TEMPERATURE_ERROR)
  92. return FindErrorIndex(ERR_TEMPERATURE_SELECTOR_TMC_OVERHEAT_ERROR);
  93. } else if (ec & (uint16_t)ErrorCode::TMC_IDLER_BIT) {
  94. if (ec & (uint16_t)ErrorCode::TMC_IOIN_MISMATCH)
  95. return FindErrorIndex(ERR_ELECTRICAL_IDLER_TMC_DRIVER_ERROR);
  96. if (ec & (uint16_t)ErrorCode::TMC_RESET)
  97. return FindErrorIndex(ERR_ELECTRICAL_IDLER_TMC_DRIVER_RESET);
  98. if (ec & (uint16_t)ErrorCode::TMC_UNDERVOLTAGE_ON_CHARGE_PUMP)
  99. return FindErrorIndex(ERR_ELECTRICAL_IDLER_TMC_UNDERVOLTAGE_ERROR);
  100. if (ec & (uint16_t)ErrorCode::TMC_SHORT_TO_GROUND)
  101. return FindErrorIndex(ERR_ELECTRICAL_IDLER_TMC_DRIVER_SHORTED);
  102. if (ec & (uint16_t)ErrorCode::TMC_OVER_TEMPERATURE_WARN)
  103. return FindErrorIndex(ERR_TEMPERATURE_IDLER_WARNING_TMC_TOO_HOT);
  104. if (ec & (uint16_t)ErrorCode::TMC_OVER_TEMPERATURE_ERROR)
  105. return FindErrorIndex(ERR_TEMPERATURE_IDLER_TMC_OVERHEAT_ERROR);
  106. }
  107. // // if nothing got caught, return a generic error
  108. // return FindError(ERR_OTHER);
  109. }
  110. void TranslateErr(uint16_t ec, char *dst, size_t dstSize) {
  111. uint16_t ei = MMUErrorCodeIndex(ec);
  112. // just to prevent the compiler from stripping the data structures from the final binary for now
  113. *dst = errorButtons[ei];
  114. snprintf(
  115. dst, dstSize, "%S %S",
  116. static_cast<const char * const>(pgm_read_ptr(&errorTitles[ei])),
  117. static_cast<const char * const>(pgm_read_ptr(&errorDescs[ei]))
  118. );
  119. }
  120. } // namespace MMU2