From 0e7f28f53bea199d9b33350c918dd42f56aba36c Mon Sep 17 00:00:00 2001 From: Caila Marashaj <98041399+caila-marashaj@users.noreply.github.com> Date: Tue, 3 Dec 2024 17:27:11 -0500 Subject: [PATCH 1/6] Grab reset flags on startup (#485) --- .../firmware/motor_task/motor_hardware.c | 101 +++++++++++++++++ .../firmware/motor_task/motor_hardware.h | 7 ++ .../firmware/motor_task/motor_policy.cpp | 4 + .../firmware/motor_task/motor_policy.hpp | 1 + .../heater-shaker/simulator/motor_thread.cpp | 8 ++ .../heater-shaker/heater-shaker/gcodes.hpp | 38 +++++++ .../heater-shaker/host_comms_task.hpp | 52 ++++++++- .../heater-shaker/heater-shaker/messages.hpp | 14 ++- .../heater-shaker/motor_task.hpp | 11 ++ .../heater-shaker/test/test_motor_policy.hpp | 7 ++ .../firmware/motor_hardware.h | 7 ++ .../firmware/motor_policy.hpp | 7 ++ .../test/test_motor_policy.hpp | 7 ++ .../thermocycler-gen2/gcodes.hpp | 38 +++++++ .../thermocycler-gen2/host_comms_task.hpp | 50 ++++++++- .../thermocycler-gen2/messages.hpp | 19 +++- .../thermocycler-gen2/motor_task.hpp | 11 ++ .../firmware/motor_task/motor_hardware.c | 102 ++++++++++++++++++ .../firmware/motor_task/motor_policy.cpp | 5 + .../simulator/motor_thread.cpp | 8 ++ 20 files changed, 487 insertions(+), 10 deletions(-) diff --git a/stm32-modules/heater-shaker/firmware/motor_task/motor_hardware.c b/stm32-modules/heater-shaker/firmware/motor_task/motor_hardware.c index 957b82885..f43dd4c01 100644 --- a/stm32-modules/heater-shaker/firmware/motor_task/motor_hardware.c +++ b/stm32-modules/heater-shaker/firmware/motor_task/motor_hardware.c @@ -19,6 +19,39 @@ motor_hardware_handles *MOTOR_HW_HANDLE = NULL; static _Atomic int16_t motor_speed_buffer[MOTOR_SPEED_BUFFER_SIZE]; static _Atomic size_t motor_speed_buffer_itr = 0; +uint16_t reset_reason; + +enum RCC_FLAGS { + _NONE, + // high speed internal clock ready + HSIRDY, // = 1 + // high speed external clock ready + HSERDY, // = 2 + // main phase-locked loop clock ready + PLLRDY, // = 3 + // hsi48 clock ready + HSI48RDY, // = 4 + // low-speed external clock ready + LSERDY, // = 5 + // lse clock security system failure + LSECSSD, // = 6 + // low-speed internal clock ready + LSIRDY, // = 7 + // brown out + BORRST, // = 8 + // option byte-loader reset + OBLRST, // = 9 + // pin reset + PINRST, // = 10 + // software reset + SFTRST, // = 11 + // independent watchdog + IWDGRST, // = 12 + // window watchdog + WWDGRST, // = 13 + // low power reset + LPWRRST, // = 14 +}; static void MX_NVIC_Init(void) { @@ -491,6 +524,69 @@ static void DAC_Init(DAC_HandleTypeDef* dac) { HAL_DAC_SetValue(dac, SOLENOID_DAC_CHANNEL, DAC_ALIGN_8B_R, 0); } +static void save_reset_reason() { + // check various reset flags to see if the HAL RCC + // reset flag matches any of them + reset_reason = 0; + + // high speed internal clock ready + if (__HAL_RCC_GET_FLAG(RCC_FLAG_HSIRDY)) { + reset_reason |= HSIRDY; + } + // high speed external clock ready + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_HSERDY)) { + reset_reason |= HSERDY; + } + // main phase-locked loop clock ready + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_PLLRDY)) { + reset_reason |= PLLRDY; + } + // hsi48 clock ready + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_HSIRDY)) { + reset_reason |= HSI48RDY; + } + // low-speed external clock ready + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_LSERDY)) { + reset_reason |= LSERDY; + } + // lse clock security system failure + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_LSERDY)) { + reset_reason |= LSECSSD; + } + // low-speed internal clock ready + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_LSIRDY)) { + reset_reason |= LSIRDY; + } + // brown out + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_PORRST)) { + reset_reason |= BORRST; + } + // option byte-loader reset + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_OBLRST)) { + reset_reason |= OBLRST; + } + // pin reset + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_PINRST)) { + reset_reason |= PINRST; + } + // software reset + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_SFTRST)) { + reset_reason |= SFTRST; + } + // independent watchdog + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_IWDGRST)) { + reset_reason |= IWDGRST; + } + // window watchdog + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_WWDGRST)) { + reset_reason |= WWDGRST; + } + // low power reset + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_LPWRRST)) { + reset_reason |= LPWRRST; + } +} + /** * Initializes the Global MSP. */ @@ -814,6 +910,7 @@ void HAL_TIM_Base_MspDeInit(TIM_HandleTypeDef* htim_base) } void motor_hardware_setup(motor_hardware_handles* handles) { + save_reset_reason(); MOTOR_HW_HANDLE = handles; memset(motor_speed_buffer, 0, sizeof(motor_speed_buffer)); MX_GPIO_Init(); @@ -840,6 +937,10 @@ void motor_hardware_solenoid_release(DAC_HandleTypeDef* dac1) { HAL_DAC_SetValue(dac1, SOLENOID_DAC_CHANNEL, DAC_ALIGN_8B_R, 0); } +uint16_t motor_hardware_reset_reason() { + return reset_reason; +} + void motor_hardware_plate_lock_on(TIM_HandleTypeDef* tim3, float power) { float power_scale = fabs(power); if (power_scale > 1.f) { diff --git a/stm32-modules/heater-shaker/firmware/motor_task/motor_hardware.h b/stm32-modules/heater-shaker/firmware/motor_task/motor_hardware.h index 6ed795616..32a6ad935 100644 --- a/stm32-modules/heater-shaker/firmware/motor_task/motor_hardware.h +++ b/stm32-modules/heater-shaker/firmware/motor_task/motor_hardware.h @@ -37,6 +37,13 @@ bool motor_hardware_plate_lock_sensor_read(uint16_t GPIO_Pin); void HAL_TIM_MspPostInit(TIM_HandleTypeDef* htim); +/** + * @brief Returns the reason saved upon startup for the last + * reset of the software. + * @return a flag containing the reason for reset. + * */ +uint16_t motor_hardware_reset_reason(); + /** * @brief To be called every time the motor control library updates its speed * measurement. This function adds new speed values to a circular buffer, diff --git a/stm32-modules/heater-shaker/firmware/motor_task/motor_policy.cpp b/stm32-modules/heater-shaker/firmware/motor_task/motor_policy.cpp index d03ce32d6..6b509ac5c 100644 --- a/stm32-modules/heater-shaker/firmware/motor_task/motor_policy.cpp +++ b/stm32-modules/heater-shaker/firmware/motor_task/motor_policy.cpp @@ -131,3 +131,7 @@ auto MotorPolicy::get_serial_number(void) -> std::array { return _serial.get_serial_number(); } + +auto MotorPolicy::last_reset_reason() const -> uint16_t { + return motor_hardware_reset_reason(); +} \ No newline at end of file diff --git a/stm32-modules/heater-shaker/firmware/motor_task/motor_policy.hpp b/stm32-modules/heater-shaker/firmware/motor_task/motor_policy.hpp index c86e2b0f7..c6d145950 100644 --- a/stm32-modules/heater-shaker/firmware/motor_task/motor_policy.hpp +++ b/stm32-modules/heater-shaker/firmware/motor_task/motor_policy.hpp @@ -46,6 +46,7 @@ class MotorPolicy { auto plate_lock_closed_sensor_read() -> bool; auto get_serial_number(void) -> std::array; + auto last_reset_reason() const -> uint16_t; private: static constexpr uint16_t MAX_SOLENOID_CURRENT_MA = 330; diff --git a/stm32-modules/heater-shaker/simulator/motor_thread.cpp b/stm32-modules/heater-shaker/simulator/motor_thread.cpp index bd7d3cc9f..ac3e1fcf6 100644 --- a/stm32-modules/heater-shaker/simulator/motor_thread.cpp +++ b/stm32-modules/heater-shaker/simulator/motor_thread.cpp @@ -109,6 +109,12 @@ struct SimMotorPolicy { } } + auto last_reset_reason() -> uint16_t { return reset_reason; } + + auto set_last_reset_reason(uint16_t sim_reason) -> void { + reset_reason = sim_reason; + } + private: int16_t rpm_setpoint = 0; int16_t rpm_current = 0; @@ -116,6 +122,8 @@ struct SimMotorPolicy { float sim_plate_lock_power = 0; bool sim_plate_lock_enabled = false; bool sim_plate_lock_braked = false; + // Simulated reset reason from HAL RCC flag + uint16_t reset_reason = 0; }; struct motor_thread::TaskControlBlock { diff --git a/stm32-modules/include/heater-shaker/heater-shaker/gcodes.hpp b/stm32-modules/include/heater-shaker/heater-shaker/gcodes.hpp index f12dfce37..b6cecc649 100644 --- a/stm32-modules/include/heater-shaker/heater-shaker/gcodes.hpp +++ b/stm32-modules/include/heater-shaker/heater-shaker/gcodes.hpp @@ -641,6 +641,44 @@ struct GetSystemInfo { } }; +struct GetResetReason { + /* + * M114- GetResetReason retrieves the value of the RCC reset flag + * that was captured at the beginning of the hardware setup + * */ + using ParseResult = std::optional; + static constexpr auto prefix = std::array{'M', '1', '1', '4'}; + + template + requires std::forward_iterator && + std::sized_sentinel_for + static auto write_response_into(InputIt buf, InLimit limit, uint16_t reason) + -> InputIt { + int res = 0; + // print a hexadecimal representation of the reset flags + res = snprintf(&*buf, (limit - buf), "M114 Last Reset Reason: %X OK\n", + reason); + if (res <= 0) { + return buf; + } + return buf + res; + } + template + requires std::forward_iterator && + std::sized_sentinel_for + static auto parse(const InputIt& input, Limit limit) + -> std::pair { + auto working = prefix_matches(input, limit, prefix); + if (working == input) { + return std::make_pair(ParseResult(), input); + } + if (working != limit && !std::isspace(*working)) { + return std::make_pair(ParseResult(), input); + } + return std::make_pair(ParseResult(GetResetReason()), working); + } +}; + struct SetSerialNumber { /* ** Set Serial Number uses a random gcode, M996, adjacent to the firmware diff --git a/stm32-modules/include/heater-shaker/heater-shaker/host_comms_task.hpp b/stm32-modules/include/heater-shaker/heater-shaker/host_comms_task.hpp index a6a4111f6..c240d8acd 100644 --- a/stm32-modules/include/heater-shaker/heater-shaker/host_comms_task.hpp +++ b/stm32-modules/include/heater-shaker/heater-shaker/host_comms_task.hpp @@ -48,7 +48,7 @@ class HostCommsTask { gcode::GetPlateLockStateDebug, gcode::SetLEDDebug, gcode::IdentifyModuleStartLED, gcode::IdentifyModuleStopLED, gcode::SetOffsetConstants, gcode::GetOffsetConstants, - gcode::DeactivateHeater>; + gcode::DeactivateHeater, gcode::GetResetReason>; using AckOnlyCache = AckCache<8, gcode::SetRPM, gcode::SetTemperature, gcode::SetAcceleration, gcode::SetPIDConstants, @@ -66,6 +66,7 @@ class HostCommsTask { using GetPlateLockStateDebugCache = AckCache<8, gcode::GetPlateLockStateDebug>; using GetOffsetConstantsCache = AckCache<8, gcode::GetOffsetConstants>; + using GetResetReasonCache = AckCache<8, gcode::GetResetReason>; public: static constexpr size_t TICKS_TO_WAIT_ON_SEND = 10; @@ -88,7 +89,9 @@ class HostCommsTask { // NOLINTNEXTLINE(readability-redundant-member-init) get_plate_lock_state_debug_cache(), // NOLINTNEXTLINE(readability-redundant-member-init) - get_offset_constants_cache() {} + get_offset_constants_cache(), + // NOLINTNEXTLINE(readability-redundant-member-init) + get_reset_reason_cache() {} HostCommsTask(const HostCommsTask& other) = delete; auto operator=(const HostCommsTask& other) -> HostCommsTask& = delete; HostCommsTask(HostCommsTask&& other) noexcept = delete; @@ -478,6 +481,50 @@ class HostCommsTask { return std::make_pair(true, tx_into); } + template + requires std::forward_iterator && + std::sized_sentinel_for + auto visit_message(const messages::GetResetReasonResponse& response, + InputIt tx_into, InputLimit tx_limit) -> InputIt { + auto cache_entry = + get_reset_reason_cache.remove_if_present(response.responding_to_id); + return std::visit( + [tx_into, tx_limit, response](auto cache_element) { + using T = std::decay_t; + if constexpr (std::is_same_v) { + return errors::write_into( + tx_into, tx_limit, + errors::ErrorCode::BAD_MESSAGE_ACKNOWLEDGEMENT); + } else { + return cache_element.write_response_into(tx_into, tx_limit, + response.reason); + } + }, + cache_entry); + } + + template + requires std::forward_iterator && + std::sized_sentinel_for + auto visit_gcode(const gcode::GetResetReason& gcode, InputIt tx_into, + InputLimit tx_limit) -> std::pair { + auto id = get_reset_reason_cache.add(gcode); + if (id == 0) { + return std::make_pair( + false, errors::write_into(tx_into, tx_limit, + errors::ErrorCode::GCODE_CACHE_FULL)); + } + auto message = messages::GetResetReasonMessage{.id = id}; + if (!task_registry->motor->get_message_queue().try_send( + message, TICKS_TO_WAIT_ON_SEND)) { + auto wrote_to = errors::write_into( + tx_into, tx_limit, errors::ErrorCode::INTERNAL_QUEUE_FULL); + get_reset_reason_cache.remove_if_present(id); + return std::make_pair(false, wrote_to); + } + return std::make_pair(true, tx_into); + } + template requires std::forward_iterator && std::sized_sentinel_for @@ -1053,6 +1100,7 @@ class HostCommsTask { GetPlateLockStateCache get_plate_lock_state_cache; GetPlateLockStateDebugCache get_plate_lock_state_debug_cache; GetOffsetConstantsCache get_offset_constants_cache; + GetResetReasonCache get_reset_reason_cache; bool may_connect_latch = true; }; diff --git a/stm32-modules/include/heater-shaker/heater-shaker/messages.hpp b/stm32-modules/include/heater-shaker/heater-shaker/messages.hpp index 49827ba14..0fc994c4f 100644 --- a/stm32-modules/include/heater-shaker/heater-shaker/messages.hpp +++ b/stm32-modules/include/heater-shaker/heater-shaker/messages.hpp @@ -269,6 +269,15 @@ struct GetOffsetConstantsResponse { double const_b, const_c; }; +struct GetResetReasonMessage { + uint32_t id; +}; + +struct GetResetReasonResponse { + uint32_t responding_to_id; + uint16_t reason; +}; + struct DeactivateHeaterMessage { uint32_t id; }; @@ -295,7 +304,7 @@ using MotorMessage = ::std::variant< ActuateSolenoidMessage, SetPlateLockPowerMessage, OpenPlateLockMessage, ClosePlateLockMessage, SetPIDConstantsMessage, PlateLockComplete, GetPlateLockStateMessage, GetPlateLockStateDebugMessage, - CheckPlateLockStatusMessage>; + CheckPlateLockStatusMessage, GetResetReasonMessage>; using SystemMessage = ::std::variant; + GetSystemInfoResponse, GetOffsetConstantsResponse, + GetResetReasonResponse>; }; // namespace messages diff --git a/stm32-modules/include/heater-shaker/heater-shaker/motor_task.hpp b/stm32-modules/include/heater-shaker/heater-shaker/motor_task.hpp index b52b78c83..f74e84136 100644 --- a/stm32-modules/include/heater-shaker/heater-shaker/motor_task.hpp +++ b/stm32-modules/include/heater-shaker/heater-shaker/motor_task.hpp @@ -531,6 +531,17 @@ class MotorTask { static_cast(get_message_queue().try_send(check_state_message)); } + template + auto visit_message(const messages::GetResetReasonMessage& msg, + Policy& policy) -> void { + auto reason = policy.last_reset_reason(); + + auto response = messages::GetResetReasonResponse{ + .responding_to_id = msg.id, .reason = reason}; + static_cast(task_registry->comms->get_message_queue().try_send( + messages::HostCommsMessage(response))); + } + template auto visit_message(const messages::CheckPlateLockStatusMessage& msg, Policy& policy) -> void { diff --git a/stm32-modules/include/heater-shaker/test/test_motor_policy.hpp b/stm32-modules/include/heater-shaker/test/test_motor_policy.hpp index 78bc83cee..32656a7ee 100644 --- a/stm32-modules/include/heater-shaker/test/test_motor_policy.hpp +++ b/stm32-modules/include/heater-shaker/test/test_motor_policy.hpp @@ -61,6 +61,12 @@ class TestMotorPolicy { auto get_serial_number(void) -> std::array; + auto last_reset_reason() const -> uint16_t { return _reset_reason; } + + auto set_last_reset_reason(uint16_t sim_reason) -> void { + _reset_reason = sim_reason; + } + private: int16_t target_rpm; int16_t current_rpm; @@ -76,4 +82,5 @@ class TestMotorPolicy { double overridden_kp = 0.0; double overridden_kd = 0.0; bool plate_lock_braked = false; + uint16_t _reset_reason = 0; }; diff --git a/stm32-modules/include/thermocycler-gen2/firmware/motor_hardware.h b/stm32-modules/include/thermocycler-gen2/firmware/motor_hardware.h index b55888f4b..f455aff7b 100644 --- a/stm32-modules/include/thermocycler-gen2/firmware/motor_hardware.h +++ b/stm32-modules/include/thermocycler-gen2/firmware/motor_hardware.h @@ -184,6 +184,13 @@ void motor_hardware_seal_switch_set_disarmed(); */ void motor_hardware_seal_switch_interrupt(); +/** + * @brief Returns the reason saved upon startup for the last + * reset of the software. + * @return a flag containing the reasons for reset. + * */ +uint16_t motor_hardware_reset_reason(); + #ifdef __cplusplus } // extern "C" #endif // __cplusplus diff --git a/stm32-modules/include/thermocycler-gen2/firmware/motor_policy.hpp b/stm32-modules/include/thermocycler-gen2/firmware/motor_policy.hpp index 05f9ed16e..b633ee378 100644 --- a/stm32-modules/include/thermocycler-gen2/firmware/motor_policy.hpp +++ b/stm32-modules/include/thermocycler-gen2/firmware/motor_policy.hpp @@ -186,6 +186,13 @@ class MotorPolicy { */ [[nodiscard]] auto seal_switches_are_shared() const -> bool; + /** + * @brief Returns the reason saved upon startup for the last + * reset of the software. + * @return a string describing the reason for reset. + * */ + [[nodiscard]] auto last_reset_reason() const -> uint16_t; + /** * @brief Call the seal callback function * diff --git a/stm32-modules/include/thermocycler-gen2/test/test_motor_policy.hpp b/stm32-modules/include/thermocycler-gen2/test/test_motor_policy.hpp index ce5434bc5..2ccbb8589 100644 --- a/stm32-modules/include/thermocycler-gen2/test/test_motor_policy.hpp +++ b/stm32-modules/include/thermocycler-gen2/test/test_motor_policy.hpp @@ -122,6 +122,12 @@ class TestMotorPolicy : public TestTMC2130Policy { _shared_switch_lines = shared; } + auto last_reset_reason() const -> uint16_t { return _reset_reason; } + + auto set_last_reset_reason(uint16_t sim_reason) -> void { + _reset_reason = sim_reason; + } + private: // Solenoid is engaged when unpowered bool _solenoid_engaged = true; @@ -138,6 +144,7 @@ class TestMotorPolicy : public TestTMC2130Policy { bool _extension_switch_armed = false; bool _retraction_switch_armed = false; double _lid_rpm = 0; + uint16_t _reset_reason = 0; // Default to shared switch lines (pre-DVT) bool _shared_switch_lines = true; Callback _callback; diff --git a/stm32-modules/include/thermocycler-gen2/thermocycler-gen2/gcodes.hpp b/stm32-modules/include/thermocycler-gen2/thermocycler-gen2/gcodes.hpp index b2c28a47a..40e58c73a 100644 --- a/stm32-modules/include/thermocycler-gen2/thermocycler-gen2/gcodes.hpp +++ b/stm32-modules/include/thermocycler-gen2/thermocycler-gen2/gcodes.hpp @@ -146,6 +146,44 @@ struct GetSystemInfo { } }; +struct GetResetReason { + /* + * M114- GetResetReason retrieves the value of the RCC reset flag + * that was captured at the beginning of the hardware setup + * */ + using ParseResult = std::optional; + static constexpr auto prefix = std::array{'M', '1', '1', '4'}; + + template + requires std::forward_iterator && + std::sized_sentinel_for + static auto write_response_into(InputIt buf, InLimit limit, uint16_t reason) + -> InputIt { + int res = 0; + // print a hexadecimal representation of the reset flags + res = snprintf(&*buf, (limit - buf), "M114 Last Reset Reason: %X OK\n", + reason); + if (res <= 0) { + return buf; + } + return buf + res; + } + template + requires std::forward_iterator && + std::sized_sentinel_for + static auto parse(const InputIt& input, Limit limit) + -> std::pair { + auto working = prefix_matches(input, limit, prefix); + if (working == input) { + return std::make_pair(ParseResult(), input); + } + if (working != limit && !std::isspace(*working)) { + return std::make_pair(ParseResult(), input); + } + return std::make_pair(ParseResult(GetResetReason()), working); + } +}; + struct SetSerialNumber { /* ** Set Serial Number uses a random gcode, M996, adjacent to the firmware diff --git a/stm32-modules/include/thermocycler-gen2/thermocycler-gen2/host_comms_task.hpp b/stm32-modules/include/thermocycler-gen2/thermocycler-gen2/host_comms_task.hpp index 4cb1523c6..175d79f8b 100644 --- a/stm32-modules/include/thermocycler-gen2/thermocycler-gen2/host_comms_task.hpp +++ b/stm32-modules/include/thermocycler-gen2/thermocycler-gen2/host_comms_task.hpp @@ -52,7 +52,7 @@ class HostCommsTask { gcode::GetOffsetConstants, gcode::OpenLid, gcode::CloseLid, gcode::LiftPlate, gcode::DeactivateAll, gcode::GetBoardRevision, gcode::GetLidSwitches, gcode::GetFrontButton, gcode::SetLidFans, - gcode::SetLightsDebug>; + gcode::SetLightsDebug, gcode::GetResetReason>; using AckOnlyCache = AckCache<8, gcode::EnterBootloader, gcode::SetSerialNumber, gcode::ActuateSolenoid, gcode::ActuateLidStepperDebug, @@ -72,6 +72,7 @@ class HostCommsTask { using GetLidStatusCache = AckCache<8, gcode::GetLidStatus>; using GetOffsetConstantsCache = AckCache<8, gcode::GetOffsetConstants>; using SealStepperDebugCache = AckCache<8, gcode::ActuateSealStepperDebug>; + using GetResetReasonCache = AckCache<8, gcode::GetResetReason>; // This is a two-stage message since both the Plate and Lid tasks have // to respond. using GetThermalPowerCache = AckCache<8, gcode::GetThermalPowerDebug, @@ -115,6 +116,8 @@ class HostCommsTask { // NOLINTNEXTLINE(readability-redundant-member-init) deactivate_all_cache(), // NOLINTNEXTLINE(readability-redundant-member-init) + get_reset_reason_cache(), + // NOLINTNEXTLINE(readability-redundant-member-init) get_switch_cache() {} HostCommsTask(const HostCommsTask& other) = delete; auto operator=(const HostCommsTask& other) -> HostCommsTask& = delete; @@ -460,6 +463,28 @@ class HostCommsTask { cache_entry); } + template + requires std::forward_iterator && + std::sized_sentinel_for + auto visit_message(const messages::GetResetReasonResponse& response, + InputIt tx_into, InputLimit tx_limit) -> InputIt { + auto cache_entry = + get_reset_reason_cache.remove_if_present(response.responding_to_id); + return std::visit( + [tx_into, tx_limit, response](auto cache_element) { + using T = std::decay_t; + if constexpr (std::is_same_v) { + return errors::write_into( + tx_into, tx_limit, + errors::ErrorCode::BAD_MESSAGE_ACKNOWLEDGEMENT); + } else { + return cache_element.write_response_into(tx_into, tx_limit, + response.reason); + } + }, + cache_entry); + } + template requires std::forward_iterator && std::sized_sentinel_for @@ -1232,6 +1257,28 @@ class HostCommsTask { return std::make_pair(true, tx_into); } + template + requires std::forward_iterator && + std::sized_sentinel_for + auto visit_gcode(const gcode::GetResetReason& gcode, InputIt tx_into, + InputLimit tx_limit) -> std::pair { + auto id = get_reset_reason_cache.add(gcode); + if (id == 0) { + return std::make_pair( + false, errors::write_into(tx_into, tx_limit, + errors::ErrorCode::GCODE_CACHE_FULL)); + } + auto message = messages::GetResetReasonMessage{.id = id}; + if (!task_registry->motor->get_message_queue().try_send( + message, TICKS_TO_WAIT_ON_SEND)) { + auto wrote_to = errors::write_into( + tx_into, tx_limit, errors::ErrorCode::INTERNAL_QUEUE_FULL); + get_reset_reason_cache.remove_if_present(id); + return std::make_pair(false, wrote_to); + } + return std::make_pair(true, tx_into); + } + template requires std::forward_iterator && std::sized_sentinel_for @@ -1524,6 +1571,7 @@ class HostCommsTask { SealStepperDebugCache seal_stepper_debug_cache; GetThermalPowerCache get_thermal_power_cache; DeactivateAllCache deactivate_all_cache; + GetResetReasonCache get_reset_reason_cache; GetSwitchCache get_switch_cache; bool may_connect_latch = true; }; diff --git a/stm32-modules/include/thermocycler-gen2/thermocycler-gen2/messages.hpp b/stm32-modules/include/thermocycler-gen2/thermocycler-gen2/messages.hpp index 00685eec7..b135d72ef 100644 --- a/stm32-modules/include/thermocycler-gen2/thermocycler-gen2/messages.hpp +++ b/stm32-modules/include/thermocycler-gen2/thermocycler-gen2/messages.hpp @@ -353,6 +353,15 @@ struct GetLidStatusResponse { motor_util::SealStepper::Status seal; }; +struct GetResetReasonMessage { + uint32_t id; +}; + +struct GetResetReasonResponse { + uint32_t responding_to_id; + uint16_t reason; +}; + struct OpenLidMessage { uint32_t id; }; @@ -421,9 +430,9 @@ using HostCommsMessage = ::std::variant< ForceUSBDisconnectMessage, GetSystemInfoResponse, GetLidTemperatureDebugResponse, GetPlateTemperatureDebugResponse, GetPlateTempResponse, GetLidTempResponse, GetSealDriveStatusResponse, - GetLidStatusResponse, GetPlatePowerResponse, GetLidPowerResponse, - GetOffsetConstantsResponse, SealStepperDebugResponse, DeactivateAllResponse, - GetLidSwitchesResponse, GetFrontButtonResponse>; + GetLidStatusResponse, GetResetReasonResponse, GetPlatePowerResponse, + GetLidPowerResponse, GetOffsetConstantsResponse, SealStepperDebugResponse, + DeactivateAllResponse, GetLidSwitchesResponse, GetFrontButtonResponse>; using ThermalPlateMessage = ::std::variant; + GetResetReasonMessage, OpenLidMessage, CloseLidMessage, PlateLiftMessage, + FrontButtonPressMessage, GetLidSwitchesMessage>; }; // namespace messages diff --git a/stm32-modules/include/thermocycler-gen2/thermocycler-gen2/motor_task.hpp b/stm32-modules/include/thermocycler-gen2/thermocycler-gen2/motor_task.hpp index b8589adb5..53dd3c5d6 100644 --- a/stm32-modules/include/thermocycler-gen2/thermocycler-gen2/motor_task.hpp +++ b/stm32-modules/include/thermocycler-gen2/thermocycler-gen2/motor_task.hpp @@ -580,6 +580,17 @@ class MotorTask { messages::HostCommsMessage(response))); } + template + auto visit_message(const messages::GetResetReasonMessage& msg, + Policy& policy) -> void { + auto reason = policy.last_reset_reason(); + + auto response = messages::GetResetReasonResponse{ + .responding_to_id = msg.id, .reason = reason}; + static_cast(_task_registry->comms->get_message_queue().try_send( + messages::HostCommsMessage(response))); + } + template auto visit_message(const messages::OpenLidMessage& msg, Policy& policy) -> void { diff --git a/stm32-modules/thermocycler-gen2/firmware/motor_task/motor_hardware.c b/stm32-modules/thermocycler-gen2/firmware/motor_task/motor_hardware.c index b4a78fc58..cf760aa3b 100644 --- a/stm32-modules/thermocycler-gen2/firmware/motor_task/motor_hardware.c +++ b/stm32-modules/thermocycler-gen2/firmware/motor_task/motor_hardware.c @@ -184,6 +184,38 @@ static motor_hardware_t _motor_hardware = { } }; +enum RCC_FLAGS { + NONE, + // high speed internal clock ready + HSIRDY, // = 1 + // high speed external clock ready + HSERDY, // = 2 + // main phase-locked loop clock ready + PLLRDY, // = 3 + // hsi48 clock ready + HSI48RDY, // = 4 + // low-speed external clock ready + LSERDY, // = 5 + // lse clock security system failure + LSECSSD, // = 6 + // low-speed internal clock ready + LSIRDY, // = 7 + // brown out + BORRST, // = 8 + // option byte-loader reset + OBLRST, // = 9 + // pin reset + PINRST, // = 10 + // software reset + SFTRST, // = 11 + // independent watchdog + IWDGRST, // = 12 + // window watchdog + WWDGRST, // = 13 + // low power reset + LPWRRST, // = 14 +}; + // ---------------------------------------------------------------------------- // Local function declaration @@ -193,11 +225,14 @@ static void init_dac1(DAC_HandleTypeDef* hdac); static void init_tim2(TIM_HandleTypeDef* htim); static void init_tim6(TIM_HandleTypeDef* htim); static bool lid_active(); +static void save_reset_reason(); +uint16_t reset_reason; // ---------------------------------------------------------------------------- // Public function implementation void motor_hardware_setup(const motor_hardware_callbacks* callbacks) { + save_reset_reason(); configASSERT(callbacks != NULL); configASSERT(callbacks->lid_stepper_complete != NULL); configASSERT(callbacks->seal_stepper_tick != NULL); @@ -407,6 +442,10 @@ void motor_hardware_seal_switch_set_disarmed() { _motor_hardware.seal.retraction_switch_armed = false; } +uint16_t motor_hardware_reset_reason() { + return reset_reason; +} + // ---------------------------------------------------------------------------- // Local function implementation @@ -623,6 +662,69 @@ static bool lid_active() { != HAL_TIM_CHANNEL_STATE_READY; } +static void save_reset_reason() { + // check various reset flags to see if the HAL RCC + // reset flag matches any of them + reset_reason = 0; + + // high speed internal clock ready + if (__HAL_RCC_GET_FLAG(RCC_FLAG_HSIRDY)) { + reset_reason |= HSIRDY; + } + // high speed external clock ready + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_HSERDY)) { + reset_reason |= HSERDY; + } + // main phase-locked loop clock ready + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_PLLRDY)) { + reset_reason |= PLLRDY; + } + // hsi48 clock ready + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_HSI48RDY)) { + reset_reason |= HSI48RDY; + } + // low-speed external clock ready + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_LSERDY)) { + reset_reason |= LSERDY; + } + // lse clock security system failure + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_LSECSSD)) { + reset_reason |= LSECSSD; + } + // low-speed internal clock ready + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_LSIRDY)) { + reset_reason |= LSIRDY; + } + // brown out + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_BORRST)) { + reset_reason |= BORRST; + } + // option byte-loader reset + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_OBLRST)) { + reset_reason |= OBLRST; + } + // pin reset + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_PINRST)) { + reset_reason |= PINRST; + } + // software reset + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_SFTRST)) { + reset_reason |= SFTRST; + } + // independent watchdog + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_IWDGRST)) { + reset_reason |= IWDGRST; + } + // window watchdog + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_WWDGRST)) { + reset_reason |= WWDGRST; + } + // low power reset + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_LPWRRST)) { + reset_reason |= LPWRRST; + } +} + // ---------------------------------------------------------------------------- // Overwritten HAL functions diff --git a/stm32-modules/thermocycler-gen2/firmware/motor_task/motor_policy.cpp b/stm32-modules/thermocycler-gen2/firmware/motor_task/motor_policy.cpp index be9cb6409..713f26c04 100644 --- a/stm32-modules/thermocycler-gen2/firmware/motor_task/motor_policy.cpp +++ b/stm32-modules/thermocycler-gen2/firmware/motor_task/motor_policy.cpp @@ -151,4 +151,9 @@ auto MotorPolicy::seal_switch_set_disarmed() -> void { // NOLINTNEXTLINE(readability-convert-member-functions-to-static) [[nodiscard]] auto MotorPolicy::seal_switches_are_shared() const -> bool { return _shared_seal_switch_lines; +} + +// NOLINTNEXTLINE(readability-convert-member-functions-to-static) +[[nodiscard]] auto MotorPolicy::last_reset_reason() const -> uint16_t { + return motor_hardware_reset_reason(); } \ No newline at end of file diff --git a/stm32-modules/thermocycler-gen2/simulator/motor_thread.cpp b/stm32-modules/thermocycler-gen2/simulator/motor_thread.cpp index 70c7868d8..1733f627f 100644 --- a/stm32-modules/thermocycler-gen2/simulator/motor_thread.cpp +++ b/stm32-modules/thermocycler-gen2/simulator/motor_thread.cpp @@ -119,6 +119,12 @@ class SimMotorPolicy : public SimTMC2130Policy { auto seal_switches_are_shared() -> bool { return false; } + auto last_reset_reason() -> uint16_t { return reset_reason; } + + auto set_last_reset_reason(uint16_t sim_reason) -> void { + reset_reason = sim_reason; + } + private: // Lowest position the lid can move before stalling static constexpr uint32_t min_lid_steps = @@ -132,6 +138,8 @@ class SimMotorPolicy : public SimTMC2130Policy { static constexpr double open_switch_pos_angle = 90.0F; // Simulated width of each lid switch in degrees static constexpr double switch_width_angle = 1.0F; + // Simulated reset reason from HAL RCC flag + uint16_t reset_reason = 0; // Check if open switch is triggered [[nodiscard]] static auto open_switch_triggered(double angle) -> bool { From 46601d6bcf38a460024332cc85468b974f92063c Mon Sep 17 00:00:00 2001 From: Alise Au <20424172+ahiuchingau@users.noreply.github.com> Date: Wed, 4 Dec 2024 15:52:16 -0500 Subject: [PATCH 2/6] fix(stacker): small bug fixes for EVT (#490) * EXEC-1036 fix stop motor message handling * update extents values * make sure we start motor timer when we are about to move --- .../flex-stacker/firmware/motor_interrupt.hpp | 1 + .../flex-stacker/flex-stacker/motor_task.hpp | 31 +++++++++++-------- 2 files changed, 19 insertions(+), 13 deletions(-) diff --git a/stm32-modules/include/flex-stacker/firmware/motor_interrupt.hpp b/stm32-modules/include/flex-stacker/firmware/motor_interrupt.hpp index c879363ed..080ea2592 100644 --- a/stm32-modules/include/flex-stacker/firmware/motor_interrupt.hpp +++ b/stm32-modules/include/flex-stacker/firmware/motor_interrupt.hpp @@ -67,6 +67,7 @@ class MotorInterruptController { motor_util::MovementType::FixedDistance, steps); _policy->enable_motor(_id); _response_id = move_id; + _policy->start_motor_timer(_id); } auto start_move(Move move) -> void { motor_util::MotorState* state = move.motor_state; diff --git a/stm32-modules/include/flex-stacker/flex-stacker/motor_task.hpp b/stm32-modules/include/flex-stacker/flex-stacker/motor_task.hpp index 09c81066b..d8921326d 100644 --- a/stm32-modules/include/flex-stacker/flex-stacker/motor_task.hpp +++ b/stm32-modules/include/flex-stacker/flex-stacker/motor_task.hpp @@ -37,7 +37,7 @@ using Error = errors::ErrorCode; struct Defaults { struct X { static constexpr float SPEED = 200.0; - static constexpr float ACCELERATION = 50.0; + static constexpr float ACCELERATION = 1500.0; static constexpr float SPEED_DISCONT = 5.0; static constexpr float MM_PER_REV = @@ -45,13 +45,13 @@ struct Defaults { static constexpr float STEPS_PER_REV = 200; static constexpr float MICROSTEP = 16; - // switch-to-switch: 202.0 mm - 5.0 mm offset - static constexpr float FAST_HOME_DISTANCE = 197.0; + // switch-to-switch: 192.5 mm - 5.0 mm offset + static constexpr float FAST_HOME_DISTANCE = 187.5; }; struct Z { static constexpr float SPEED = 200.0; - static constexpr float ACCELERATION = 50.0; + static constexpr float ACCELERATION = 80.0; static constexpr float SPEED_DISCONT = 5.0; static constexpr float MM_PER_REV = @@ -59,13 +59,13 @@ struct Defaults { static constexpr float STEPS_PER_REV = 200; static constexpr float MICROSTEP = 16; - // switch-to-switch: 113.75 mm - 5.0 mm offset - static constexpr float FAST_HOME_DISTANCE = 108.75; + // switch-to-switch: 136.0 mm - 5.0 mm offset + static constexpr float FAST_HOME_DISTANCE = 131.0; }; struct L { - static constexpr float SPEED = 200.0; - static constexpr float ACCELERATION = 50.0; + static constexpr float SPEED = 100.0; + static constexpr float ACCELERATION = 100.0; static constexpr float SPEED_DISCONT = 5.0; static constexpr float MM_PER_REV = @@ -167,6 +167,13 @@ class MotorTask { return Error::NO_ERROR; } + auto stop_motors(uint32_t id = 0) -> void { + _z_controller.stop_movement(id, true); + _x_controller.stop_movement(id, false); + _l_controller.stop_movement(id, false); + _move_queue.reset(); + } + auto make_move(uint32_t id, MotorID motor_id, bool direction, float distance, bool has_next_move = false) -> Move { return Move{.motor_id = motor_id, @@ -312,9 +319,9 @@ class MotorTask { template auto visit_message(const messages::StopMotorMessage& m, Policy& policy) -> void { - static_cast(m); static_cast(policy); - controller_from_id(m.motor_id).stop_movement(m.id, true); + stop_motors(m.id); + send_ack_message(m.id); } template @@ -419,9 +426,7 @@ class MotorTask { // don't care about other interrupts return; } - _z_controller.stop_movement(0, true); - _x_controller.stop_movement(0, false); - _l_controller.stop_movement(0, false); + stop_motors(); } /** From 863df3163be7796fd465977962d3f2d3c32976c5 Mon Sep 17 00:00:00 2001 From: Alise Au <20424172+ahiuchingau@users.noreply.github.com> Date: Thu, 5 Dec 2024 11:41:12 -0500 Subject: [PATCH 3/6] fix(stacker): motor driver conf update (#492) * add drv conf for motor driver * decrease thigh threshold to make sure stallguard is not activated --- .../flex-stacker/flex-stacker/motor_driver_task.hpp | 9 ++++++--- 1 file changed, 6 insertions(+), 3 deletions(-) diff --git a/stm32-modules/include/flex-stacker/flex-stacker/motor_driver_task.hpp b/stm32-modules/include/flex-stacker/flex-stacker/motor_driver_task.hpp index 9f3fef392..dc156a056 100644 --- a/stm32-modules/include/flex-stacker/flex-stacker/motor_driver_task.hpp +++ b/stm32-modules/include/flex-stacker/flex-stacker/motor_driver_task.hpp @@ -31,13 +31,14 @@ static constexpr tmc2160::TMC2160RegisterMap motor_z_config{ .s2g_level = 0x6, .shortfilter = 1, .shortdelay = 0}, + .drvconf = {.bbmclks = 4}, .glob_scale = {.global_scaler = 0x0}, .ihold_irun = {.hold_current = 10, .run_current = 31, .hold_current_delay = 15}, .tpwmthrs = {.threshold = 0x80000}, .tcoolthrs = {.threshold = 0x2FF}, - .thigh = {.threshold = 0x81}, + .thigh = {.threshold = 0x1}, .chopconf = {.toff = 0b111, .hstrt = 0b100, .hend = 0b11, @@ -58,13 +59,14 @@ static constexpr tmc2160::TMC2160RegisterMap motor_x_config{ .s2g_level = 0x6, .shortfilter = 1, .shortdelay = 0}, + .drvconf = {.bbmclks = 4}, .glob_scale = {.global_scaler = 0x0}, .ihold_irun = {.hold_current = 12, .run_current = 31, .hold_current_delay = 7}, .tpwmthrs = {.threshold = 0x80000}, .tcoolthrs = {.threshold = 0x2FF}, - .thigh = {.threshold = 0x81}, + .thigh = {.threshold = 0x1}, .chopconf = {.toff = 0b111, .hstrt = 0b111, .hend = 0b1001, @@ -85,13 +87,14 @@ static constexpr tmc2160::TMC2160RegisterMap motor_l_config{ .s2g_level = 0x6, .shortfilter = 1, .shortdelay = 0}, + .drvconf = {.bbmclks = 4}, .glob_scale = {.global_scaler = 0x0}, .ihold_irun = {.hold_current = 7, .run_current = 8, .hold_current_delay = 7}, .tpwmthrs = {.threshold = 0x80000}, .tcoolthrs = {.threshold = 0x2FF}, - .thigh = {.threshold = 0x81}, + .thigh = {.threshold = 0x1}, .chopconf = {.toff = 0b111, .hstrt = 0b111, .hend = 0b1001, From 71ea8b2699143eb1ac743631b02cc14c60c85f4c Mon Sep 17 00:00:00 2001 From: Caila Marashaj <98041399+caila-marashaj@users.noreply.github.com> Date: Thu, 5 Dec 2024 17:30:37 -0500 Subject: [PATCH 4/6] feat(flex-stacker): grab reset reason (#491) --- .../firmware/system/system_hardware.c | 101 ++++++++++++++++++ .../firmware/system/system_policy.cpp | 5 + .../flex-stacker/firmware/system_hardware.h | 2 + .../flex-stacker/firmware/system_policy.hpp | 1 + .../flex-stacker/flex-stacker/gcodes.hpp | 38 +++++++ .../flex-stacker/host_comms_task.hpp | 52 ++++++++- .../flex-stacker/flex-stacker/messages.hpp | 13 ++- .../flex-stacker/flex-stacker/system_task.hpp | 11 ++ 8 files changed, 219 insertions(+), 4 deletions(-) diff --git a/stm32-modules/flex-stacker/firmware/system/system_hardware.c b/stm32-modules/flex-stacker/firmware/system/system_hardware.c index 233642fc1..0405f8a64 100644 --- a/stm32-modules/flex-stacker/firmware/system/system_hardware.c +++ b/stm32-modules/flex-stacker/firmware/system/system_hardware.c @@ -13,6 +13,106 @@ // address 4 in the bootable region is the address of the first instruction that // should run, aka the data that should be loaded into $pc. const uint32_t *const sysmem_boot_loc = (uint32_t*)SYSMEM_BOOT; +uint16_t reset_reason; + +enum RCC_FLAGS { + _NONE, + // high speed internal clock ready + HSIRDY, // = 1 + // high speed external clock ready + HSERDY, // = 2 + // main phase-locked loop clock ready + PLLRDY, // = 3 + // hsi48 clock ready + HSI48RDY, // = 4 + // low-speed external clock ready + LSERDY, // = 5 + // lse clock security system failure + LSECSSD, // = 6 + // low-speed internal clock ready + LSIRDY, // = 7 + // brown out + BORRST, // = 8 + // option byte-loader reset + OBLRST, // = 9 + // pin reset + PINRST, // = 10 + // software reset + SFTRST, // = 11 + // independent watchdog + IWDGRST, // = 12 + // window watchdog + WWDGRST, // = 13 + // low power reset + LPWRRST, // = 14 +}; + +static void save_reset_reason() { + // check various reset flags to see if the HAL RCC + // reset flag matches any of them + reset_reason = 0; + + // high speed internal clock ready + if (__HAL_RCC_GET_FLAG(RCC_FLAG_HSIRDY)) { + reset_reason |= HSIRDY; + } + // high speed external clock ready + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_HSERDY)) { + reset_reason |= HSERDY; + } + // main phase-locked loop clock ready + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_PLLRDY)) { + reset_reason |= PLLRDY; + } + // hsi48 clock ready + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_HSIRDY)) { + reset_reason |= HSI48RDY; + } + // low-speed external clock ready + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_LSERDY)) { + reset_reason |= LSERDY; + } + // lse clock security system failure + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_LSERDY)) { + reset_reason |= LSECSSD; + } + // low-speed internal clock ready + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_LSIRDY)) { + reset_reason |= LSIRDY; + } + // brown out + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_BORRST)) { + reset_reason |= BORRST; + } + // option byte-loader reset + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_OBLRST)) { + reset_reason |= OBLRST; + } + // pin reset + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_PINRST)) { + reset_reason |= PINRST; + } + // software reset + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_SFTRST)) { + reset_reason |= SFTRST; + } + // independent watchdog + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_IWDGRST)) { + reset_reason |= IWDGRST; + } + // window watchdog + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_WWDGRST)) { + reset_reason |= WWDGRST; + } + // low power reset + else if (__HAL_RCC_GET_FLAG(RCC_FLAG_LPWRRST)) { + reset_reason |= LPWRRST; + } +} + +uint16_t system_hardware_reset_reason() { + return reset_reason; +} /** PUBLIC FUNCTION IMPLEMENTATION */ @@ -60,6 +160,7 @@ void system_hardware_enter_bootloader(void) { } void system_hardware_gpio_init(void) { + save_reset_reason(); GPIO_InitTypeDef init = {0}; /* GPIO Ports Clock Enable */ diff --git a/stm32-modules/flex-stacker/firmware/system/system_policy.cpp b/stm32-modules/flex-stacker/firmware/system/system_policy.cpp index e6382f49a..33062a93e 100644 --- a/stm32-modules/flex-stacker/firmware/system/system_policy.cpp +++ b/stm32-modules/flex-stacker/firmware/system/system_policy.cpp @@ -65,3 +65,8 @@ auto SystemPolicy::initialize() -> void { system_hardware_gpio_init(); } auto SystemPolicy::get_door_closed() -> bool { return system_hardware_read_door_closed(); } + +// NOLINTNEXTLINE(readability-convert-member-functions-to-static) +auto SystemPolicy::last_reset_reason() const -> uint16_t { + return system_hardware_reset_reason(); +} diff --git a/stm32-modules/include/flex-stacker/firmware/system_hardware.h b/stm32-modules/include/flex-stacker/firmware/system_hardware.h index 51741b2ce..83ae9d0af 100644 --- a/stm32-modules/include/flex-stacker/firmware/system_hardware.h +++ b/stm32-modules/include/flex-stacker/firmware/system_hardware.h @@ -5,6 +5,7 @@ extern "C" { #endif // __cplusplus #include +#include #define HOPPER_DOR_CLOSED_PIN GPIO_PIN_13 #define HOPPER_DOR_CLOSED_GPIO_PORT GPIOC @@ -15,6 +16,7 @@ extern "C" { void system_hardware_enter_bootloader(void); void system_hardware_gpio_init(void); bool system_hardware_read_door_closed(void); +uint16_t system_hardware_reset_reason(void); #ifdef __cplusplus } // extern "C" diff --git a/stm32-modules/include/flex-stacker/firmware/system_policy.hpp b/stm32-modules/include/flex-stacker/firmware/system_policy.hpp index 87b7745fa..dfdede638 100644 --- a/stm32-modules/include/flex-stacker/firmware/system_policy.hpp +++ b/stm32-modules/include/flex-stacker/firmware/system_policy.hpp @@ -23,4 +23,5 @@ class SystemPolicy { -> errors::ErrorCode; auto get_serial_number() -> std::array; auto get_door_closed() -> bool; + [[nodiscard]] auto last_reset_reason() const -> uint16_t; }; diff --git a/stm32-modules/include/flex-stacker/flex-stacker/gcodes.hpp b/stm32-modules/include/flex-stacker/flex-stacker/gcodes.hpp index 21251861e..45561752e 100644 --- a/stm32-modules/include/flex-stacker/flex-stacker/gcodes.hpp +++ b/stm32-modules/include/flex-stacker/flex-stacker/gcodes.hpp @@ -142,6 +142,44 @@ struct GetSystemInfo { } }; +struct GetResetReason { + /* + * M114- GetResetReason retrieves the value of the RCC reset flag + * that was captured at the beginning of the hardware setup + * */ + using ParseResult = std::optional; + static constexpr auto prefix = std::array{'M', '1', '1', '4'}; + + template + requires std::forward_iterator && + std::sized_sentinel_for + static auto write_response_into(InputIt buf, InLimit limit, uint16_t reason) + -> InputIt { + int res = 0; + // print a hexadecimal representation of the reset flags + res = snprintf(&*buf, (limit - buf), "M114 Last Reset Reason: %X OK\n", + reason); + if (res <= 0) { + return buf; + } + return buf + res; + } + template + requires std::forward_iterator && + std::sized_sentinel_for + static auto parse(const InputIt& input, Limit limit) + -> std::pair { + auto working = prefix_matches(input, limit, prefix); + if (working == input) { + return std::make_pair(ParseResult(), input); + } + if (working != limit && !std::isspace(*working)) { + return std::make_pair(ParseResult(), input); + } + return std::make_pair(ParseResult(GetResetReason()), working); + } +}; + struct SetSerialNumber { using ParseResult = std::optional; static constexpr auto prefix = std::array{'M', '9', '9', '6'}; diff --git a/stm32-modules/include/flex-stacker/flex-stacker/host_comms_task.hpp b/stm32-modules/include/flex-stacker/flex-stacker/host_comms_task.hpp index 9f607ebd5..25638ef17 100644 --- a/stm32-modules/include/flex-stacker/flex-stacker/host_comms_task.hpp +++ b/stm32-modules/include/flex-stacker/flex-stacker/host_comms_task.hpp @@ -48,7 +48,7 @@ class HostCommsTask { gcode::GetLimitSwitches, gcode::SetMicrosteps, gcode::GetMoveParams, gcode::SetMotorStallGuard, gcode::GetMotorStallGuard, gcode::HomeMotor, gcode::GetPlatformSensors, gcode::GetDoorClosed, gcode::GetEstopStatus, - gcode::StopMotor>; + gcode::StopMotor, gcode::GetResetReason>; using AckOnlyCache = AckCache<8, gcode::EnterBootloader, gcode::SetSerialNumber, gcode::SetTMCRegister, gcode::SetRunCurrent, @@ -64,6 +64,7 @@ class HostCommsTask { using GetDoorClosedCache = AckCache<8, gcode::GetDoorClosed>; using GetPlatformSensorsCache = AckCache<8, gcode::GetPlatformSensors>; using GetEstopCache = AckCache<8, gcode::GetEstopStatus>; + using GetResetReasonCache = AckCache<8, gcode::GetResetReason>; public: static constexpr size_t TICKS_TO_WAIT_ON_SEND = 10; @@ -88,7 +89,9 @@ class HostCommsTask { // NOLINTNEXTLINE(readability-redundant-member-init) get_platform_sensors_cache(), // NOLINTNEXTLINE(readability-redundant-member-init) - get_estop_cache() {} + get_estop_cache(), + // NOLINTNEXTLINE(readability-redundant-member-init) + get_reset_reason_cache() {} HostCommsTask(const HostCommsTask& other) = delete; auto operator=(const HostCommsTask& other) -> HostCommsTask& = delete; HostCommsTask(HostCommsTask&& other) noexcept = delete; @@ -293,6 +296,50 @@ class HostCommsTask { cache_entry); } + template + requires std::forward_iterator && + std::sized_sentinel_for + auto visit_message(const messages::GetResetReasonResponse& response, + InputIt tx_into, InputLimit tx_limit) -> InputIt { + auto cache_entry = + get_reset_reason_cache.remove_if_present(response.responding_to_id); + return std::visit( + [tx_into, tx_limit, response](auto cache_element) { + using T = std::decay_t; + if constexpr (std::is_same_v) { + return errors::write_into( + tx_into, tx_limit, + errors::ErrorCode::BAD_MESSAGE_ACKNOWLEDGEMENT); + } else { + return cache_element.write_response_into(tx_into, tx_limit, + response.reason); + } + }, + cache_entry); + } + + template + requires std::forward_iterator && + std::sized_sentinel_for + auto visit_gcode(const gcode::GetResetReason& gcode, InputIt tx_into, + InputLimit tx_limit) -> std::pair { + auto id = get_reset_reason_cache.add(gcode); + if (id == 0) { + return std::make_pair( + false, errors::write_into(tx_into, tx_limit, + errors::ErrorCode::GCODE_CACHE_FULL)); + } + auto message = messages::GetResetReasonMessage{.id = id}; + + if (!task_registry->send(message, TICKS_TO_WAIT_ON_SEND)) { + auto wrote_to = errors::write_into( + tx_into, tx_limit, errors::ErrorCode::INTERNAL_QUEUE_FULL); + get_reset_reason_cache.remove_if_present(id); + return std::make_pair(false, wrote_to); + } + return std::make_pair(true, tx_into); + } + template requires std::forward_iterator && std::sized_sentinel_for @@ -1020,6 +1067,7 @@ class HostCommsTask { GetDoorClosedCache get_door_closed_cache; GetPlatformSensorsCache get_platform_sensors_cache; GetEstopCache get_estop_cache; + GetResetReasonCache get_reset_reason_cache; bool may_connect_latch = true; }; diff --git a/stm32-modules/include/flex-stacker/flex-stacker/messages.hpp b/stm32-modules/include/flex-stacker/flex-stacker/messages.hpp index d9e6df612..c721f04c8 100644 --- a/stm32-modules/include/flex-stacker/flex-stacker/messages.hpp +++ b/stm32-modules/include/flex-stacker/flex-stacker/messages.hpp @@ -74,6 +74,15 @@ struct GetSystemInfoResponse { const char* hw_version; }; +struct GetResetReasonMessage { + uint32_t id; +}; + +struct GetResetReasonResponse { + uint32_t responding_to_id; + uint16_t reason; +}; + struct SetSerialNumberMessage { uint32_t id; static constexpr std::size_t SERIAL_NUMBER_LENGTH = @@ -278,12 +287,12 @@ using HostCommsMessage = GetTMCRegisterResponse, GetLimitSwitchesResponses, GetMoveParamsResponse, GetMotorStallGuardResponse, GetDoorClosedResponse, GetPlatformSensorsResponse, - GetEstopResponse>; + GetEstopResponse, GetResetReasonResponse>; using SystemMessage = ::std::variant; + GetDoorClosedMessage, GetResetReasonMessage>; using MotorDriverMessage = ::std::variant(_task_registry->send(response)); } + template + auto visit_message(const messages::GetResetReasonMessage& msg, + Policy& policy) -> void { + auto reason = policy.last_reset_reason(); + + auto response = messages::GetResetReasonResponse{ + .responding_to_id = msg.id, .reason = reason}; + static_cast( + _task_registry->send(response, Queues::HostCommsAddress)); + } + template auto visit_message(const messages::SetSerialNumberMessage& message, Policy& policy) { From 791584668d8d444fe8133a6880e04736f4d6ac51 Mon Sep 17 00:00:00 2001 From: Alise Au <20424172+ahiuchingau@users.noreply.github.com> Date: Thu, 12 Dec 2024 16:03:31 -0500 Subject: [PATCH 5/6] update tpowerdown and hold current delay (#493) --- .../flex-stacker/flex-stacker/motor_driver_task.hpp | 9 ++++++--- .../flex-stacker/flex-stacker/tmc2160_registers.hpp | 4 ++-- 2 files changed, 8 insertions(+), 5 deletions(-) diff --git a/stm32-modules/include/flex-stacker/flex-stacker/motor_driver_task.hpp b/stm32-modules/include/flex-stacker/flex-stacker/motor_driver_task.hpp index dc156a056..82ca48d81 100644 --- a/stm32-modules/include/flex-stacker/flex-stacker/motor_driver_task.hpp +++ b/stm32-modules/include/flex-stacker/flex-stacker/motor_driver_task.hpp @@ -35,7 +35,8 @@ static constexpr tmc2160::TMC2160RegisterMap motor_z_config{ .glob_scale = {.global_scaler = 0x0}, .ihold_irun = {.hold_current = 10, .run_current = 31, - .hold_current_delay = 15}, + .hold_current_delay = 1}, + .tpowerdown = {.time = tmc2160::PowerDownDelay::seconds_to_reg(0.1)}, .tpwmthrs = {.threshold = 0x80000}, .tcoolthrs = {.threshold = 0x2FF}, .thigh = {.threshold = 0x1}, @@ -63,7 +64,8 @@ static constexpr tmc2160::TMC2160RegisterMap motor_x_config{ .glob_scale = {.global_scaler = 0x0}, .ihold_irun = {.hold_current = 12, .run_current = 31, - .hold_current_delay = 7}, + .hold_current_delay = 1}, + .tpowerdown = {.time = tmc2160::PowerDownDelay::seconds_to_reg(0.1)}, .tpwmthrs = {.threshold = 0x80000}, .tcoolthrs = {.threshold = 0x2FF}, .thigh = {.threshold = 0x1}, @@ -91,7 +93,8 @@ static constexpr tmc2160::TMC2160RegisterMap motor_l_config{ .glob_scale = {.global_scaler = 0x0}, .ihold_irun = {.hold_current = 7, .run_current = 8, - .hold_current_delay = 7}, + .hold_current_delay = 1}, + .tpowerdown = {.time = tmc2160::PowerDownDelay::seconds_to_reg(0.1)}, .tpwmthrs = {.threshold = 0x80000}, .tcoolthrs = {.threshold = 0x2FF}, .thigh = {.threshold = 0x1}, diff --git a/stm32-modules/include/flex-stacker/flex-stacker/tmc2160_registers.hpp b/stm32-modules/include/flex-stacker/flex-stacker/tmc2160_registers.hpp index f7bbf885c..a78b85d98 100644 --- a/stm32-modules/include/flex-stacker/flex-stacker/tmc2160_registers.hpp +++ b/stm32-modules/include/flex-stacker/flex-stacker/tmc2160_registers.hpp @@ -261,11 +261,11 @@ struct __attribute__((packed, __may_alias__)) PowerDownDelay { static constexpr uint32_t max_val = 0xFF; static constexpr uint32_t reset = 10; - [[nodiscard]] static auto reg_to_seconds(uint8_t reg) -> double { + [[nodiscard]] static constexpr auto reg_to_seconds(uint8_t reg) -> double { return (static_cast(reg) / static_cast(max_val)) * max_time; } - static auto seconds_to_reg(double seconds) -> uint8_t { + static constexpr auto seconds_to_reg(double seconds) -> uint8_t { if (seconds > max_time) { return max_val; } From b204d5e73b6f710d026f7ecc85325e0fe4bc2205 Mon Sep 17 00:00:00 2001 From: Brayan Almonte Date: Thu, 12 Dec 2024 16:52:59 -0500 Subject: [PATCH 6/6] fix(flex-stacker): fix invalid Latch motor gear ratio for EVT flex-stackers. (#494) --- stm32-modules/include/flex-stacker/flex-stacker/motor_task.hpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/stm32-modules/include/flex-stacker/flex-stacker/motor_task.hpp b/stm32-modules/include/flex-stacker/flex-stacker/motor_task.hpp index d8921326d..8cd16392b 100644 --- a/stm32-modules/include/flex-stacker/flex-stacker/motor_task.hpp +++ b/stm32-modules/include/flex-stacker/flex-stacker/motor_task.hpp @@ -69,7 +69,7 @@ struct Defaults { static constexpr float SPEED_DISCONT = 5.0; static constexpr float MM_PER_REV = - lms::GearBoxConfig::mm_per_rev(16.0, 16.0 / 30.0); + lms::GearBoxConfig::mm_per_rev(30, 30.0 / 16.0); static constexpr float STEPS_PER_REV = 200; static constexpr float MICROSTEP = 16; };