- CONFIG_COMPILER_STACK_CHECK_MODE_NORM=y (buffer overflow detection) - CONFIG_HEAP_POISONING_LIGHT=y (use-after-free/corruption detection) - CONFIG_ESP_TASK_WDT_PANIC=y (auto-reboot on hung task) - Remove unused #include "esp_now.h" (CVE-2025-52471 mitigation) - Replace hardcoded default IP 192.168.129.11 with 0.0.0.0 in Kconfig
13 KiB
13 KiB
ESP32 Hacking Tasks
Last Updated: 2026-02-14
Completed: v1.12-dev — Security Hardening & Pentest
Security hardening deployed to amber-maple. Full pentest executed 2026-02-14.
- Auth whitelist: only read-only queries work without HMAC auth
- AUTH OFF disabled remotely (serial console or FACTORY reset only)
- HMAC 128-bit (32 hex chars), replay window +/-5s, nonce dedup cache (8 entries)
- STATUS split: minimal (unauthed) vs full (authed) response
- Rate limiter: 50ms inter-command throttle (20 cmd/s max)
- NVS write throttle: 20 writes per 10s window
- CSI buffer bounds checking (UDP_REM macro)
- PMF (802.11w) required:
CONFIG_ESP_WIFI_PMF_REQUIRED=y - mDNS: hostname only, no service advertisement
- Serial console AUTH management (UART0, 921600 baud)
- ALERT command (temp/heap thresholds, EVENT emission)
- Secret auto-generated on first boot, redacted in boot log
- Pentest: mDNS service discovery — PASS (no service ads)
- Pentest: Port scan — PASS (only 5353/udp + 5501/udp open, 0 TCP)
- Pentest: Firmware binary analysis — PASS (no hardcoded secrets)
- Pentest: eFuse readout — all security fuses unburned (expected for dev)
- Pentest: HMAC timing oracle — PASS (constant-time comparison effective)
- Pentest: Command injection (27 tests) — PASS (all handled safely)
- Pentest: Replay attack (6 tests) — PASS (all rejected)
- Pentest: NVS partition analysis — auth_secret in plaintext (expected without flash encryption)
- Pentest: ESP-IDF CVE check (12 CVEs) — 8 N/A, 4 LOW risk
- Pentest: Binary structure — no stack canaries, no heap poisoning (fix recommended)
- Pentest results documented in
docs/PENTEST-RESULTS.md
Completed: v1.11.0 — Diagnostics & Usability
Deployed to fleet 2026-02-14.
- CSI ON/OFF command to toggle CSI collection
- HELP command (lists all 30 commands with syntax)
- FACTORY command (erase NVS config + reboot)
- CONFIG command (dump all running config key=value)
- PING command (echo reply, returns OK PONG)
- LOG command (runtime log level: NONE/ERROR/WARN/INFO/DEBUG/VERBOSE)
- RSSI RESET command (reset min/max counters)
- Tagged v1.11.0 and OTA deployed to all 3 sensors
Web Backend (~/git/esp32-web/)
Tracked separately in ~/git/esp32-web/TASKS.md. Currently at v0.1.5.
Firmware Backlog
P1 - High
- Test OTA rollback — crasher firmware flashed to amber-maple, bootloader rolled back to v1.11.0 (2026-02-14)
- Enable stack canaries:
CONFIG_COMPILER_STACK_CHECK_MODE_NORM=y(2026-02-14) - Enable heap poisoning:
CONFIG_HEAP_POISONING_LIGHT=y(2026-02-14) - Enable WDT panic:
CONFIG_ESP_TASK_WDT_PANIC=y(2026-02-14) - Remove unused
#include "esp_now.h"(2026-02-14) - Remove hardcoded default IP from Kconfig (2026-02-14)
P2 - Normal
- Tune presence threshold per room with real-world testing
- Power consumption measurements using POWERTEST + external meter
- Multi-target (send UDP to 2+ destinations)
P3 - Low
- ALERT command — temp/heap threshold monitoring with EVENT emission (2026-02-14)
- Deep sleep mode with wake-on-CSI-motion
- Battery-optimized duty cycling
- AP+STA config portal (captive portal for initial setup)
Documentation
- Document esp-crab dual-antenna capabilities
- Document esp-radar console features
- Pin mapping for ESP32-DevKitC V1
Completed: v1.10 - LED Quiet Mode & CI Hardening
- LED quiet mode (off normally, solid on motion/presence, blinks on OTA)
- Default LED to quiet mode
- Build metadata in STATUS reply
- CI security checks (secrets scan, config validation)
- Firmware size check and version tag validation
- Size optimization (
-Os, saves ~75KB) - CSI ON/OFF toggle command
Completed: v1.9 - Multi-Channel Scanning & BLE Fingerprinting
- CHANSCAN command (ON/OFF/NOW/INTERVAL)
- Channel hopping (1-13) with 100ms dwell, pause CSI during scan
- chanscan= field in STATUS, chanscan_int NVS persistence
- BLE fingerprinting: company_id, tx_power, adv_flags in BLE_DATA
- Historical presence sessions support
Completed: v1.8 - HTTPS OTA Support
- Support HTTPS URLs for OTA updates (esp_https_ota)
- CI: Upload firmware to Gitea releases for OTA
- CI: Simplify deploy script to pure POSIX sh
Completed: v1.7 - Baseline Calibration & Presence Detection
- CALIBRATE command (capture N seconds of CSI, average per-subcarrier amplitudes)
- CALIBRATE STATUS / CALIBRATE CLEAR subcommands
- Presence scoring (normalized Euclidean distance vs baseline, rolling window of 50)
- PRESENCE ON/OFF command (NVS persisted, requires valid baseline)
- PRESENCE THRESHOLD command (0.001-1.0, NVS persisted, default 0.05)
- Presence events (
EVENT,<hostname>,presence=<0|1> score=<float>) with 10s holdoff - Calibration done event (
EVENT,<hostname>,calibrate=done packets=<n> nsub=<n>) - presence= and pr_score= fields in STATUS reply
- NVS persistence: bl_amps (blob), bl_nsub (i8), presence (i8), pr_thresh (i32)
- config_save_blob / config_erase_key NVS helpers
- n_sub field in csi_features_t, amps_out parameter in csi_extract_features
- Pi-side: parse presence/calibrate events in watch daemon
- Pi-side: store in sensor_events table (
esp-ctl osint events) - Pi-side: HA webhooks for presence_change and calibrate_done
- Fleet OTA to v1.7, calibrated hollow-acorn + amber-maple
Completed: v1.6 - Power Management
- ESP-IDF power management framework (DFS 240/80 MHz + light sleep)
- sdkconfig: CONFIG_PM_ENABLE, CONFIG_FREERTOS_USE_TICKLESS_IDLE
- POWERSAVE command (WiFi modem sleep toggle, NVS persisted, default off)
- powersave= field in STATUS reply
- POWERTEST save/restore of powersave state
- Deployed to fleet, NVS persistence verified
Completed: v1.5 - Event Handling & NVS Persistence
- EVENT packet parsing in watch daemon (motion, wifi_reconnect, powertest)
- Sensor heartbeat tracking (offline/online detection, configurable timeout)
- sensor_events table with indexed queries (
esp-ctl osint events) - HA webhooks for sensor_offline, sensor_online, motion_change
- NVS persistence for SCANRATE and PROBERATE commands
- Sensor events count in
esp-ctl osint stats
Completed: v1.4 - Multi-Sensor & Validation
- Multi-sensor BLE correlation in esp-ctl (zone tracking by source sensor)
- Zone tracking with EMA RSSI (
esp-ctl osint zones,device_zonestable) - Per-sensor breakdown in MAC profile (
esp-ctl osint mac) - POWERTEST command (7-phase power profiling with EVENT markers)
- Parallel OTA fleet updates (
esp-fleet ota --parallel)
Completed: v1.3 - Security & OSINT
- HMAC command authentication (firmware + esp-ctl/esp-cmd/esp-fleet/esp-ota)
- AUTH command (set/query/disable secret, NVS persisted)
- auth=on/off in STATUS
- Deauth flood detection (ring buffer, aggregate ALERT_DATA)
- FLOODTHRESH command (count + window, NVS persisted)
- flood_thresh field in STATUS
- MAC OUI vendor lookup (
esp-ctl oui, IEEE CSV database) - OSINT SQLite database (probe_ssids, device_sightings tables)
- Watch daemon (
esp-ctl watch— listen + enrich + store) - OSINT query CLI (
esp-ctl osint probes/devices/mac/stats) - Home Assistant webhook integration (deauth_flood, unknown_probe, unknown_ble)
- Watch config file (
~/.config/esp-ctl/watch.yaml)
Completed: v1.2
- On-device CSI feature extraction (amp_rms, amp_std, amp_max, amp_max_idx, energy)
- CSIMODE command: RAW, COMPACT, HYBRID N (NVS persisted)
- Compact payload format
"F:rms,std,max,idx,energy"(~80% bandwidth reduction) - Hybrid mode: compact every packet, raw every Nth
- STATUS fields: csi_mode, hybrid_n
- Adaptive sampling reuses extracted energy (no duplicate computation in COMPACT/HYBRID)
Completed: v1.1
- Sensor ID in data packets (hostname prefix on CSI_DATA, BLE_DATA, EVENT)
- Deauth/disassoc frame detection (ALERT_DATA via promiscuous mode)
- Chip temperature reporting in STATUS reply
- BLE alerting (
esp-ctl ble --known, alert on unknown MACs) - BLE dwell time tracking (dwell column in
--tracksummary) - Timestamped event logging (
esp-ctl listen --timestamp) - Alert filter in esp-ctl (
listen -f alert) - Runtime HOSTNAME command (NVS persisted, mDNS updated)
- WiFi probe request capture (PROBE_DATA via promiscuous mode, 10s dedup)
- mDNS service advertisement (
_esp-csi._udp) - mDNS sensor discovery (
esp-ctl discover) - Probe filter in esp-ctl (
listen -f probe) - OTA fleet to same firmware (
ca526ef) - CSI packet counter in STATUS (
csi_count=) - Raw uptime in STATUS (
uptime_s=) - WiFi reconnect EVENT emission
- SCANRATE command (BLE scan interval tuning, 5-300s)
- PROBERATE command (probe dedup cooldown tuning, 1-300s)
- Fix: promiscuous mode disables CSI on original ESP32 — guarded with
#if - FreeRTOS CPU runtime stats in PROFILE (trace facility enabled)
- Increased cmd_task stack 4KB → 6KB (was 516 bytes free)
- WiFi channel in STATUS (
channel=) - Boot counter in NVS (
boots=) - RSSI min/max tracking from CSI frames (
rssi_min=,rssi_max=) - Actual CSI rate in STATUS (
csi_rate=)
Completed: v0.5 - BLE Scanning
- Enable Bluetooth alongside WiFi (NimBLE, BLE ON/OFF command)
- Periodic BLE advertisement scanning
- Report device MAC, RSSI, name via UDP
- Pi-side BLE device tracking (
esp-ctl ble --track) - PROFILE command (heap, stack watermarks, CPU runtime stats)
- TARGET command (runtime UDP destination config)
Completed: v0.4 - Adaptive Sampling
- On-device CSI wander calculation (coefficient of variation)
- Adaptive rate: 10 pkt/s idle (3s holdoff) → 100 pkt/s on motion
- EVENT notification to Pi on rate change
- ADAPTIVE ON/OFF command (NVS persisted)
- THRESHOLD command for tuning sensitivity (NVS persisted)
- RATE command disables adaptive mode
- adaptive/motion fields in STATUS reply
- OTA deployed and verified on amber-maple
Completed: v0.3 - OTA Updates
- Dual OTA partition table (
partitions.csv) - 4MB flash, custom partitions, rollback in sdkconfig.defaults
- Firmware: OTA command, ota_task, LED_OTA, rollback validation
- Firmware: version in STATUS reply
- Pi-side
esp-otatool (HTTP server + OTA orchestration) esp-fleet otasubcommand (sequential fleet update)- Build and USB-flash amber-maple (partition table change)
- End-to-end OTA test verified
- Regenerate sdkconfig.sample
- Update CHEATSHEET.md, USAGE.md
Completed: v0.2 - Remote Management
- Firmware: UDP command listener (port 5501)
- Firmware: LED status indicator (GPIO2)
- Firmware: NVS config persistence (rate, tx_power)
- Firmware: REBOOT, IDENTIFY, STATUS commands
- Firmware: RATE command (10-100 Hz, restarts ping)
- Firmware: POWER command (2-20 dBm)
- Pi-side:
esp-cmdCLI tool - Pi-side:
esp-fleetfleet management tool - mDNS hostname, watchdog, human-readable uptime
Completed: v0.1 - Documentation
- Copy firmware sources to project
- Document current firmware and settings
- Document build & flash workflow
- Create .gitignore for build artifacts
- Test building firmware from this repo
- Document CSI config options
Notes
- Adaptive threshold varies by environment; 0.001-0.01 is a good starting range
- NVS keys (28 total):
send_rate,tx_power,adaptive,threshold,ble_scan,target_ip,target_port,hostname,boot_count,csi_mode,hybrid_n,auth_secret,flood_thresh,flood_window,scan_rate,probe_rate,powersave,presence,pr_thresh,bl_nsub,bl_amps,chanscan,chanscan_int,led_quiet,csi_enabled,alert_temp,alert_heap - UDP commands (28 total): STATUS, CONFIG, RATE, POWER, TARGET, HOSTNAME, CSI, CSIMODE, ADAPTIVE, THRESHOLD, BLE, SCANRATE, PROBERATE, CALIBRATE, PRESENCE, CHANSCAN, LED, POWERSAVE, AUTH, FLOODTHRESH, ALERT, OTA, POWERTEST, PROFILE, PING, LOG, RSSI RESET, IDENTIFY, REBOOT, FACTORY, HELP
- EVENT packets include sensor hostname:
EVENT,<hostname>,motion=... rate=... wander=... - Alert events:
EVENT,<hostname>,alert=heap free=<n> thresh=<n>andEVENT,<hostname>,alert=temp value=<n> thresh=<n> - ALERT_DATA format:
ALERT_DATA,<hostname>,<deauth|disassoc>,<sender_mac>,<target_mac>,<rssi>orALERT_DATA,<hostname>,deauth_flood,<count>,<window_s> - STATUS fields:
uptime=,uptime_s=,heap=,rssi=,channel=,tx_power=,rate=,csi_rate=,hostname=,version=,adaptive=,motion=,ble=,target=,temp=,csi_count=,boots=,rssi_min=,rssi_max=,csi=,csi_mode=,hybrid_n=,auth=,flood_thresh=,powersave=,presence=,pr_score=,chanscan=,led=,alert_temp=,alert_heap=,nvs_used=,nvs_free=,nvs_total=,part_size=,built=,idf=,chip= - PROBE_DATA format:
PROBE_DATA,<hostname>,<mac>,<rssi>,<ssid> - Probe requests deduped per MAC (default 10s cooldown, tunable via PROBERATE)
- mDNS service:
_esp-csi._udpon data port (for sensor discovery) - HOSTNAME command:
HOSTNAME <name>sets NVS + mDNS,HOSTNAMEqueries current - ESP32 limitation: Promiscuous mode (deauth/probe detection) disabled — breaks CSI. Works on ESP32-C6+.