STM32 firmware: - safety.h/c: EstopSource enum, safety_remote_estop/clear/get/active() CDC 'E'=ESTOP_REMOTE, 'F'=ESTOP_CELLULAR_TIMEOUT, 'Z'=clear latch - usbd_cdc_if: cdc_estop_request/cdc_estop_clear_request volatile flags - status: status_update() +remote_estop param; both LEDs fast-blink 200ms - main.c: immediate motor cutoff highest-priority; arming gated by !safety_remote_estop_active(); motor estop auto-clear gated; telemetry 'es' field 0-4; status_update() updated to 5 args Safety: IMMEDIATE motor cutoff, latched until explicit Z + DISARMED, cannot re-arm via MQTT alone (requires RC arm hold). IWDG-safe. Jetson bridge: - remote_estop_node.py: paho-mqtt + pyserial, cellular watchdog 5s - estop_params.yaml, remote_estop.launch.py - setup.py / package.xml: register node + paho-mqtt dep - docker-compose.yml: remote-estop service - test_remote_estop.py: kill/clear/watchdog/latency unit tests Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
86 lines
2.7 KiB
C
86 lines
2.7 KiB
C
/*
|
|
* safety.c — SaltyLab Safety Systems
|
|
*
|
|
* IWDG: 40kHz LSI, prescaler 32 → tick = 0.8ms.
|
|
* WATCHDOG_TIMEOUT_MS from config.h (default 50ms → reload = 62).
|
|
* Formula: reload = (timeout_ms / (prescaler / 40000)) - 1
|
|
* reload = (50 * 40000 / 32) - 1 = 62499 → clamp to 4095 (12-bit max)
|
|
* At PSC=256: tick = 6.4ms, reload = ceil(50/6.4)-1 = 7 → ~57.6ms
|
|
* Using PSC=32: tick = 0.8ms, reload = ceil(50/0.8)-1 = 62 → 50ms ✓
|
|
*/
|
|
|
|
#include "safety.h"
|
|
#include "config.h"
|
|
#include "crsf.h"
|
|
#include "stm32f7xx_hal.h"
|
|
|
|
/* IWDG prescaler 32 → LSI(40kHz)/32 = 1250 ticks/sec → 0.8ms/tick */
|
|
#define IWDG_PRESCALER IWDG_PRESCALER_32
|
|
/* Integer formula: timeout_ms * LSI_HZ / (prescaler * 1000)
|
|
* = WATCHDOG_TIMEOUT_MS * 40000 / (32 * 1000) = WATCHDOG_TIMEOUT_MS * 40 / 32 */
|
|
#define IWDG_RELOAD (WATCHDOG_TIMEOUT_MS * 40UL / 32UL)
|
|
|
|
#if IWDG_RELOAD > 4095
|
|
# error "WATCHDOG_TIMEOUT_MS too large for IWDG_PRESCALER_32 — increase prescaler"
|
|
#endif
|
|
|
|
static IWDG_HandleTypeDef hiwdg;
|
|
|
|
/* Arm interlock */
|
|
static uint32_t s_arm_start_ms = 0;
|
|
static bool s_arm_pending = false;
|
|
|
|
/* Tilt fault alert state — edge-detect to fire buzzer once */
|
|
static bool s_was_faulted = false;
|
|
|
|
static EstopSource s_estop_source = ESTOP_CLEAR;
|
|
|
|
void safety_init(void) {
|
|
hiwdg.Instance = IWDG;
|
|
hiwdg.Init.Prescaler = IWDG_PRESCALER;
|
|
hiwdg.Init.Reload = IWDG_RELOAD;
|
|
hiwdg.Init.Window = IWDG_WINDOW_DISABLE;
|
|
HAL_IWDG_Init(&hiwdg); /* Starts watchdog immediately */
|
|
}
|
|
|
|
void safety_refresh(void) {
|
|
if (hiwdg.Instance) HAL_IWDG_Refresh(&hiwdg);
|
|
}
|
|
|
|
bool safety_rc_alive(uint32_t now) {
|
|
/* If crsf_state has never received a frame, last_rx_ms == 0 */
|
|
if (crsf_state.last_rx_ms == 0) return false;
|
|
return (now - crsf_state.last_rx_ms) < RC_TIMEOUT_MS;
|
|
}
|
|
|
|
void safety_alert_tilt_fault(bool faulted) {
|
|
if (faulted && !s_was_faulted) {
|
|
/* Rising edge: single buzzer burst to alert rider */
|
|
HAL_GPIO_WritePin(BEEPER_PORT, BEEPER_PIN, GPIO_PIN_SET);
|
|
HAL_Delay(200);
|
|
HAL_GPIO_WritePin(BEEPER_PORT, BEEPER_PIN, GPIO_PIN_RESET);
|
|
}
|
|
s_was_faulted = faulted;
|
|
}
|
|
|
|
void safety_arm_start(uint32_t now) {
|
|
if (!s_arm_pending) {
|
|
s_arm_start_ms = now;
|
|
s_arm_pending = true;
|
|
}
|
|
}
|
|
|
|
bool safety_arm_ready(uint32_t now) {
|
|
if (!s_arm_pending) return false;
|
|
return (now - s_arm_start_ms) >= ARMING_HOLD_MS;
|
|
}
|
|
|
|
void safety_arm_cancel(void) {
|
|
s_arm_pending = false;
|
|
}
|
|
|
|
void safety_remote_estop(EstopSource src) { s_estop_source = src; }
|
|
void safety_remote_estop_clear(void) { s_estop_source = ESTOP_CLEAR; }
|
|
EstopSource safety_get_estop(void) { return s_estop_source; }
|
|
bool safety_remote_estop_active(void) { return s_estop_source >= ESTOP_REMOTE; }
|