saltylab-firmware/src/watchdog.c
sl-controls f4e71777ec fix: Resolve all compile and linker errors (Issue #337)
Fixed 7 compile errors across 6 files:

1. servo.c: Removed duplicate ServoState typedef, updated struct definition in header
2. watchdog.c: Fixed IWDG handle usage - moved to global scope for IRQHandler access
3. ultrasonic.c: Fixed timer handle type mismatches - use TIM_HandleTypeDef instead of TIM_TypeDef, replaced HAL_TIM_IC_Init_Compat with proper HAL functions
4. main.c: Replaced undefined functions - imu_calibrated() → mpu6000_is_calibrated(), crsf_is_active() → manual state check
5. ina219.c: Stubbed I2C functions pending HAL implementation

Build now passes with ZERO errors.
- RAM: 6.5% (16964 bytes / 262144)
- Flash: 10.6% (55368 bytes / 524288)

Co-Authored-By: Claude Haiku 4.5 <noreply@anthropic.com>
2026-03-03 19:00:12 -05:00

149 lines
4.5 KiB
C

#include "watchdog.h"
#include "stm32f7xx_hal.h"
#include <string.h>
/* ================================================================
* IWDG Hardware Configuration
* ================================================================ */
/* LSI frequency: approximately 32 kHz (typical, 20-40 kHz) */
#define LSI_FREQUENCY_HZ 32000
/* IWDG prescaler values */
#define IWDG_PSC_4 0 /* Divider = 4 */
#define IWDG_PSC_8 1 /* Divider = 8 */
#define IWDG_PSC_16 2 /* Divider = 16 */
#define IWDG_PSC_32 3 /* Divider = 32 */
#define IWDG_PSC_64 4 /* Divider = 64 */
#define IWDG_PSC_128 5 /* Divider = 128 */
#define IWDG_PSC_256 6 /* Divider = 256 */
/* Reload register range: 0-4095 */
#define IWDG_RELOAD_MIN 1
#define IWDG_RELOAD_MAX 4095
/* ================================================================
* Watchdog State
* ================================================================ */
typedef struct {
bool is_initialized; /* Whether watchdog has been initialized */
bool is_running; /* Whether watchdog is currently active */
uint32_t timeout_ms; /* Configured timeout in milliseconds */
uint8_t prescaler; /* IWDG prescaler value */
uint16_t reload_value; /* IWDG reload register value */
} WatchdogState;
static WatchdogState s_watchdog = {
.is_initialized = false,
.is_running = false,
.timeout_ms = 0,
.prescaler = IWDG_PSC_32,
.reload_value = 0
};
static IWDG_HandleTypeDef s_iwdg_handle = {0};
/* ================================================================
* Helper Functions
* ================================================================ */
/* Calculate prescaler and reload values for desired timeout */
static bool watchdog_calculate_config(uint32_t timeout_ms,
uint8_t *out_prescaler,
uint16_t *out_reload)
{
if (timeout_ms < 1 || timeout_ms > 32000) {
return false; /* Out of reasonable range */
}
/* Try prescalers from smallest to largest */
const uint8_t prescalers[] = {IWDG_PSC_4, IWDG_PSC_8, IWDG_PSC_16,
IWDG_PSC_32, IWDG_PSC_64, IWDG_PSC_128,
IWDG_PSC_256};
const uint16_t dividers[] = {4, 8, 16, 32, 64, 128, 256};
for (int i = 0; i < 7; i++) {
uint16_t divider = dividers[i];
/* timeout_ms = (reload * divider * 1000) / LSI_FREQUENCY_HZ */
uint32_t reload = (timeout_ms * LSI_FREQUENCY_HZ) / (divider * 1000);
if (reload >= IWDG_RELOAD_MIN && reload <= IWDG_RELOAD_MAX) {
*out_prescaler = prescalers[i];
*out_reload = (uint16_t)reload;
return true;
}
}
return false; /* No suitable prescaler found */
}
/* ================================================================
* Public API
* ================================================================ */
bool watchdog_init(uint32_t timeout_ms)
{
if (s_watchdog.is_initialized) {
return false; /* Already initialized */
}
/* Validate and calculate prescaler/reload values */
uint8_t prescaler;
uint16_t reload;
if (!watchdog_calculate_config(timeout_ms, &prescaler, &reload)) {
return false;
}
s_watchdog.prescaler = prescaler;
s_watchdog.reload_value = reload;
s_watchdog.timeout_ms = timeout_ms;
/* Configure and start IWDG */
s_iwdg_handle.Instance = IWDG;
s_iwdg_handle.Init.Prescaler = prescaler;
s_iwdg_handle.Init.Reload = reload;
s_iwdg_handle.Init.Window = reload; /* Window == Reload means full timeout */
HAL_IWDG_Init(&s_iwdg_handle);
s_watchdog.is_initialized = true;
s_watchdog.is_running = true;
return true;
}
void watchdog_kick(void)
{
if (s_watchdog.is_running) {
HAL_IWDG_Refresh(&s_iwdg_handle); /* Reset IWDG counter */
}
}
uint32_t watchdog_get_timeout(void)
{
return s_watchdog.timeout_ms;
}
bool watchdog_is_running(void)
{
return s_watchdog.is_running;
}
bool watchdog_was_reset_by_watchdog(void)
{
/* Check RCC reset source flags */
/* IWDG reset sets the IWDGRSTF flag in RCC_CSR */
uint32_t reset_flags = RCC->CSR;
/* IWDGRSTF is bit 29 of RCC_CSR */
bool was_iwdg_reset = (reset_flags & RCC_CSR_IWDGRSTF) != 0;
/* Clear the flag by writing to RMVF (Bit 24) */
if (was_iwdg_reset) {
RCC->CSR |= RCC_CSR_RMVF; /* Clear reset flags */
}
return was_iwdg_reset;
}