mmu2_error_converter.cpp 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146
  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 uint8_t FindErrorIndex(uint16_t pec) {
  21. constexpr uint16_t errorCodesSize = sizeof(errorCodes) / sizeof(errorCodes[0]);
  22. constexpr const auto *errorCodesEnd = errorCodes + errorCodesSize;
  23. const auto *i = find_if_cx(errorCodes, errorCodesEnd, [pec](uint16_t ed){ return ed == pec; });
  24. return (i != errorCodesEnd) ? (i-errorCodes) : (errorCodesSize - 1);
  25. }
  26. // check that the searching algoritm works
  27. static_assert( FindErrorIndex(ERR_MECHANICAL_FINDA_DIDNT_TRIGGER) == 0);
  28. static_assert( FindErrorIndex(ERR_MECHANICAL_FINDA_DIDNT_GO_OFF) == 1);
  29. static_assert( FindErrorIndex(ERR_MECHANICAL_FSENSOR_DIDNT_TRIGGER) == 2);
  30. static_assert( FindErrorIndex(ERR_MECHANICAL_FSENSOR_DIDNT_GO_OFF) == 3);
  31. uint8_t PrusaErrorCodeIndex(uint16_t ec) {
  32. switch (ec) {
  33. case (uint16_t)ErrorCode::FINDA_DIDNT_SWITCH_ON:
  34. return FindErrorIndex(ERR_MECHANICAL_FINDA_DIDNT_TRIGGER);
  35. case (uint16_t)ErrorCode::FINDA_DIDNT_SWITCH_OFF:
  36. return FindErrorIndex(ERR_MECHANICAL_FINDA_DIDNT_GO_OFF);
  37. case (uint16_t)ErrorCode::FSENSOR_DIDNT_SWITCH_ON:
  38. return FindErrorIndex(ERR_MECHANICAL_FSENSOR_DIDNT_TRIGGER);
  39. case (uint16_t)ErrorCode::FSENSOR_DIDNT_SWITCH_OFF:
  40. return FindErrorIndex(ERR_MECHANICAL_FSENSOR_DIDNT_GO_OFF);
  41. case (uint16_t)ErrorCode::STALLED_PULLEY:
  42. case (uint16_t)ErrorCode::MOVE_PULLEY_FAILED:
  43. return FindErrorIndex(ERR_MECHANICAL_PULLEY_CANNOT_MOVE);
  44. case (uint16_t)ErrorCode::HOMING_SELECTOR_FAILED:
  45. return FindErrorIndex(ERR_MECHANICAL_SELECTOR_CANNOT_HOME);
  46. case (uint16_t)ErrorCode::MOVE_SELECTOR_FAILED:
  47. return FindErrorIndex(ERR_MECHANICAL_SELECTOR_CANNOT_MOVE);
  48. case (uint16_t)ErrorCode::HOMING_IDLER_FAILED:
  49. return FindErrorIndex(ERR_MECHANICAL_IDLER_CANNOT_HOME);
  50. case (uint16_t)ErrorCode::MMU_NOT_RESPONDING:
  51. return FindErrorIndex(ERR_MECHANICAL_IDLER_CANNOT_MOVE);
  52. case (uint16_t)ErrorCode::PROTOCOL_ERROR:
  53. return FindErrorIndex(ERR_CONNECT_COMMUNICATION_ERROR);
  54. case (uint16_t)ErrorCode::FILAMENT_ALREADY_LOADED:
  55. return FindErrorIndex(ERR_SYSTEM_FILAMENT_ALREADY_LOADED);
  56. case (uint16_t)ErrorCode::INVALID_TOOL:
  57. return FindErrorIndex(ERR_SYSTEM_INVALID_TOOL);
  58. case (uint16_t)ErrorCode::QUEUE_FULL:
  59. return FindErrorIndex(ERR_SYSTEM_QUEUE_FULL);
  60. case (uint16_t)ErrorCode::VERSION_MISMATCH:
  61. return FindErrorIndex(ERR_SYSTEM_FW_UPDATE_NEEDED);
  62. case (uint16_t)ErrorCode::INTERNAL:
  63. return FindErrorIndex(ERR_SYSTEM_FW_RUNTIME_ERROR);
  64. case (uint16_t)ErrorCode::FINDA_VS_EEPROM_DISREPANCY:
  65. return FindErrorIndex(ERR_SYSTEM_UNLOAD_MANUALLY);
  66. }
  67. // TMC-related errors - multiple of these can occur at once
  68. // - 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)
  69. // By carefully ordering the checks here we can prioritize the errors being reported to the user.
  70. if (ec & (uint16_t)ErrorCode::TMC_PULLEY_BIT) {
  71. if (ec & (uint16_t)ErrorCode::TMC_IOIN_MISMATCH)
  72. return FindErrorIndex(ERR_ELECTRICAL_PULLEY_TMC_DRIVER_ERROR);
  73. if (ec & (uint16_t)ErrorCode::TMC_RESET)
  74. return FindErrorIndex(ERR_ELECTRICAL_PULLEY_TMC_DRIVER_RESET);
  75. if (ec & (uint16_t)ErrorCode::TMC_UNDERVOLTAGE_ON_CHARGE_PUMP)
  76. return FindErrorIndex(ERR_ELECTRICAL_PULLEY_TMC_UNDERVOLTAGE_ERROR);
  77. if (ec & (uint16_t)ErrorCode::TMC_SHORT_TO_GROUND)
  78. return FindErrorIndex(ERR_ELECTRICAL_PULLEY_TMC_DRIVER_SHORTED);
  79. if (ec & (uint16_t)ErrorCode::TMC_OVER_TEMPERATURE_WARN)
  80. return FindErrorIndex(ERR_TEMPERATURE_PULLEY_WARNING_TMC_TOO_HOT);
  81. if (ec & (uint16_t)ErrorCode::TMC_OVER_TEMPERATURE_ERROR)
  82. return FindErrorIndex(ERR_TEMPERATURE_PULLEY_TMC_OVERHEAT_ERROR);
  83. } else if (ec & (uint16_t)ErrorCode::TMC_SELECTOR_BIT) {
  84. if (ec & (uint16_t)ErrorCode::TMC_IOIN_MISMATCH)
  85. return FindErrorIndex(ERR_ELECTRICAL_SELECTOR_TMC_DRIVER_ERROR);
  86. if (ec & (uint16_t)ErrorCode::TMC_RESET)
  87. return FindErrorIndex(ERR_ELECTRICAL_SELECTOR_TMC_DRIVER_RESET);
  88. if (ec & (uint16_t)ErrorCode::TMC_UNDERVOLTAGE_ON_CHARGE_PUMP)
  89. return FindErrorIndex(ERR_ELECTRICAL_SELECTOR_TMC_UNDERVOLTAGE_ERROR);
  90. if (ec & (uint16_t)ErrorCode::TMC_SHORT_TO_GROUND)
  91. return FindErrorIndex(ERR_ELECTRICAL_SELECTOR_TMC_DRIVER_SHORTED);
  92. if (ec & (uint16_t)ErrorCode::TMC_OVER_TEMPERATURE_WARN)
  93. return FindErrorIndex(ERR_TEMPERATURE_SELECTOR_WARNING_TMC_TOO_HOT);
  94. if (ec & (uint16_t)ErrorCode::TMC_OVER_TEMPERATURE_ERROR)
  95. return FindErrorIndex(ERR_TEMPERATURE_SELECTOR_TMC_OVERHEAT_ERROR);
  96. } else if (ec & (uint16_t)ErrorCode::TMC_IDLER_BIT) {
  97. if (ec & (uint16_t)ErrorCode::TMC_IOIN_MISMATCH)
  98. return FindErrorIndex(ERR_ELECTRICAL_IDLER_TMC_DRIVER_ERROR);
  99. if (ec & (uint16_t)ErrorCode::TMC_RESET)
  100. return FindErrorIndex(ERR_ELECTRICAL_IDLER_TMC_DRIVER_RESET);
  101. if (ec & (uint16_t)ErrorCode::TMC_UNDERVOLTAGE_ON_CHARGE_PUMP)
  102. return FindErrorIndex(ERR_ELECTRICAL_IDLER_TMC_UNDERVOLTAGE_ERROR);
  103. if (ec & (uint16_t)ErrorCode::TMC_SHORT_TO_GROUND)
  104. return FindErrorIndex(ERR_ELECTRICAL_IDLER_TMC_DRIVER_SHORTED);
  105. if (ec & (uint16_t)ErrorCode::TMC_OVER_TEMPERATURE_WARN)
  106. return FindErrorIndex(ERR_TEMPERATURE_IDLER_WARNING_TMC_TOO_HOT);
  107. if (ec & (uint16_t)ErrorCode::TMC_OVER_TEMPERATURE_ERROR)
  108. return FindErrorIndex(ERR_TEMPERATURE_IDLER_TMC_OVERHEAT_ERROR);
  109. }
  110. // if nothing got caught, return a generic runtime error
  111. return FindErrorIndex(ERR_SYSTEM_FW_RUNTIME_ERROR);
  112. }
  113. uint16_t PrusaErrorCode(uint8_t i){
  114. return pgm_read_word(errorCodes + i);
  115. }
  116. const char * const PrusaErrorTitle(uint8_t i){
  117. return (const char * const)pgm_read_ptr(errorTitles + i);
  118. }
  119. const char * const PrusaErrorDesc(uint8_t i){
  120. return (const char * const)pgm_read_ptr(errorDescs + i);
  121. }
  122. uint8_t PrusaErrorButtons(uint8_t i){
  123. return pgm_read_byte(errorButtons + i);
  124. }
  125. const char * const PrusaErrorButtonTitle(uint8_t bi){
  126. // -1 represents the hidden NoOperation button which is not drawn in any way
  127. return (const char * const)pgm_read_ptr(btnOperation + bi - 1);
  128. }
  129. const char * const PrusaErrorButtonMore(){
  130. return btnMore;
  131. }
  132. } // namespace MMU2