Compare commits
15 Commits
cea8926604
...
clearpilot
| Author | SHA1 | Date | |
|---|---|---|---|
| 2331aa00a0 | |||
| dfb7b7404f | |||
| 9ac334b7cf | |||
| 3cbb81f9f1 | |||
| 3b47412100 | |||
| 5e7c8bed52 | |||
| 6cf21a3698 | |||
| adcffad276 | |||
| 6e7117b177 | |||
| 8ccdb47c88 | |||
| b84c268b6e | |||
| ffa9da2f97 | |||
| 5e7911e599 | |||
| 6fcd4b37ba | |||
| 8d5903b945 |
44
CLAUDE.md
44
CLAUDE.md
@@ -15,7 +15,7 @@ ClearPilot is a custom fork of **FrogPilot** (itself a fork of comma.ai's openpi
|
|||||||
- **Native dashcamd**: C++ process capturing raw camera frames via VisionIPC with OMX H.264 hardware encoding
|
- **Native dashcamd**: C++ process capturing raw camera frames via VisionIPC with OMX H.264 hardware encoding
|
||||||
- **Standstill power saving**: model inference throttled to 1fps and fan quieted when car is stopped
|
- **Standstill power saving**: model inference throttled to 1fps and fan quieted when car is stopped
|
||||||
- **ClearPilot menu**: sidebar settings panel replacing stock home screen (Home, Dashcam, Debug panels)
|
- **ClearPilot menu**: sidebar settings panel replacing stock home screen (Home, Dashcam, Debug panels)
|
||||||
- **Status window**: live system stats (temp, fan, storage, RAM, WiFi, VPN, GPS, telemetry status)
|
- **Status window**: live system stats (temp, fan, storage, RAM, WiFi, VPN, GPS, telemetry, dashcam status)
|
||||||
- **Debug button (LFA)**: steering wheel button repurposed for screen toggle and future UI actions
|
- **Debug button (LFA)**: steering wheel button repurposed for screen toggle and future UI actions
|
||||||
- **Telemetry system**: diff-based CSV logger via ZMQ IPC, toggleable from Debug panel
|
- **Telemetry system**: diff-based CSV logger via ZMQ IPC, toggleable from Debug panel
|
||||||
- **Bench mode**: `--bench` flag for onroad UI testing without car connected
|
- **Bench mode**: `--bench` flag for onroad UI testing without car connected
|
||||||
@@ -101,7 +101,7 @@ ClearPilot uses memory params (`/dev/shm/params/d/`) for UI toggles that should
|
|||||||
- **Python access**: Use `Params("/dev/shm/params")`
|
- **Python access**: Use `Params("/dev/shm/params")`
|
||||||
- **Defaults**: Set in `manager_init()` via `Params("/dev/shm/params").put(key, value)`
|
- **Defaults**: Set in `manager_init()` via `Params("/dev/shm/params").put(key, value)`
|
||||||
- **UI toggles**: Use `ToggleControl` with manual `toggleFlipped` lambda that writes via `Params("/dev/shm/params")`. Do NOT use `ParamControl` for memory params — it reads/writes persistent params only
|
- **UI toggles**: Use `ToggleControl` with manual `toggleFlipped` lambda that writes via `Params("/dev/shm/params")`. Do NOT use `ParamControl` for memory params — it reads/writes persistent params only
|
||||||
- **Current memory params**: `TelemetryEnabled` (default "0"), `VpnEnabled` (default "1"), `ModelStandby` (default "0"), `ScreenDisplayMode`
|
- **Current memory params**: `TelemetryEnabled` (default "0"), `VpnEnabled` (default "1"), `ModelStandby` (default "0"), `ScreenDisplayMode`, `DashcamState` (default "stopped"), `DashcamFrames` (default "0")
|
||||||
- **IMPORTANT — method names differ between C++ and Python**: C++ uses camelCase (`putBool`, `getBool`, `getInt`), Python uses snake_case (`put_bool`, `get_bool`, `get_int`). This is a common source of silent failures — the wrong casing compiles/runs but doesn't work.
|
- **IMPORTANT — method names differ between C++ and Python**: C++ uses camelCase (`putBool`, `getBool`, `getInt`), Python uses snake_case (`put_bool`, `get_bool`, `get_int`). This is a common source of silent failures — the wrong casing compiles/runs but doesn't work.
|
||||||
|
|
||||||
### Building Native (C++) Processes
|
### Building Native (C++) Processes
|
||||||
@@ -245,20 +245,21 @@ A single `session.log` in each session directory records major events:
|
|||||||
- **Segment length**: 3 minutes per file
|
- **Segment length**: 3 minutes per file
|
||||||
- **Save path**: `/data/media/0/videos/YYYYMMDD-HHMMSS/YYYYMMDD-HHMMSS.mp4` (trip directories)
|
- **Save path**: `/data/media/0/videos/YYYYMMDD-HHMMSS/YYYYMMDD-HHMMSS.mp4` (trip directories)
|
||||||
- **GPS subtitles**: companion `.srt` file per segment with 1Hz entries (speed MPH, lat/lon, UTC timestamp)
|
- **GPS subtitles**: companion `.srt` file per segment with 1Hz entries (speed MPH, lat/lon, UTC timestamp)
|
||||||
- **Trip lifecycle**: starts recording on launch with 10-min idle timer; car in drive cancels timer; park/off restarts timer; ignition cycle = new trip
|
- **Trip lifecycle**: waits in WAITING state until valid system time + GPS fix + car in drive; records until car parked 10 min or ignition off; then returns to WAITING
|
||||||
- **Graceful shutdown**: thermald sets `DashcamShutdown` param, dashcamd closes segment and acks within 15s
|
- **Graceful shutdown**: thermald sets `DashcamShutdown` param, dashcamd closes segment and acks within 15s
|
||||||
- **Storage**: ~56 MB per 3-minute segment at 2500 kbps
|
- **Storage**: ~56 MB per 3-minute segment at 2500 kbps (verified: actual bitrate ~2570 kbps)
|
||||||
|
- **Crash handler**: SIGSEGV/SIGABRT handler writes backtrace to `/tmp/dashcamd_crash.log`
|
||||||
- **Storage device**: WDC SDINDDH4-128G UFS 2.1 — automotive grade, ~384 TB write endurance, no concern for continuous writes
|
- **Storage device**: WDC SDINDDH4-128G UFS 2.1 — automotive grade, ~384 TB write endurance, no concern for continuous writes
|
||||||
|
|
||||||
### Key Differences from Old Screen Recorder
|
### OmxEncoder
|
||||||
|
|
||||||
| | Old (screen recorder) | New (dashcamd) |
|
The OMX encoder (`selfdrive/frogpilot/screenrecorder/omx_encoder.cc`) was ported from upstream FrogPilot with the following key properties:
|
||||||
|---|---|---|
|
|
||||||
| Source | `QWidget::grab()` screen capture | Raw NV12 from VisionIPC |
|
- Each encoder instance calls `OMX_Init()` in constructor and `OMX_Deinit()` in destructor — manages its own OMX lifecycle
|
||||||
| Resolution | 1440x720 | 1928x1208 |
|
- Constructor takes 5 args: `(path, width, height, fps, bitrate)` — no h265/downscale params
|
||||||
| Works with screen off | No (needs visible widget) | Yes (independent of UI) |
|
- `encoder_close()` calls `av_write_trailer` + ffmpeg faststart remux (`-movflags +faststart`)
|
||||||
| Process type | Part of UI process | Standalone native process |
|
- Destructor has null guards and error handling on all OMX state transitions
|
||||||
| Encoder input | RGBA -> NV12 conversion | NV12 direct (added `encode_frame_nv12`) |
|
- ClearPilot addition: `encode_frame_nv12()` for direct NV12 input (dashcamd), alongside original `encode_frame_rgba()` (screen recorder)
|
||||||
|
|
||||||
### Key Files
|
### Key Files
|
||||||
|
|
||||||
@@ -266,7 +267,7 @@ A single `session.log` in each session directory records major events:
|
|||||||
|------|------|
|
|------|------|
|
||||||
| `selfdrive/clearpilot/dashcamd.cc` | Main dashcam process — VisionIPC -> OMX encoder |
|
| `selfdrive/clearpilot/dashcamd.cc` | Main dashcam process — VisionIPC -> OMX encoder |
|
||||||
| `selfdrive/clearpilot/SConscript` | Build config for dashcamd |
|
| `selfdrive/clearpilot/SConscript` | Build config for dashcamd |
|
||||||
| `selfdrive/frogpilot/screenrecorder/omx_encoder.cc` | OMX encoder (added `encode_frame_nv12` method) |
|
| `selfdrive/frogpilot/screenrecorder/omx_encoder.cc` | OMX encoder (upstream FrogPilot port + `encode_frame_nv12`) |
|
||||||
| `selfdrive/frogpilot/screenrecorder/omx_encoder.h` | Encoder header |
|
| `selfdrive/frogpilot/screenrecorder/omx_encoder.h` | Encoder header |
|
||||||
| `selfdrive/manager/process_config.py` | dashcamd registered as NativeProcess, camerad always_run, encoderd disabled |
|
| `selfdrive/manager/process_config.py` | dashcamd registered as NativeProcess, camerad always_run, encoderd disabled |
|
||||||
| `system/loggerd/deleter.py` | Trip-aware storage rotation (oldest trip first, then segments within active trip) |
|
| `system/loggerd/deleter.py` | Trip-aware storage rotation (oldest trip first, then segments within active trip) |
|
||||||
@@ -275,12 +276,14 @@ A single `session.log` in each session directory records major events:
|
|||||||
|
|
||||||
- `DashcamDebug` — when `"1"`, dashcamd runs even without car connected (for bench testing)
|
- `DashcamDebug` — when `"1"`, dashcamd runs even without car connected (for bench testing)
|
||||||
- `DashcamShutdown` — set by thermald before power-off, dashcamd acks by clearing it
|
- `DashcamShutdown` — set by thermald before power-off, dashcamd acks by clearing it
|
||||||
|
- `DashcamState` (memory param) — "stopped", "waiting", or "recording" — published every 5s
|
||||||
|
- `DashcamFrames` (memory param) — per-trip encoded frame count, resets each trip — published every 5s
|
||||||
|
|
||||||
## Standstill Power Saving
|
## Standstill Power Saving
|
||||||
|
|
||||||
When `carState.standstill` is true:
|
When `carState.standstill` is true:
|
||||||
|
|
||||||
- **modeld**: skips GPU inference on 19/20 frames (1fps vs 20fps), reports 0 frame drops to avoid triggering `modeldLagging` in controlsd
|
- **modeld**: skips GPU inference on 19/20 frames (1fps vs 20fps), reports 0 frame drops to avoid triggering `modeldLagging` in controlsd. Runs full 20fps during calibration (`liveCalibration.calStatus != calibrated`)
|
||||||
- **dmonitoringmodeld**: same 1fps throttle, added `carState` subscription
|
- **dmonitoringmodeld**: same 1fps throttle, added `carState` subscription
|
||||||
- **Fan controller**: uses offroad clamps (0-30%) instead of onroad (30-100%) at standstill; thermal protection still active via feedforward if temp > 60°C
|
- **Fan controller**: uses offroad clamps (0-30%) instead of onroad (30-100%) at standstill; thermal protection still active via feedforward if temp > 60°C
|
||||||
|
|
||||||
@@ -321,6 +324,12 @@ The Hyundai Tucson's LFA steering wheel button cycles through 5 display modes vi
|
|||||||
|
|
||||||
**Not in drive (parked/off):** any except 3 → 3 (screen off), state 3 → 0 (auto-normal)
|
**Not in drive (parked/off):** any except 3 → 3 (screen off), state 3 → 0 (auto-normal)
|
||||||
|
|
||||||
|
**Shift to drive from screen off:** auto-resets to mode 0 (auto-normal) via `controlsd`
|
||||||
|
|
||||||
|
**Shift to park from nightrider:** auto-switches to mode 3 (screen off) via `home.cc`
|
||||||
|
|
||||||
|
**Tap screen while screen off:** resets to mode 0 (auto-normal) via `window.cc` touch handler
|
||||||
|
|
||||||
### Nightrider Mode
|
### Nightrider Mode
|
||||||
|
|
||||||
- Camera feed suppressed (OpenGL clears to black instead of rendering camera texture)
|
- Camera feed suppressed (OpenGL clears to black instead of rendering camera texture)
|
||||||
@@ -357,7 +366,10 @@ Display power is managed by `Device::updateWakefulness()` in `selfdrive/ui/ui.cc
|
|||||||
|
|
||||||
- **Ignition off (offroad)**: screen blanks after `ScreenTimeout` seconds (default 120) of no touch. Tapping wakes it.
|
- **Ignition off (offroad)**: screen blanks after `ScreenTimeout` seconds (default 120) of no touch. Tapping wakes it.
|
||||||
- **Ignition on (onroad)**: screen stays on unconditionally — ignition=true short-circuits the timeout check.
|
- **Ignition on (onroad)**: screen stays on unconditionally — ignition=true short-circuits the timeout check.
|
||||||
- **Debug button (LFA)**: cycles through display modes including screen off (state 3). Only state 3 calls `Hardware::set_display_power(false)`.
|
- **ScreenDisplayMode 3 override**: `updateWakefulness` checks `ScreenDisplayMode` first — if mode 3, calls `setAwake(false)` unconditionally, preventing ignition-on from overriding screen-off.
|
||||||
|
- **Debug button (LFA)**: cycles through display modes including screen off (state 3).
|
||||||
|
- **Park transition**: always shows splash screen; if coming from nightrider mode, auto-switches to screen off (mode 3) via `home.cc`.
|
||||||
|
- **Touch wake**: tapping screen while in mode 3 resets to mode 0 via `window.cc` event filter.
|
||||||
|
|
||||||
## Offroad UI (ClearPilot Menu)
|
## Offroad UI (ClearPilot Menu)
|
||||||
|
|
||||||
@@ -500,7 +512,7 @@ Power On
|
|||||||
- **GPS data**: logged directly by telemetryd via cereal `gpsLocation` subscription at 1Hz — group: `gps` (latitude, longitude, speed, altitude, bearing, accuracy)
|
- **GPS data**: logged directly by telemetryd via cereal `gpsLocation` subscription at 1Hz — group: `gps` (latitude, longitude, speed, altitude, bearing, accuracy)
|
||||||
- **CSV location**: `/data/log2/current/telemetry.csv` (or session directory)
|
- **CSV location**: `/data/log2/current/telemetry.csv` (or session directory)
|
||||||
- **Onroad overlay**: when telemetry enabled, status bar shows temp, fan %, model FPS, and STANDSTILL indicator
|
- **Onroad overlay**: when telemetry enabled, status bar shows temp, fan %, model FPS, and STANDSTILL indicator
|
||||||
- **Speed limit**: `speed_limit.calculated` is the final computed speed limit value (in vehicle units, MPH when `is_metric=False`). This is the value that will be used for the future speed limit warning chime feature
|
- **Speed limit**: processed by `selfdrive/clearpilot/speed_logic.py` (SpeedState class), converts m/s to display units, writes to memory params. Cruise warning signs appear when cruise set speed exceeds speed limit by threshold (10 mph if limit >= 50, 7 mph if < 50) or is 5+ mph under. Ding sound plays when warning sign appears or speed limit changes while visible (30s cooldown)
|
||||||
|
|
||||||
### Key Dependencies
|
### Key Dependencies
|
||||||
|
|
||||||
|
|||||||
@@ -110,6 +110,8 @@ std::unordered_map<std::string, uint32_t> keys = {
|
|||||||
{"CurrentBootlog", PERSISTENT},
|
{"CurrentBootlog", PERSISTENT},
|
||||||
{"CurrentRoute", CLEAR_ON_MANAGER_START | CLEAR_ON_ONROAD_TRANSITION},
|
{"CurrentRoute", CLEAR_ON_MANAGER_START | CLEAR_ON_ONROAD_TRANSITION},
|
||||||
{"DashcamDebug", PERSISTENT},
|
{"DashcamDebug", PERSISTENT},
|
||||||
|
{"DashcamFrames", PERSISTENT},
|
||||||
|
{"DashcamState", PERSISTENT},
|
||||||
{"DashcamShutdown", CLEAR_ON_MANAGER_START},
|
{"DashcamShutdown", CLEAR_ON_MANAGER_START},
|
||||||
{"DisableLogging", CLEAR_ON_MANAGER_START | CLEAR_ON_ONROAD_TRANSITION},
|
{"DisableLogging", CLEAR_ON_MANAGER_START | CLEAR_ON_ONROAD_TRANSITION},
|
||||||
{"DisablePowerDown", PERSISTENT},
|
{"DisablePowerDown", PERSISTENT},
|
||||||
@@ -217,6 +219,7 @@ std::unordered_map<std::string, uint32_t> keys = {
|
|||||||
|
|
||||||
// FrogPilot parameters
|
// FrogPilot parameters
|
||||||
{"AccelerationPath", PERSISTENT},
|
{"AccelerationPath", PERSISTENT},
|
||||||
|
{"BenchCruiseActive", CLEAR_ON_MANAGER_START},
|
||||||
{"BenchCruiseSpeed", CLEAR_ON_MANAGER_START},
|
{"BenchCruiseSpeed", CLEAR_ON_MANAGER_START},
|
||||||
{"ClpUiState", CLEAR_ON_MANAGER_START},
|
{"ClpUiState", CLEAR_ON_MANAGER_START},
|
||||||
{"BenchEngaged", CLEAR_ON_MANAGER_START},
|
{"BenchEngaged", CLEAR_ON_MANAGER_START},
|
||||||
@@ -250,6 +253,16 @@ std::unordered_map<std::string, uint32_t> keys = {
|
|||||||
|
|
||||||
{"CarSpeedLimitWarning", PERSISTENT},
|
{"CarSpeedLimitWarning", PERSISTENT},
|
||||||
{"CarSpeedLimitLiteral", PERSISTENT},
|
{"CarSpeedLimitLiteral", PERSISTENT},
|
||||||
|
{"CarIsMetric", PERSISTENT},
|
||||||
|
|
||||||
|
{"ClearpilotSpeedDisplay", PERSISTENT},
|
||||||
|
{"ClearpilotSpeedLimitDisplay", PERSISTENT},
|
||||||
|
{"ClearpilotHasSpeed", PERSISTENT},
|
||||||
|
{"ClearpilotIsMetric", PERSISTENT},
|
||||||
|
{"ClearpilotSpeedUnit", PERSISTENT},
|
||||||
|
{"ClearpilotCruiseWarning", PERSISTENT},
|
||||||
|
{"ClearpilotCruiseWarningSpeed", PERSISTENT},
|
||||||
|
{"ClearpilotPlayDing", PERSISTENT},
|
||||||
|
|
||||||
// {"SpeedLimitLatDesired", PERSISTENT},
|
// {"SpeedLimitLatDesired", PERSISTENT},
|
||||||
// {"SpeedLimitVTSC", PERSISTENT},
|
// {"SpeedLimitVTSC", PERSISTENT},
|
||||||
|
|||||||
@@ -32,7 +32,9 @@ def main():
|
|||||||
continue
|
continue
|
||||||
|
|
||||||
cloudlog.info("starting athena daemon")
|
cloudlog.info("starting athena daemon")
|
||||||
proc = Process(name='athenad', target=launcher, args=('selfdrive.athena.athenad', 'athenad'))
|
from openpilot.selfdrive.manager.process import _log_dir
|
||||||
|
log_path = _log_dir + "/athenad.log"
|
||||||
|
proc = Process(name='athenad', target=launcher, args=('selfdrive.athena.athenad', 'athenad', log_path))
|
||||||
proc.start()
|
proc.start()
|
||||||
proc.join()
|
proc.join()
|
||||||
cloudlog.event("athenad exited", exitcode=proc.exitcode)
|
cloudlog.event("athenad exited", exitcode=proc.exitcode)
|
||||||
|
|||||||
@@ -212,6 +212,7 @@ class CarState(CarStateBase):
|
|||||||
|
|
||||||
# self.params_memory.put_int("CarSpeedLimitLiteral", self.calculate_speed_limit(cp, cp_cam))
|
# self.params_memory.put_int("CarSpeedLimitLiteral", self.calculate_speed_limit(cp, cp_cam))
|
||||||
self.params_memory.put_float("CarSpeedLimit", self.calculate_speed_limit(cp, cp_cam) * speed_conv)
|
self.params_memory.put_float("CarSpeedLimit", self.calculate_speed_limit(cp, cp_cam) * speed_conv)
|
||||||
|
self.params_memory.put("CarIsMetric", "1" if self.is_metric else "0")
|
||||||
self.params_memory.put_float("CarCruiseDisplayActual", cp_cruise.vl["SCC11"]["VSetDis"])
|
self.params_memory.put_float("CarCruiseDisplayActual", cp_cruise.vl["SCC11"]["VSetDis"])
|
||||||
|
|
||||||
|
|
||||||
@@ -420,6 +421,7 @@ class CarState(CarStateBase):
|
|||||||
# print(self.calculate_speed_limit(cp, cp_cam))
|
# print(self.calculate_speed_limit(cp, cp_cam))
|
||||||
# self.params_memory.put_float("CarSpeedLimitLiteral", self.calculate_speed_limit(cp, cp_cam))
|
# self.params_memory.put_float("CarSpeedLimitLiteral", self.calculate_speed_limit(cp, cp_cam))
|
||||||
self.params_memory.put_float("CarSpeedLimit", self.calculate_speed_limit(cp, cp_cam) * speed_factor)
|
self.params_memory.put_float("CarSpeedLimit", self.calculate_speed_limit(cp, cp_cam) * speed_factor)
|
||||||
|
self.params_memory.put("CarIsMetric", "1" if self.is_metric else "0")
|
||||||
|
|
||||||
# CLEARPILOT: telemetry logging — disabled, re-enable when needed
|
# CLEARPILOT: telemetry logging — disabled, re-enable when needed
|
||||||
# speed_limit_bus = cp if self.CP.flags & HyundaiFlags.CANFD_HDA2 else cp_cam
|
# speed_limit_bus = cp if self.CP.flags & HyundaiFlags.CANFD_HDA2 else cp_cam
|
||||||
|
|||||||
@@ -7,7 +7,9 @@ Usage:
|
|||||||
python3 -m selfdrive.clearpilot.bench_cmd speed 20
|
python3 -m selfdrive.clearpilot.bench_cmd speed 20
|
||||||
python3 -m selfdrive.clearpilot.bench_cmd speedlimit 45
|
python3 -m selfdrive.clearpilot.bench_cmd speedlimit 45
|
||||||
python3 -m selfdrive.clearpilot.bench_cmd cruise 55
|
python3 -m selfdrive.clearpilot.bench_cmd cruise 55
|
||||||
|
python3 -m selfdrive.clearpilot.bench_cmd cruiseactive 0|1|2 (0=disabled, 1=active, 2=paused)
|
||||||
python3 -m selfdrive.clearpilot.bench_cmd engaged 1
|
python3 -m selfdrive.clearpilot.bench_cmd engaged 1
|
||||||
|
python3 -m selfdrive.clearpilot.bench_cmd ding (trigger speed limit ding sound)
|
||||||
python3 -m selfdrive.clearpilot.bench_cmd debugbutton (simulate LKAS debug button press)
|
python3 -m selfdrive.clearpilot.bench_cmd debugbutton (simulate LKAS debug button press)
|
||||||
python3 -m selfdrive.clearpilot.bench_cmd dump
|
python3 -m selfdrive.clearpilot.bench_cmd dump
|
||||||
python3 -m selfdrive.clearpilot.bench_cmd wait_ready
|
python3 -m selfdrive.clearpilot.bench_cmd wait_ready
|
||||||
@@ -89,6 +91,7 @@ def main():
|
|||||||
"speed": "BenchSpeed",
|
"speed": "BenchSpeed",
|
||||||
"speedlimit": "BenchSpeedLimit",
|
"speedlimit": "BenchSpeedLimit",
|
||||||
"cruise": "BenchCruiseSpeed",
|
"cruise": "BenchCruiseSpeed",
|
||||||
|
"cruiseactive": "BenchCruiseActive",
|
||||||
"engaged": "BenchEngaged",
|
"engaged": "BenchEngaged",
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -106,6 +109,10 @@ def main():
|
|||||||
elif cmd == "wait_ready":
|
elif cmd == "wait_ready":
|
||||||
wait_ready()
|
wait_ready()
|
||||||
|
|
||||||
|
elif cmd == "ding":
|
||||||
|
params.put("ClearpilotPlayDing", "1")
|
||||||
|
print("Ding triggered")
|
||||||
|
|
||||||
elif cmd == "debugbutton":
|
elif cmd == "debugbutton":
|
||||||
# Simulate LKAS debug button — same state machine as controlsd.clearpilot_state_control()
|
# Simulate LKAS debug button — same state machine as controlsd.clearpilot_state_control()
|
||||||
current = params.get_int("ScreenDisplayMode")
|
current = params.get_int("ScreenDisplayMode")
|
||||||
|
|||||||
@@ -8,6 +8,7 @@ configurable vehicle state. Control values via params in /dev/shm/params:
|
|||||||
BenchSpeed - vehicle speed in mph (default: 0)
|
BenchSpeed - vehicle speed in mph (default: 0)
|
||||||
BenchSpeedLimit - speed limit in mph (default: 0, 0=hidden)
|
BenchSpeedLimit - speed limit in mph (default: 0, 0=hidden)
|
||||||
BenchCruiseSpeed - cruise set speed in mph (default: 0, 0=not set)
|
BenchCruiseSpeed - cruise set speed in mph (default: 0, 0=not set)
|
||||||
|
BenchCruiseActive - 0=disabled, 1=active, 2=paused/standstill (default: 0)
|
||||||
BenchGear - P, D, R, N (default: P)
|
BenchGear - P, D, R, N (default: P)
|
||||||
BenchEngaged - 0 or 1, cruise engaged (default: 0)
|
BenchEngaged - 0 or 1, cruise engaged (default: 0)
|
||||||
|
|
||||||
@@ -21,8 +22,8 @@ import time
|
|||||||
import cereal.messaging as messaging
|
import cereal.messaging as messaging
|
||||||
from cereal import log, car
|
from cereal import log, car
|
||||||
from openpilot.common.params import Params
|
from openpilot.common.params import Params
|
||||||
|
from openpilot.common.conversions import Conversions as CV
|
||||||
MS_PER_MPH = 0.44704
|
from openpilot.selfdrive.clearpilot.speed_logic import SpeedState
|
||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
@@ -33,9 +34,13 @@ def main():
|
|||||||
params_mem.put("BenchSpeed", "0")
|
params_mem.put("BenchSpeed", "0")
|
||||||
params_mem.put("BenchSpeedLimit", "0")
|
params_mem.put("BenchSpeedLimit", "0")
|
||||||
params_mem.put("BenchCruiseSpeed", "0")
|
params_mem.put("BenchCruiseSpeed", "0")
|
||||||
|
params_mem.put("BenchCruiseActive", "0")
|
||||||
params_mem.put("BenchGear", "P")
|
params_mem.put("BenchGear", "P")
|
||||||
params_mem.put("BenchEngaged", "0")
|
params_mem.put("BenchEngaged", "0")
|
||||||
|
|
||||||
|
# ClearPilot speed processing
|
||||||
|
speed_state = SpeedState()
|
||||||
|
|
||||||
# thermald handles deviceState (reads our fake pandaStates for ignition)
|
# thermald handles deviceState (reads our fake pandaStates for ignition)
|
||||||
pm = messaging.PubMaster([
|
pm = messaging.PubMaster([
|
||||||
"pandaStates", "carState", "controlsState",
|
"pandaStates", "carState", "controlsState",
|
||||||
@@ -61,11 +66,25 @@ def main():
|
|||||||
speed_limit_mph = float((params_mem.get("BenchSpeedLimit", encoding="utf-8") or "0").strip())
|
speed_limit_mph = float((params_mem.get("BenchSpeedLimit", encoding="utf-8") or "0").strip())
|
||||||
cruise_mph = float((params_mem.get("BenchCruiseSpeed", encoding="utf-8") or "0").strip())
|
cruise_mph = float((params_mem.get("BenchCruiseSpeed", encoding="utf-8") or "0").strip())
|
||||||
gear_str = (params_mem.get("BenchGear", encoding="utf-8") or "P").strip().upper()
|
gear_str = (params_mem.get("BenchGear", encoding="utf-8") or "P").strip().upper()
|
||||||
|
cruise_active_str = (params_mem.get("BenchCruiseActive", encoding="utf-8") or "0").strip()
|
||||||
engaged = (params_mem.get("BenchEngaged", encoding="utf-8") or "0").strip() == "1"
|
engaged = (params_mem.get("BenchEngaged", encoding="utf-8") or "0").strip() == "1"
|
||||||
|
|
||||||
speed_ms = speed_mph * MS_PER_MPH
|
speed_ms = speed_mph * CV.MPH_TO_MS
|
||||||
|
speed_limit_ms = speed_limit_mph * CV.MPH_TO_MS
|
||||||
|
cruise_ms = cruise_mph * CV.MPH_TO_MS
|
||||||
gear = gear_map.get(gear_str, car.CarState.GearShifter.park)
|
gear = gear_map.get(gear_str, car.CarState.GearShifter.park)
|
||||||
|
|
||||||
|
# Cruise state: 0=disabled, 1=active, 2=paused
|
||||||
|
cruise_active = cruise_active_str == "1"
|
||||||
|
cruise_standstill = cruise_active_str == "2"
|
||||||
|
|
||||||
|
# ClearPilot speed processing (~2 Hz at 10 Hz loop)
|
||||||
|
if frame % 5 == 0:
|
||||||
|
has_speed = speed_mph > 0
|
||||||
|
speed_state.update(speed_ms, has_speed, speed_limit_ms, is_metric=False,
|
||||||
|
cruise_speed_ms=cruise_ms, cruise_active=cruise_active or cruise_standstill,
|
||||||
|
cruise_standstill=cruise_standstill)
|
||||||
|
|
||||||
# pandaStates — 10 Hz (thermald reads ignition from this)
|
# pandaStates — 10 Hz (thermald reads ignition from this)
|
||||||
if frame % 1 == 0:
|
if frame % 1 == 0:
|
||||||
dat = messaging.new_message("pandaStates", 1)
|
dat = messaging.new_message("pandaStates", 1)
|
||||||
@@ -82,7 +101,7 @@ def main():
|
|||||||
cs.standstill = speed_ms < 0.1
|
cs.standstill = speed_ms < 0.1
|
||||||
cs.cruiseState.available = True
|
cs.cruiseState.available = True
|
||||||
cs.cruiseState.enabled = engaged
|
cs.cruiseState.enabled = engaged
|
||||||
cs.cruiseState.speed = cruise_mph * MS_PER_MPH if cruise_mph > 0 else 0
|
cs.cruiseState.speed = cruise_mph * CV.MPH_TO_MS if cruise_mph > 0 else 0
|
||||||
pm.send("carState", dat)
|
pm.send("carState", dat)
|
||||||
|
|
||||||
# controlsState — 10 Hz
|
# controlsState — 10 Hz
|
||||||
|
|||||||
Binary file not shown.
@@ -7,49 +7,39 @@
|
|||||||
* Trip directory structure:
|
* Trip directory structure:
|
||||||
* /data/media/0/videos/YYYYMMDD-HHMMSS/ (trip directory, named at trip start)
|
* /data/media/0/videos/YYYYMMDD-HHMMSS/ (trip directory, named at trip start)
|
||||||
* YYYYMMDD-HHMMSS.mp4 (3-minute segments)
|
* YYYYMMDD-HHMMSS.mp4 (3-minute segments)
|
||||||
|
* YYYYMMDD-HHMMSS.srt (GPS subtitle sidecar)
|
||||||
*
|
*
|
||||||
* Trip lifecycle state machine:
|
* Trip lifecycle state machine:
|
||||||
*
|
*
|
||||||
* On process start (after time-valid wait):
|
* WAITING:
|
||||||
* - Create trip directory, start recording immediately with 10-min idle timer
|
* - Process starts in this state
|
||||||
* - If car is already in drive, timer is cancelled and recording continues
|
* - Waits for valid system time (year >= 2024) AND car in drive gear
|
||||||
* - If car stays parked/off for 10 minutes, trip ends
|
* - Transitions to RECORDING when both conditions met
|
||||||
*
|
*
|
||||||
* IDLE_TIMEOUT → RECORDING:
|
* RECORDING:
|
||||||
* - Car enters drive gear before timer expires → cancel timer, resume recording
|
* - Actively encoding frames, car is in drive
|
||||||
* in the same trip (no new trip directory)
|
* - Car leaves drive → start 10-min idle timer → IDLE_TIMEOUT
|
||||||
*
|
*
|
||||||
* RECORDING → IDLE_TIMEOUT:
|
* IDLE_TIMEOUT:
|
||||||
* - Car leaves drive gear (park, off, neutral) → start 10-minute idle timer,
|
* - Car left drive, still recording with timer running
|
||||||
* continue recording frames during the timeout period
|
* - Car re-enters drive → cancel timer → RECORDING
|
||||||
*
|
* - Timer expires → close trip → WAITING
|
||||||
* IDLE_TIMEOUT → TRIP_ENDED:
|
* - Ignition off → close trip → WAITING
|
||||||
* - 10-minute timer expires → close segment, trip is over
|
|
||||||
*
|
|
||||||
* TRIP_ENDED → RECORDING (new trip):
|
|
||||||
* - Car enters drive gear → create new trip directory, start recording
|
|
||||||
*
|
|
||||||
* Any state → (new trip) on ignition off→on:
|
|
||||||
* - Ignition transitions off→on → close current trip if active, create new trip
|
|
||||||
* directory, start recording with 10-min idle timer. This applies even from
|
|
||||||
* TRIP_ENDED — turning the car on always starts a new trip. If the car is in
|
|
||||||
* park after ignition on, the idle timer runs; entering drive cancels it.
|
|
||||||
*
|
*
|
||||||
* Graceful shutdown (DashcamShutdown param):
|
* Graceful shutdown (DashcamShutdown param):
|
||||||
* - thermald sets DashcamShutdown="1" before device power-off
|
* - thermald sets DashcamShutdown="1" before device power-off
|
||||||
* - dashcamd closes current segment, sets DashcamShutdown="0" (ack), exits
|
* - dashcamd closes current segment, acks, exits
|
||||||
* - thermald waits up to 15s for ack, then proceeds with shutdown
|
|
||||||
*
|
*
|
||||||
* GPS subtitle track:
|
* Published params (memory, every 5s):
|
||||||
* - Each .mp4 segment has a companion .srt subtitle file
|
* - DashcamState: "waiting" or "recording"
|
||||||
* - Updated at most once per second from gpsLocation cereal messages
|
* - DashcamFrames: per-trip encoded frame count (resets each trip)
|
||||||
* - Format: "35 MPH | 44.9216°N 93.3260°W | 2026-04-13 05:19:00 UTC"
|
|
||||||
* - Most players auto-detect .srt files alongside .mp4 files
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <cstdio>
|
#include <cstdio>
|
||||||
#include <ctime>
|
#include <ctime>
|
||||||
#include <string>
|
#include <string>
|
||||||
|
#include <signal.h>
|
||||||
|
#include <execinfo.h>
|
||||||
#include <sys/stat.h>
|
#include <sys/stat.h>
|
||||||
#include <sys/resource.h>
|
#include <sys/resource.h>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
@@ -73,10 +63,9 @@ const double IDLE_TIMEOUT_SECONDS = 600.0; // 10 minutes
|
|||||||
ExitHandler do_exit;
|
ExitHandler do_exit;
|
||||||
|
|
||||||
enum TripState {
|
enum TripState {
|
||||||
IDLE, // no trip active, waiting for drive
|
WAITING, // no trip, waiting for valid time + drive gear
|
||||||
RECORDING, // actively recording, car in drive
|
RECORDING, // actively recording, car in drive
|
||||||
IDLE_TIMEOUT, // car parked/off, recording with 10-min timer
|
IDLE_TIMEOUT, // car left drive, recording with 10-min timer
|
||||||
TRIP_ENDED, // trip closed, waiting for next drive
|
|
||||||
};
|
};
|
||||||
|
|
||||||
static std::string make_timestamp() {
|
static std::string make_timestamp() {
|
||||||
@@ -115,21 +104,26 @@ static std::string srt_time(int seconds) {
|
|||||||
return std::string(buf);
|
return std::string(buf);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void crash_handler(int sig) {
|
||||||
|
FILE *f = fopen("/tmp/dashcamd_crash.log", "w");
|
||||||
|
if (f) {
|
||||||
|
fprintf(f, "CRASH: signal %d\n", sig);
|
||||||
|
void *bt[30];
|
||||||
|
int n = backtrace(bt, 30);
|
||||||
|
backtrace_symbols_fd(bt, n, fileno(f));
|
||||||
|
fclose(f);
|
||||||
|
}
|
||||||
|
_exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
int main(int argc, char *argv[]) {
|
int main(int argc, char *argv[]) {
|
||||||
|
signal(SIGSEGV, crash_handler);
|
||||||
|
signal(SIGABRT, crash_handler);
|
||||||
setpriority(PRIO_PROCESS, 0, -10);
|
setpriority(PRIO_PROCESS, 0, -10);
|
||||||
|
|
||||||
// Ensure base output directory exists
|
// Ensure base output directory exists
|
||||||
mkdir(VIDEOS_BASE.c_str(), 0755);
|
mkdir(VIDEOS_BASE.c_str(), 0755);
|
||||||
|
|
||||||
// Wait for valid system time so trip/segment names have real timestamps
|
|
||||||
LOGW("dashcamd: waiting for system time");
|
|
||||||
while (!do_exit) {
|
|
||||||
if (system_time_valid()) break;
|
|
||||||
usleep(1000000); // 1 Hz poll
|
|
||||||
}
|
|
||||||
if (do_exit) return 0;
|
|
||||||
LOGW("dashcamd: system time valid");
|
|
||||||
|
|
||||||
LOGW("dashcamd: started, connecting to camerad road stream");
|
LOGW("dashcamd: started, connecting to camerad road stream");
|
||||||
VisionIpcClient vipc("camerad", VISION_STREAM_ROAD, false);
|
VisionIpcClient vipc("camerad", VISION_STREAM_ROAD, false);
|
||||||
while (!do_exit && !vipc.connect(false)) {
|
while (!do_exit && !vipc.connect(false)) {
|
||||||
@@ -156,51 +150,50 @@ int main(int argc, char *argv[]) {
|
|||||||
// Subscribe to carState (gear), deviceState (ignition), gpsLocation (subtitles)
|
// Subscribe to carState (gear), deviceState (ignition), gpsLocation (subtitles)
|
||||||
SubMaster sm({"carState", "deviceState", "gpsLocation"});
|
SubMaster sm({"carState", "deviceState", "gpsLocation"});
|
||||||
Params params;
|
Params params;
|
||||||
|
Params params_memory("/dev/shm/params");
|
||||||
|
|
||||||
// Trip state
|
// Trip state
|
||||||
TripState state = IDLE;
|
TripState state = WAITING;
|
||||||
OmxEncoder *encoder = nullptr;
|
OmxEncoder *encoder = nullptr;
|
||||||
std::string trip_dir;
|
std::string trip_dir;
|
||||||
int frame_count = 0;
|
int frame_count = 0; // per-segment (for rotation)
|
||||||
|
int trip_frames = 0; // per-trip (published to params)
|
||||||
int recv_count = 0;
|
int recv_count = 0;
|
||||||
uint64_t segment_start_ts = 0;
|
uint64_t segment_start_ts = 0;
|
||||||
double idle_timer_start = 0.0;
|
double idle_timer_start = 0.0;
|
||||||
|
|
||||||
// SRT subtitle state
|
// SRT subtitle state
|
||||||
FILE *srt_file = nullptr;
|
FILE *srt_file = nullptr;
|
||||||
int srt_index = 0; // subtitle entry counter (1-based)
|
int srt_index = 0;
|
||||||
int srt_segment_sec = 0; // seconds elapsed in current segment
|
int srt_segment_sec = 0;
|
||||||
double last_srt_write = 0; // monotonic time of last SRT write
|
double last_srt_write = 0;
|
||||||
|
|
||||||
// Ignition tracking for off→on detection
|
// Ignition tracking
|
||||||
bool prev_started = false;
|
bool prev_started = false;
|
||||||
bool started_initialized = false;
|
bool started_initialized = false;
|
||||||
|
|
||||||
// Param check throttle (don't hit filesystem every frame)
|
// Param publish throttle
|
||||||
int param_check_counter = 0;
|
int param_check_counter = 0;
|
||||||
|
double last_param_write = 0;
|
||||||
|
|
||||||
// Helper: start a new trip with recording + optional idle timer
|
// Publish initial state
|
||||||
|
params_memory.put("DashcamState", "waiting");
|
||||||
|
params_memory.put("DashcamFrames", "0");
|
||||||
|
|
||||||
|
LOGW("dashcamd: entering main loop in WAITING state");
|
||||||
|
|
||||||
|
// Helper: start a new trip
|
||||||
auto start_new_trip = [&]() {
|
auto start_new_trip = [&]() {
|
||||||
// Close existing encoder if any
|
|
||||||
if (encoder) {
|
|
||||||
if (state == RECORDING || state == IDLE_TIMEOUT) {
|
|
||||||
encoder->encoder_close();
|
|
||||||
}
|
|
||||||
delete encoder;
|
|
||||||
encoder = nullptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
trip_dir = VIDEOS_BASE + "/" + make_timestamp();
|
trip_dir = VIDEOS_BASE + "/" + make_timestamp();
|
||||||
mkdir(trip_dir.c_str(), 0755);
|
mkdir(trip_dir.c_str(), 0755);
|
||||||
LOGW("dashcamd: new trip %s", trip_dir.c_str());
|
LOGW("dashcamd: new trip %s", trip_dir.c_str());
|
||||||
|
|
||||||
encoder = new OmxEncoder(trip_dir.c_str(), width, height, CAMERA_FPS, BITRATE, false, false);
|
encoder = new OmxEncoder(trip_dir.c_str(), width, height, CAMERA_FPS, BITRATE);
|
||||||
|
|
||||||
std::string seg_name = make_timestamp();
|
std::string seg_name = make_timestamp();
|
||||||
LOGW("dashcamd: opening segment %s", seg_name.c_str());
|
LOGW("dashcamd: opening segment %s", seg_name.c_str());
|
||||||
encoder->encoder_open((seg_name + ".mp4").c_str());
|
encoder->encoder_open((seg_name + ".mp4").c_str());
|
||||||
|
|
||||||
// Open companion SRT file
|
|
||||||
std::string srt_path = trip_dir + "/" + seg_name + ".srt";
|
std::string srt_path = trip_dir + "/" + seg_name + ".srt";
|
||||||
srt_file = fopen(srt_path.c_str(), "w");
|
srt_file = fopen(srt_path.c_str(), "w");
|
||||||
srt_index = 0;
|
srt_index = 0;
|
||||||
@@ -208,38 +201,38 @@ int main(int argc, char *argv[]) {
|
|||||||
last_srt_write = 0;
|
last_srt_write = 0;
|
||||||
|
|
||||||
frame_count = 0;
|
frame_count = 0;
|
||||||
|
trip_frames = 0;
|
||||||
segment_start_ts = nanos_since_boot();
|
segment_start_ts = nanos_since_boot();
|
||||||
state = RECORDING;
|
state = RECORDING;
|
||||||
|
|
||||||
|
params_memory.put("DashcamState", "recording");
|
||||||
|
params_memory.put("DashcamFrames", "0");
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// Helper: close current trip
|
||||||
auto close_trip = [&]() {
|
auto close_trip = [&]() {
|
||||||
if (srt_file) { fclose(srt_file); srt_file = nullptr; }
|
if (srt_file) { fclose(srt_file); srt_file = nullptr; }
|
||||||
if (encoder) {
|
if (encoder) {
|
||||||
if (state == RECORDING || state == IDLE_TIMEOUT) {
|
encoder->encoder_close();
|
||||||
encoder->encoder_close();
|
LOGW("dashcamd: segment closed");
|
||||||
LOGW("dashcamd: segment closed");
|
|
||||||
}
|
|
||||||
delete encoder;
|
delete encoder;
|
||||||
encoder = nullptr;
|
encoder = nullptr;
|
||||||
}
|
}
|
||||||
state = TRIP_ENDED;
|
state = WAITING;
|
||||||
frame_count = 0;
|
frame_count = 0;
|
||||||
|
trip_frames = 0;
|
||||||
idle_timer_start = 0.0;
|
idle_timer_start = 0.0;
|
||||||
LOGW("dashcamd: trip ended");
|
LOGW("dashcamd: trip ended, returning to WAITING");
|
||||||
};
|
|
||||||
|
|
||||||
// Start recording immediately — if the car is in drive, great; if parked/off,
|
params_memory.put("DashcamState", "waiting");
|
||||||
// the 10-min idle timer will stop the trip if drive is never detected.
|
params_memory.put("DashcamFrames", "0");
|
||||||
start_new_trip();
|
};
|
||||||
idle_timer_start = nanos_since_boot() / 1e9;
|
|
||||||
state = IDLE_TIMEOUT;
|
|
||||||
LOGW("dashcamd: recording started, waiting for drive (10-min idle timer active)");
|
|
||||||
|
|
||||||
while (!do_exit) {
|
while (!do_exit) {
|
||||||
VisionBuf *buf = vipc.recv();
|
VisionBuf *buf = vipc.recv();
|
||||||
if (buf == nullptr) continue;
|
if (buf == nullptr) continue;
|
||||||
|
|
||||||
// CLEARPILOT: skip frames to match target FPS (SOURCE_FPS -> CAMERA_FPS)
|
// Skip frames to match target FPS (SOURCE_FPS -> CAMERA_FPS)
|
||||||
recv_count++;
|
recv_count++;
|
||||||
if (SOURCE_FPS > CAMERA_FPS && (recv_count % (SOURCE_FPS / CAMERA_FPS)) != 0) continue;
|
if (SOURCE_FPS > CAMERA_FPS && (recv_count % (SOURCE_FPS / CAMERA_FPS)) != 0) continue;
|
||||||
|
|
||||||
@@ -257,25 +250,21 @@ int main(int argc, char *argv[]) {
|
|||||||
gear == cereal::CarState::GearShifter::LOW ||
|
gear == cereal::CarState::GearShifter::LOW ||
|
||||||
gear == cereal::CarState::GearShifter::MANUMATIC);
|
gear == cereal::CarState::GearShifter::MANUMATIC);
|
||||||
|
|
||||||
// Detect ignition off→on transition (new ignition cycle = new trip)
|
// Detect ignition off → close any active trip
|
||||||
if (started_initialized && !prev_started && started) {
|
if (started_initialized && prev_started && !started) {
|
||||||
LOGW("dashcamd: ignition on — new cycle");
|
LOGW("dashcamd: ignition off");
|
||||||
if (state == RECORDING || state == IDLE_TIMEOUT) {
|
if (state == RECORDING || state == IDLE_TIMEOUT) {
|
||||||
close_trip();
|
close_trip();
|
||||||
}
|
}
|
||||||
// Start recording immediately, idle timer until drive is detected
|
|
||||||
start_new_trip();
|
|
||||||
idle_timer_start = now;
|
|
||||||
state = IDLE_TIMEOUT;
|
|
||||||
}
|
}
|
||||||
prev_started = started;
|
prev_started = started;
|
||||||
started_initialized = true;
|
started_initialized = true;
|
||||||
|
|
||||||
// Check for graceful shutdown request (every ~1 second = 20 frames)
|
// Check for graceful shutdown request (every ~1 second)
|
||||||
if (++param_check_counter >= CAMERA_FPS) {
|
if (++param_check_counter >= CAMERA_FPS) {
|
||||||
param_check_counter = 0;
|
param_check_counter = 0;
|
||||||
if (params.getBool("DashcamShutdown")) {
|
if (params.getBool("DashcamShutdown")) {
|
||||||
LOGW("dashcamd: shutdown requested, closing trip");
|
LOGW("dashcamd: shutdown requested");
|
||||||
if (state == RECORDING || state == IDLE_TIMEOUT) {
|
if (state == RECORDING || state == IDLE_TIMEOUT) {
|
||||||
close_trip();
|
close_trip();
|
||||||
}
|
}
|
||||||
@@ -285,32 +274,30 @@ int main(int argc, char *argv[]) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// State machine transitions
|
// State machine
|
||||||
switch (state) {
|
switch (state) {
|
||||||
case IDLE:
|
case WAITING: {
|
||||||
case TRIP_ENDED:
|
bool has_gps = sm.valid("gpsLocation") && sm["gpsLocation"].getGpsLocation().getHasFix();
|
||||||
if (in_drive) {
|
if (in_drive && system_time_valid() && has_gps) {
|
||||||
start_new_trip();
|
start_new_trip();
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
case RECORDING:
|
case RECORDING:
|
||||||
if (!in_drive) {
|
if (!in_drive) {
|
||||||
// Car left drive — start idle timeout
|
|
||||||
idle_timer_start = now;
|
idle_timer_start = now;
|
||||||
state = IDLE_TIMEOUT;
|
state = IDLE_TIMEOUT;
|
||||||
LOGW("dashcamd: car not in drive, starting 10-min idle timer");
|
LOGW("dashcamd: car left drive, starting 10-min idle timer");
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case IDLE_TIMEOUT:
|
case IDLE_TIMEOUT:
|
||||||
if (in_drive) {
|
if (in_drive) {
|
||||||
// Back in drive — cancel timer, resume recording same trip
|
|
||||||
idle_timer_start = 0.0;
|
idle_timer_start = 0.0;
|
||||||
state = RECORDING;
|
state = RECORDING;
|
||||||
LOGW("dashcamd: back in drive, resuming trip");
|
LOGW("dashcamd: back in drive, resuming trip");
|
||||||
} else if ((now - idle_timer_start) >= IDLE_TIMEOUT_SECONDS) {
|
} else if ((now - idle_timer_start) >= IDLE_TIMEOUT_SECONDS) {
|
||||||
// Timer expired — end trip
|
|
||||||
LOGW("dashcamd: idle timeout expired");
|
LOGW("dashcamd: idle timeout expired");
|
||||||
close_trip();
|
close_trip();
|
||||||
}
|
}
|
||||||
@@ -341,16 +328,28 @@ int main(int argc, char *argv[]) {
|
|||||||
|
|
||||||
uint64_t ts = nanos_since_boot() - segment_start_ts;
|
uint64_t ts = nanos_since_boot() - segment_start_ts;
|
||||||
|
|
||||||
|
// Validate buffer before encoding
|
||||||
|
if (buf->y == nullptr || buf->uv == nullptr || buf->width == 0 || buf->height == 0) {
|
||||||
|
LOGE("dashcamd: invalid frame buf y=%p uv=%p %zux%zu, skipping", buf->y, buf->uv, buf->width, buf->height);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
// Feed NV12 frame directly to OMX encoder
|
// Feed NV12 frame directly to OMX encoder
|
||||||
encoder->encode_frame_nv12(buf->y, y_stride, buf->uv, uv_stride, width, height, ts);
|
encoder->encode_frame_nv12(buf->y, y_stride, buf->uv, uv_stride, width, height, ts);
|
||||||
frame_count++;
|
frame_count++;
|
||||||
|
trip_frames++;
|
||||||
|
|
||||||
|
// Publish state every 5 seconds
|
||||||
|
if (now - last_param_write >= 5.0) {
|
||||||
|
params_memory.put("DashcamFrames", std::to_string(trip_frames));
|
||||||
|
last_param_write = now;
|
||||||
|
}
|
||||||
|
|
||||||
// Write GPS subtitle at most once per second
|
// Write GPS subtitle at most once per second
|
||||||
if (srt_file && (now - last_srt_write) >= 1.0) {
|
if (srt_file && (now - last_srt_write) >= 1.0) {
|
||||||
last_srt_write = now;
|
last_srt_write = now;
|
||||||
srt_index++;
|
srt_index++;
|
||||||
|
|
||||||
// Read GPS data
|
|
||||||
double lat = 0, lon = 0, speed_ms = 0;
|
double lat = 0, lon = 0, speed_ms = 0;
|
||||||
bool has_gps = sm.valid("gpsLocation") && sm["gpsLocation"].getGpsLocation().getHasFix();
|
bool has_gps = sm.valid("gpsLocation") && sm["gpsLocation"].getGpsLocation().getHasFix();
|
||||||
if (has_gps) {
|
if (has_gps) {
|
||||||
@@ -383,13 +382,11 @@ int main(int argc, char *argv[]) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Clean exit
|
// Clean exit
|
||||||
if (srt_file) { fclose(srt_file); srt_file = nullptr; }
|
if (state == RECORDING || state == IDLE_TIMEOUT) {
|
||||||
if (encoder) {
|
close_trip();
|
||||||
if (state == RECORDING || state == IDLE_TIMEOUT) {
|
|
||||||
encoder->encoder_close();
|
|
||||||
}
|
|
||||||
delete encoder;
|
|
||||||
}
|
}
|
||||||
|
params_memory.put("DashcamState", "stopped");
|
||||||
|
params_memory.put("DashcamFrames", "0");
|
||||||
LOGW("dashcamd: stopped");
|
LOGW("dashcamd: stopped");
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|||||||
BIN
selfdrive/clearpilot/sounds/ding.wav
Normal file
BIN
selfdrive/clearpilot/sounds/ding.wav
Normal file
Binary file not shown.
90
selfdrive/clearpilot/speed_logic.py
Normal file
90
selfdrive/clearpilot/speed_logic.py
Normal file
@@ -0,0 +1,90 @@
|
|||||||
|
"""
|
||||||
|
ClearPilot speed processing module.
|
||||||
|
|
||||||
|
Shared logic for converting raw speed and speed limit data into display-ready
|
||||||
|
values. Called from controlsd (live mode) and bench_onroad (bench mode).
|
||||||
|
|
||||||
|
Reads raw inputs, converts to display units (mph or kph based on car's CAN
|
||||||
|
unit setting), detects speed limit changes, and writes results to params_memory
|
||||||
|
for the onroad UI to read.
|
||||||
|
"""
|
||||||
|
import math
|
||||||
|
import time
|
||||||
|
from openpilot.common.params import Params
|
||||||
|
from openpilot.common.conversions import Conversions as CV
|
||||||
|
|
||||||
|
|
||||||
|
class SpeedState:
|
||||||
|
def __init__(self):
|
||||||
|
self.params_memory = Params("/dev/shm/params")
|
||||||
|
self.prev_speed_limit = 0
|
||||||
|
|
||||||
|
# Ding state tracking
|
||||||
|
self.last_ding_time = 0.0
|
||||||
|
self.prev_warning = ""
|
||||||
|
self.prev_warning_speed_limit = 0
|
||||||
|
|
||||||
|
def update(self, speed_ms: float, has_speed: bool, speed_limit_ms: float, is_metric: bool,
|
||||||
|
cruise_speed_ms: float = 0.0, cruise_active: bool = False, cruise_standstill: bool = False):
|
||||||
|
"""
|
||||||
|
Convert raw m/s values to display-ready strings and write to params_memory.
|
||||||
|
"""
|
||||||
|
now = time.monotonic()
|
||||||
|
|
||||||
|
if is_metric:
|
||||||
|
speed_display = speed_ms * CV.MS_TO_KPH
|
||||||
|
speed_limit_display = speed_limit_ms * CV.MS_TO_KPH
|
||||||
|
cruise_display = cruise_speed_ms * CV.MS_TO_KPH
|
||||||
|
unit = "km/h"
|
||||||
|
else:
|
||||||
|
speed_display = speed_ms * CV.MS_TO_MPH
|
||||||
|
speed_limit_display = speed_limit_ms * CV.MS_TO_MPH
|
||||||
|
cruise_display = cruise_speed_ms * CV.MS_TO_MPH
|
||||||
|
unit = "mph"
|
||||||
|
|
||||||
|
speed_int = int(math.floor(speed_display))
|
||||||
|
speed_limit_int = int(round(speed_limit_display))
|
||||||
|
cruise_int = int(round(cruise_display))
|
||||||
|
|
||||||
|
self.prev_speed_limit = speed_limit_int
|
||||||
|
|
||||||
|
# Write display-ready values to params_memory
|
||||||
|
self.params_memory.put("ClearpilotHasSpeed", "1" if has_speed and speed_int > 0 else "0")
|
||||||
|
self.params_memory.put("ClearpilotSpeedDisplay", str(speed_int) if has_speed and speed_int > 0 else "")
|
||||||
|
self.params_memory.put("ClearpilotSpeedLimitDisplay", str(speed_limit_int) if speed_limit_int > 0 else "0")
|
||||||
|
self.params_memory.put("ClearpilotSpeedUnit", unit)
|
||||||
|
self.params_memory.put("ClearpilotIsMetric", "1" if is_metric else "0")
|
||||||
|
|
||||||
|
# Cruise warning logic
|
||||||
|
warning = ""
|
||||||
|
warning_speed = ""
|
||||||
|
cruise_engaged = cruise_active and not cruise_standstill
|
||||||
|
|
||||||
|
if speed_limit_int >= 20 and cruise_engaged and cruise_int > 0:
|
||||||
|
over_threshold = 10 if speed_limit_int >= 50 else 7
|
||||||
|
if cruise_int >= speed_limit_int + over_threshold:
|
||||||
|
warning = "over"
|
||||||
|
warning_speed = str(cruise_int)
|
||||||
|
elif cruise_int <= speed_limit_int - 5:
|
||||||
|
warning = "under"
|
||||||
|
warning_speed = str(cruise_int)
|
||||||
|
|
||||||
|
self.params_memory.put("ClearpilotCruiseWarning", warning)
|
||||||
|
self.params_memory.put("ClearpilotCruiseWarningSpeed", warning_speed)
|
||||||
|
|
||||||
|
# Ding logic: play when warning sign appears or speed limit changes while visible
|
||||||
|
should_ding = False
|
||||||
|
if warning:
|
||||||
|
if not self.prev_warning:
|
||||||
|
# Warning sign just appeared
|
||||||
|
should_ding = True
|
||||||
|
elif speed_limit_int != self.prev_warning_speed_limit:
|
||||||
|
# Speed limit changed while warning sign is visible
|
||||||
|
should_ding = True
|
||||||
|
|
||||||
|
if should_ding and now - self.last_ding_time >= 30:
|
||||||
|
self.params_memory.put("ClearpilotPlayDing", "1")
|
||||||
|
self.last_ding_time = now
|
||||||
|
|
||||||
|
self.prev_warning = warning
|
||||||
|
self.prev_warning_speed_limit = speed_limit_int if warning else 0
|
||||||
@@ -38,6 +38,7 @@ from openpilot.selfdrive.frogpilot.controls.lib.frogpilot_functions import CRUIS
|
|||||||
|
|
||||||
from openpilot.selfdrive.frogpilot.controls.lib.model_manager import RADARLESS_MODELS
|
from openpilot.selfdrive.frogpilot.controls.lib.model_manager import RADARLESS_MODELS
|
||||||
from openpilot.selfdrive.clearpilot.telemetry import tlog
|
from openpilot.selfdrive.clearpilot.telemetry import tlog
|
||||||
|
from openpilot.selfdrive.clearpilot.speed_logic import SpeedState
|
||||||
from openpilot.selfdrive.frogpilot.controls.lib.speed_limit_controller import SpeedLimitController
|
from openpilot.selfdrive.frogpilot.controls.lib.speed_limit_controller import SpeedLimitController
|
||||||
|
|
||||||
SOFT_DISABLE_TIME = 3 # seconds
|
SOFT_DISABLE_TIME = 3 # seconds
|
||||||
@@ -80,6 +81,10 @@ class Controls:
|
|||||||
self.params_memory.put_bool("CPTLkasButtonAction", False)
|
self.params_memory.put_bool("CPTLkasButtonAction", False)
|
||||||
self.params_memory.put_int("ScreenDisplayMode", 0)
|
self.params_memory.put_int("ScreenDisplayMode", 0)
|
||||||
|
|
||||||
|
# ClearPilot speed processing
|
||||||
|
self.speed_state = SpeedState()
|
||||||
|
self.speed_state_frame = 0
|
||||||
|
|
||||||
self.radarless_model = self.params.get("Model", encoding='utf-8') in RADARLESS_MODELS
|
self.radarless_model = self.params.get("Model", encoding='utf-8') in RADARLESS_MODELS
|
||||||
|
|
||||||
with car.CarParams.from_bytes(self.params.get("CarParams", block=True)) as msg:
|
with car.CarParams.from_bytes(self.params.get("CarParams", block=True)) as msg:
|
||||||
@@ -108,8 +113,9 @@ class Controls:
|
|||||||
self.sm = messaging.SubMaster(['deviceState', 'pandaStates', 'peripheralState', 'modelV2', 'liveCalibration',
|
self.sm = messaging.SubMaster(['deviceState', 'pandaStates', 'peripheralState', 'modelV2', 'liveCalibration',
|
||||||
'carOutput', 'driverMonitoringState', 'longitudinalPlan', 'liveLocationKalman',
|
'carOutput', 'driverMonitoringState', 'longitudinalPlan', 'liveLocationKalman',
|
||||||
'managerState', 'liveParameters', 'radarState', 'liveTorqueParameters',
|
'managerState', 'liveParameters', 'radarState', 'liveTorqueParameters',
|
||||||
'testJoystick', 'frogpilotPlan'] + self.camera_packets + self.sensor_packets,
|
'testJoystick', 'frogpilotPlan', 'gpsLocation'] + self.camera_packets + self.sensor_packets,
|
||||||
ignore_alive=ignore, ignore_avg_freq=ignore+['radarState', 'testJoystick'], ignore_valid=['testJoystick', ],
|
ignore_alive=ignore+['gpsLocation'], ignore_avg_freq=ignore+['radarState', 'testJoystick', 'gpsLocation'],
|
||||||
|
ignore_valid=['testJoystick', 'gpsLocation'],
|
||||||
frequency=int(1/DT_CTRL))
|
frequency=int(1/DT_CTRL))
|
||||||
|
|
||||||
self.joystick_mode = self.params.get_bool("JoystickDebugMode")
|
self.joystick_mode = self.params.get_bool("JoystickDebugMode")
|
||||||
@@ -198,6 +204,8 @@ class Controls:
|
|||||||
self.always_on_lateral_main = self.always_on_lateral and self.params.get_bool("AlwaysOnLateralMain")
|
self.always_on_lateral_main = self.always_on_lateral and self.params.get_bool("AlwaysOnLateralMain")
|
||||||
|
|
||||||
self.drive_added = False
|
self.drive_added = False
|
||||||
|
self.driving_gear = False
|
||||||
|
self.was_driving_gear = False
|
||||||
self.fcw_random_event_triggered = False
|
self.fcw_random_event_triggered = False
|
||||||
self.holiday_theme_alerted = False
|
self.holiday_theme_alerted = False
|
||||||
self.onroad_distance_pressed = False
|
self.onroad_distance_pressed = False
|
||||||
@@ -1257,6 +1265,12 @@ class Controls:
|
|||||||
self.events.add(EventName.clpDebug)
|
self.events.add(EventName.clpDebug)
|
||||||
|
|
||||||
def clearpilot_state_control(self, CC, CS):
|
def clearpilot_state_control(self, CC, CS):
|
||||||
|
# CLEARPILOT: auto-reset display when shifting into drive from screen-off
|
||||||
|
if self.driving_gear and not self.was_driving_gear:
|
||||||
|
if self.params_memory.get_int("ScreenDisplayMode") == 3:
|
||||||
|
self.params_memory.put_int("ScreenDisplayMode", 0)
|
||||||
|
self.was_driving_gear = self.driving_gear
|
||||||
|
|
||||||
if any(be.pressed and be.type == FrogPilotButtonType.lkas for be in CS.buttonEvents):
|
if any(be.pressed and be.type == FrogPilotButtonType.lkas for be in CS.buttonEvents):
|
||||||
current = self.params_memory.get_int("ScreenDisplayMode")
|
current = self.params_memory.get_int("ScreenDisplayMode")
|
||||||
|
|
||||||
@@ -1269,6 +1283,21 @@ class Controls:
|
|||||||
new_mode = 0 if current == 3 else 3
|
new_mode = 0 if current == 3 else 3
|
||||||
|
|
||||||
self.params_memory.put_int("ScreenDisplayMode", new_mode)
|
self.params_memory.put_int("ScreenDisplayMode", new_mode)
|
||||||
|
|
||||||
|
# ClearPilot speed processing (~2 Hz at 100 Hz loop)
|
||||||
|
self.speed_state_frame += 1
|
||||||
|
if self.speed_state_frame % 50 == 0:
|
||||||
|
gps = self.sm['gpsLocation']
|
||||||
|
has_gps = self.sm.valid['gpsLocation'] and gps.hasFix
|
||||||
|
speed_ms = gps.speed if has_gps else 0.0
|
||||||
|
speed_limit_ms = self.params_memory.get_float("CarSpeedLimit")
|
||||||
|
is_metric = (self.params_memory.get("CarIsMetric", encoding="utf-8") or "0") == "1"
|
||||||
|
cruise_speed_ms = CS.cruiseState.speed
|
||||||
|
cruise_active = CS.cruiseState.enabled
|
||||||
|
cruise_standstill = CS.cruiseState.standstill
|
||||||
|
self.speed_state.update(speed_ms, has_gps, speed_limit_ms, is_metric,
|
||||||
|
cruise_speed_ms, cruise_active, cruise_standstill)
|
||||||
|
|
||||||
return CC
|
return CC
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
|
|||||||
@@ -35,120 +35,19 @@ int ABGRToNV12(const uint8_t* src_abgr,
|
|||||||
int halfwidth = (width + 1) >> 1;
|
int halfwidth = (width + 1) >> 1;
|
||||||
void (*ABGRToUVRow)(const uint8_t* src_abgr0, int src_stride_abgr,
|
void (*ABGRToUVRow)(const uint8_t* src_abgr0, int src_stride_abgr,
|
||||||
uint8_t* dst_u, uint8_t* dst_v, int width) =
|
uint8_t* dst_u, uint8_t* dst_v, int width) =
|
||||||
ABGRToUVRow_C;
|
ABGRToUVRow_NEON;
|
||||||
void (*ABGRToYRow)(const uint8_t* src_abgr, uint8_t* dst_y, int width) =
|
void (*ABGRToYRow)(const uint8_t* src_abgr, uint8_t* dst_y, int width) =
|
||||||
ABGRToYRow_C;
|
ABGRToYRow_NEON;
|
||||||
void (*MergeUVRow_)(const uint8_t* src_u, const uint8_t* src_v, uint8_t* dst_uv, int width) = MergeUVRow_C;
|
void (*MergeUVRow_)(const uint8_t* src_u, const uint8_t* src_v, uint8_t* dst_uv, int width) = MergeUVRow_NEON;
|
||||||
if (!src_abgr || !dst_y || !dst_uv || width <= 0 || height == 0) {
|
if (!src_abgr || !dst_y || !dst_uv || width <= 0 || height == 0) {
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
if (height < 0) { // Negative height means invert the image.
|
if (height < 0) {
|
||||||
height = -height;
|
height = -height;
|
||||||
src_abgr = src_abgr + (height - 1) * src_stride_abgr;
|
src_abgr = src_abgr + (height - 1) * src_stride_abgr;
|
||||||
src_stride_abgr = -src_stride_abgr;
|
src_stride_abgr = -src_stride_abgr;
|
||||||
}
|
|
||||||
#if defined(HAS_ABGRTOYROW_SSSE3) && defined(HAS_ABGRTOUVROW_SSSE3)
|
|
||||||
if (TestCpuFlag(kCpuHasSSSE3)) {
|
|
||||||
ABGRToUVRow = ABGRToUVRow_Any_SSSE3;
|
|
||||||
ABGRToYRow = ABGRToYRow_Any_SSSE3;
|
|
||||||
if (IS_ALIGNED(width, 16)) {
|
|
||||||
ABGRToUVRow = ABGRToUVRow_SSSE3;
|
|
||||||
ABGRToYRow = ABGRToYRow_SSSE3;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
#if defined(HAS_ABGRTOYROW_AVX2) && defined(HAS_ABGRTOUVROW_AVX2)
|
|
||||||
if (TestCpuFlag(kCpuHasAVX2)) {
|
|
||||||
ABGRToUVRow = ABGRToUVRow_Any_AVX2;
|
|
||||||
ABGRToYRow = ABGRToYRow_Any_AVX2;
|
|
||||||
if (IS_ALIGNED(width, 32)) {
|
|
||||||
ABGRToUVRow = ABGRToUVRow_AVX2;
|
|
||||||
ABGRToYRow = ABGRToYRow_AVX2;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#if defined(HAS_ABGRTOYROW_NEON)
|
|
||||||
if (TestCpuFlag(kCpuHasNEON)) {
|
|
||||||
ABGRToYRow = ABGRToYRow_Any_NEON;
|
|
||||||
if (IS_ALIGNED(width, 8)) {
|
|
||||||
ABGRToYRow = ABGRToYRow_NEON;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#if defined(HAS_ABGRTOUVROW_NEON)
|
|
||||||
if (TestCpuFlag(kCpuHasNEON)) {
|
|
||||||
ABGRToUVRow = ABGRToUVRow_Any_NEON;
|
|
||||||
if (IS_ALIGNED(width, 16)) {
|
|
||||||
ABGRToUVRow = ABGRToUVRow_NEON;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#if defined(HAS_ABGRTOYROW_MMI) && defined(HAS_ABGRTOUVROW_MMI)
|
|
||||||
if (TestCpuFlag(kCpuHasMMI)) {
|
|
||||||
ABGRToYRow = ABGRToYRow_Any_MMI;
|
|
||||||
ABGRToUVRow = ABGRToUVRow_Any_MMI;
|
|
||||||
if (IS_ALIGNED(width, 8)) {
|
|
||||||
ABGRToYRow = ABGRToYRow_MMI;
|
|
||||||
}
|
|
||||||
if (IS_ALIGNED(width, 16)) {
|
|
||||||
ABGRToUVRow = ABGRToUVRow_MMI;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#if defined(HAS_ABGRTOYROW_MSA) && defined(HAS_ABGRTOUVROW_MSA)
|
|
||||||
if (TestCpuFlag(kCpuHasMSA)) {
|
|
||||||
ABGRToYRow = ABGRToYRow_Any_MSA;
|
|
||||||
ABGRToUVRow = ABGRToUVRow_Any_MSA;
|
|
||||||
if (IS_ALIGNED(width, 16)) {
|
|
||||||
ABGRToYRow = ABGRToYRow_MSA;
|
|
||||||
}
|
|
||||||
if (IS_ALIGNED(width, 32)) {
|
|
||||||
ABGRToUVRow = ABGRToUVRow_MSA;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#if defined(HAS_MERGEUVROW_SSE2)
|
|
||||||
if (TestCpuFlag(kCpuHasSSE2)) {
|
|
||||||
MergeUVRow_ = MergeUVRow_Any_SSE2;
|
|
||||||
if (IS_ALIGNED(halfwidth, 16)) {
|
|
||||||
MergeUVRow_ = MergeUVRow_SSE2;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#if defined(HAS_MERGEUVROW_AVX2)
|
|
||||||
if (TestCpuFlag(kCpuHasAVX2)) {
|
|
||||||
MergeUVRow_ = MergeUVRow_Any_AVX2;
|
|
||||||
if (IS_ALIGNED(halfwidth, 32)) {
|
|
||||||
MergeUVRow_ = MergeUVRow_AVX2;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#if defined(HAS_MERGEUVROW_NEON)
|
|
||||||
if (TestCpuFlag(kCpuHasNEON)) {
|
|
||||||
MergeUVRow_ = MergeUVRow_Any_NEON;
|
|
||||||
if (IS_ALIGNED(halfwidth, 16)) {
|
|
||||||
MergeUVRow_ = MergeUVRow_NEON;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#if defined(HAS_MERGEUVROW_MMI)
|
|
||||||
if (TestCpuFlag(kCpuHasMMI)) {
|
|
||||||
MergeUVRow_ = MergeUVRow_Any_MMI;
|
|
||||||
if (IS_ALIGNED(halfwidth, 8)) {
|
|
||||||
MergeUVRow_ = MergeUVRow_MMI;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#if defined(HAS_MERGEUVROW_MSA)
|
|
||||||
if (TestCpuFlag(kCpuHasMSA)) {
|
|
||||||
MergeUVRow_ = MergeUVRow_Any_MSA;
|
|
||||||
if (IS_ALIGNED(halfwidth, 16)) {
|
|
||||||
MergeUVRow_ = MergeUVRow_MSA;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
{
|
{
|
||||||
// Allocate a rows of uv.
|
|
||||||
align_buffer_64(row_u, ((halfwidth + 31) & ~31) * 2);
|
align_buffer_64(row_u, ((halfwidth + 31) & ~31) * 2);
|
||||||
uint8_t* row_v = row_u + ((halfwidth + 31) & ~31);
|
uint8_t* row_v = row_u + ((halfwidth + 31) & ~31);
|
||||||
|
|
||||||
@@ -182,9 +81,9 @@ extern ExitHandler do_exit;
|
|||||||
// ***** OMX callback functions *****
|
// ***** OMX callback functions *****
|
||||||
|
|
||||||
void OmxEncoder::wait_for_state(OMX_STATETYPE state_) {
|
void OmxEncoder::wait_for_state(OMX_STATETYPE state_) {
|
||||||
std::unique_lock lk(this->state_lock);
|
std::unique_lock lk(state_lock);
|
||||||
while (this->state != state_) {
|
while (state != state_) {
|
||||||
this->state_cv.wait(lk);
|
state_cv.wait(lk);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -236,270 +135,203 @@ static const char* omx_color_fomat_name(uint32_t format) __attribute__((unused))
|
|||||||
static const char* omx_color_fomat_name(uint32_t format) {
|
static const char* omx_color_fomat_name(uint32_t format) {
|
||||||
switch (format) {
|
switch (format) {
|
||||||
case OMX_COLOR_FormatUnused: return "OMX_COLOR_FormatUnused";
|
case OMX_COLOR_FormatUnused: return "OMX_COLOR_FormatUnused";
|
||||||
case OMX_COLOR_FormatMonochrome: return "OMX_COLOR_FormatMonochrome";
|
|
||||||
case OMX_COLOR_Format8bitRGB332: return "OMX_COLOR_Format8bitRGB332";
|
|
||||||
case OMX_COLOR_Format12bitRGB444: return "OMX_COLOR_Format12bitRGB444";
|
|
||||||
case OMX_COLOR_Format16bitARGB4444: return "OMX_COLOR_Format16bitARGB4444";
|
|
||||||
case OMX_COLOR_Format16bitARGB1555: return "OMX_COLOR_Format16bitARGB1555";
|
|
||||||
case OMX_COLOR_Format16bitRGB565: return "OMX_COLOR_Format16bitRGB565";
|
|
||||||
case OMX_COLOR_Format16bitBGR565: return "OMX_COLOR_Format16bitBGR565";
|
|
||||||
case OMX_COLOR_Format18bitRGB666: return "OMX_COLOR_Format18bitRGB666";
|
|
||||||
case OMX_COLOR_Format18bitARGB1665: return "OMX_COLOR_Format18bitARGB1665";
|
|
||||||
case OMX_COLOR_Format19bitARGB1666: return "OMX_COLOR_Format19bitARGB1666";
|
|
||||||
case OMX_COLOR_Format24bitRGB888: return "OMX_COLOR_Format24bitRGB888";
|
|
||||||
case OMX_COLOR_Format24bitBGR888: return "OMX_COLOR_Format24bitBGR888";
|
|
||||||
case OMX_COLOR_Format24bitARGB1887: return "OMX_COLOR_Format24bitARGB1887";
|
|
||||||
case OMX_COLOR_Format25bitARGB1888: return "OMX_COLOR_Format25bitARGB1888";
|
|
||||||
case OMX_COLOR_Format32bitBGRA8888: return "OMX_COLOR_Format32bitBGRA8888";
|
|
||||||
case OMX_COLOR_Format32bitARGB8888: return "OMX_COLOR_Format32bitARGB8888";
|
|
||||||
case OMX_COLOR_FormatYUV411Planar: return "OMX_COLOR_FormatYUV411Planar";
|
|
||||||
case OMX_COLOR_FormatYUV411PackedPlanar: return "OMX_COLOR_FormatYUV411PackedPlanar";
|
|
||||||
case OMX_COLOR_FormatYUV420Planar: return "OMX_COLOR_FormatYUV420Planar";
|
|
||||||
case OMX_COLOR_FormatYUV420PackedPlanar: return "OMX_COLOR_FormatYUV420PackedPlanar";
|
|
||||||
case OMX_COLOR_FormatYUV420SemiPlanar: return "OMX_COLOR_FormatYUV420SemiPlanar";
|
case OMX_COLOR_FormatYUV420SemiPlanar: return "OMX_COLOR_FormatYUV420SemiPlanar";
|
||||||
case OMX_COLOR_FormatYUV422Planar: return "OMX_COLOR_FormatYUV422Planar";
|
|
||||||
case OMX_COLOR_FormatYUV422PackedPlanar: return "OMX_COLOR_FormatYUV422PackedPlanar";
|
|
||||||
case OMX_COLOR_FormatYUV422SemiPlanar: return "OMX_COLOR_FormatYUV422SemiPlanar";
|
|
||||||
case OMX_COLOR_FormatYCbYCr: return "OMX_COLOR_FormatYCbYCr";
|
|
||||||
case OMX_COLOR_FormatYCrYCb: return "OMX_COLOR_FormatYCrYCb";
|
|
||||||
case OMX_COLOR_FormatCbYCrY: return "OMX_COLOR_FormatCbYCrY";
|
|
||||||
case OMX_COLOR_FormatCrYCbY: return "OMX_COLOR_FormatCrYCbY";
|
|
||||||
case OMX_COLOR_FormatYUV444Interleaved: return "OMX_COLOR_FormatYUV444Interleaved";
|
|
||||||
case OMX_COLOR_FormatRawBayer8bit: return "OMX_COLOR_FormatRawBayer8bit";
|
|
||||||
case OMX_COLOR_FormatRawBayer10bit: return "OMX_COLOR_FormatRawBayer10bit";
|
|
||||||
case OMX_COLOR_FormatRawBayer8bitcompressed: return "OMX_COLOR_FormatRawBayer8bitcompressed";
|
|
||||||
case OMX_COLOR_FormatL2: return "OMX_COLOR_FormatL2";
|
|
||||||
case OMX_COLOR_FormatL4: return "OMX_COLOR_FormatL4";
|
|
||||||
case OMX_COLOR_FormatL8: return "OMX_COLOR_FormatL8";
|
|
||||||
case OMX_COLOR_FormatL16: return "OMX_COLOR_FormatL16";
|
|
||||||
case OMX_COLOR_FormatL24: return "OMX_COLOR_FormatL24";
|
|
||||||
case OMX_COLOR_FormatL32: return "OMX_COLOR_FormatL32";
|
|
||||||
case OMX_COLOR_FormatYUV420PackedSemiPlanar: return "OMX_COLOR_FormatYUV420PackedSemiPlanar";
|
|
||||||
case OMX_COLOR_FormatYUV422PackedSemiPlanar: return "OMX_COLOR_FormatYUV422PackedSemiPlanar";
|
|
||||||
case OMX_COLOR_Format18BitBGR666: return "OMX_COLOR_Format18BitBGR666";
|
|
||||||
case OMX_COLOR_Format24BitARGB6666: return "OMX_COLOR_Format24BitARGB6666";
|
|
||||||
case OMX_COLOR_Format24BitABGR6666: return "OMX_COLOR_Format24BitABGR6666";
|
|
||||||
|
|
||||||
case OMX_COLOR_FormatAndroidOpaque: return "OMX_COLOR_FormatAndroidOpaque";
|
|
||||||
case OMX_TI_COLOR_FormatYUV420PackedSemiPlanar: return "OMX_TI_COLOR_FormatYUV420PackedSemiPlanar";
|
|
||||||
case OMX_QCOM_COLOR_FormatYVU420SemiPlanar: return "OMX_QCOM_COLOR_FormatYVU420SemiPlanar";
|
|
||||||
case OMX_QCOM_COLOR_FormatYUV420PackedSemiPlanar64x32Tile2m8ka: return "OMX_QCOM_COLOR_FormatYUV420PackedSemiPlanar64x32Tile2m8ka";
|
|
||||||
case OMX_SEC_COLOR_FormatNV12Tiled: return "OMX_SEC_COLOR_FormatNV12Tiled";
|
|
||||||
case OMX_QCOM_COLOR_FormatYUV420PackedSemiPlanar32m: return "OMX_QCOM_COLOR_FormatYUV420PackedSemiPlanar32m";
|
case OMX_QCOM_COLOR_FormatYUV420PackedSemiPlanar32m: return "OMX_QCOM_COLOR_FormatYUV420PackedSemiPlanar32m";
|
||||||
|
|
||||||
case QOMX_COLOR_FormatYVU420PackedSemiPlanar32m4ka: return "QOMX_COLOR_FormatYVU420PackedSemiPlanar32m4ka";
|
|
||||||
case QOMX_COLOR_FormatYUV420PackedSemiPlanar16m2ka: return "QOMX_COLOR_FormatYUV420PackedSemiPlanar16m2ka";
|
|
||||||
case QOMX_COLOR_FORMATYUV420PackedSemiPlanar32mMultiView: return "QOMX_COLOR_FORMATYUV420PackedSemiPlanar32mMultiView";
|
|
||||||
case QOMX_COLOR_FORMATYUV420PackedSemiPlanar32mCompressed: return "QOMX_COLOR_FORMATYUV420PackedSemiPlanar32mCompressed";
|
|
||||||
case QOMX_COLOR_Format32bitRGBA8888: return "QOMX_COLOR_Format32bitRGBA8888";
|
case QOMX_COLOR_Format32bitRGBA8888: return "QOMX_COLOR_Format32bitRGBA8888";
|
||||||
case QOMX_COLOR_Format32bitRGBA8888Compressed: return "QOMX_COLOR_Format32bitRGBA8888Compressed";
|
default: return "unkn";
|
||||||
|
|
||||||
default:
|
|
||||||
return "unkn";
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// ***** encoder functions *****
|
// ***** encoder functions *****
|
||||||
|
|
||||||
OmxEncoder::OmxEncoder(const char* path, int width, int height, int fps, int bitrate, bool h265, bool downscale) {
|
OmxEncoder::OmxEncoder(const char* path, int width, int height, int fps, int bitrate) {
|
||||||
this->path = path;
|
this->path = path;
|
||||||
this->width = width;
|
this->width = width;
|
||||||
this->height = height;
|
this->height = height;
|
||||||
this->fps = fps;
|
this->fps = fps;
|
||||||
this->remuxing = !h265;
|
|
||||||
|
|
||||||
this->downscale = downscale;
|
OMX_ERRORTYPE err = OMX_Init();
|
||||||
if (this->downscale) {
|
if (err != OMX_ErrorNone) {
|
||||||
this->y_ptr2 = (uint8_t *)malloc(this->width*this->height);
|
LOGE("OMX_Init failed: %x", err);
|
||||||
this->u_ptr2 = (uint8_t *)malloc(this->width*this->height/4);
|
return;
|
||||||
this->v_ptr2 = (uint8_t *)malloc(this->width*this->height/4);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
auto component = (OMX_STRING)(h265 ? "OMX.qcom.video.encoder.hevc" : "OMX.qcom.video.encoder.avc");
|
OMX_STRING component = (OMX_STRING)("OMX.qcom.video.encoder.avc");
|
||||||
int err = OMX_GetHandle(&this->handle, component, this, &omx_callbacks);
|
err = OMX_GetHandle(&handle, component, this, &omx_callbacks);
|
||||||
if (err != OMX_ErrorNone) {
|
if (err != OMX_ErrorNone) {
|
||||||
LOGE("error getting codec: %x", err);
|
LOGE("Error getting codec: %x", err);
|
||||||
|
OMX_Deinit();
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// setup input port
|
// setup input port
|
||||||
|
|
||||||
OMX_PARAM_PORTDEFINITIONTYPE in_port = {0};
|
OMX_PARAM_PORTDEFINITIONTYPE in_port = {0};
|
||||||
in_port.nSize = sizeof(in_port);
|
in_port.nSize = sizeof(in_port);
|
||||||
in_port.nPortIndex = (OMX_U32) PORT_INDEX_IN;
|
in_port.nPortIndex = (OMX_U32) PORT_INDEX_IN;
|
||||||
OMX_CHECK(OMX_GetParameter(this->handle, OMX_IndexParamPortDefinition, (OMX_PTR) &in_port));
|
OMX_CHECK(OMX_GetParameter(handle, OMX_IndexParamPortDefinition, (OMX_PTR) &in_port));
|
||||||
|
|
||||||
in_port.format.video.nFrameWidth = this->width;
|
in_port.format.video.nFrameWidth = width;
|
||||||
in_port.format.video.nFrameHeight = this->height;
|
in_port.format.video.nFrameHeight = height;
|
||||||
in_port.format.video.nStride = VENUS_Y_STRIDE(COLOR_FMT_NV12, this->width);
|
in_port.format.video.nStride = VENUS_Y_STRIDE(COLOR_FMT_NV12, width);
|
||||||
in_port.format.video.nSliceHeight = this->height;
|
in_port.format.video.nSliceHeight = height;
|
||||||
in_port.nBufferSize = VENUS_BUFFER_SIZE(COLOR_FMT_NV12, this->width, this->height);
|
in_port.nBufferSize = VENUS_BUFFER_SIZE(COLOR_FMT_NV12, width, height);
|
||||||
in_port.format.video.xFramerate = (this->fps * 65536);
|
in_port.format.video.xFramerate = (fps * 65536);
|
||||||
in_port.format.video.eCompressionFormat = OMX_VIDEO_CodingUnused;
|
in_port.format.video.eCompressionFormat = OMX_VIDEO_CodingUnused;
|
||||||
in_port.format.video.eColorFormat = (OMX_COLOR_FORMATTYPE)QOMX_COLOR_FORMATYUV420PackedSemiPlanar32m;
|
in_port.format.video.eColorFormat = (OMX_COLOR_FORMATTYPE)QOMX_COLOR_FORMATYUV420PackedSemiPlanar32m;
|
||||||
|
|
||||||
OMX_CHECK(OMX_SetParameter(this->handle, OMX_IndexParamPortDefinition, (OMX_PTR) &in_port));
|
OMX_CHECK(OMX_SetParameter(handle, OMX_IndexParamPortDefinition, (OMX_PTR) &in_port));
|
||||||
OMX_CHECK(OMX_GetParameter(this->handle, OMX_IndexParamPortDefinition, (OMX_PTR) &in_port));
|
OMX_CHECK(OMX_GetParameter(handle, OMX_IndexParamPortDefinition, (OMX_PTR) &in_port));
|
||||||
this->in_buf_headers.resize(in_port.nBufferCountActual);
|
in_buf_headers.resize(in_port.nBufferCountActual);
|
||||||
|
|
||||||
// setup output port
|
// setup output port
|
||||||
|
OMX_PARAM_PORTDEFINITIONTYPE out_port;
|
||||||
OMX_PARAM_PORTDEFINITIONTYPE out_port = {0};
|
memset(&out_port, 0, sizeof(OMX_PARAM_PORTDEFINITIONTYPE));
|
||||||
out_port.nSize = sizeof(out_port);
|
out_port.nSize = sizeof(out_port);
|
||||||
|
out_port.nVersion.s.nVersionMajor = 1;
|
||||||
|
out_port.nVersion.s.nVersionMinor = 0;
|
||||||
|
out_port.nVersion.s.nRevision = 0;
|
||||||
|
out_port.nVersion.s.nStep = 0;
|
||||||
out_port.nPortIndex = (OMX_U32) PORT_INDEX_OUT;
|
out_port.nPortIndex = (OMX_U32) PORT_INDEX_OUT;
|
||||||
OMX_CHECK(OMX_GetParameter(this->handle, OMX_IndexParamPortDefinition, (OMX_PTR)&out_port));
|
|
||||||
out_port.format.video.nFrameWidth = this->width;
|
OMX_ERRORTYPE error = OMX_GetParameter(handle, OMX_IndexParamPortDefinition, (OMX_PTR)&out_port);
|
||||||
out_port.format.video.nFrameHeight = this->height;
|
if (error != OMX_ErrorNone) {
|
||||||
|
LOGE("Error getting output port parameters: 0x%08x", error);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
out_port.format.video.nFrameWidth = width;
|
||||||
|
out_port.format.video.nFrameHeight = height;
|
||||||
out_port.format.video.xFramerate = 0;
|
out_port.format.video.xFramerate = 0;
|
||||||
out_port.format.video.nBitrate = bitrate;
|
out_port.format.video.nBitrate = bitrate;
|
||||||
if (h265) {
|
out_port.format.video.eCompressionFormat = OMX_VIDEO_CodingAVC;
|
||||||
out_port.format.video.eCompressionFormat = OMX_VIDEO_CodingHEVC;
|
|
||||||
} else {
|
|
||||||
out_port.format.video.eCompressionFormat = OMX_VIDEO_CodingAVC;
|
|
||||||
}
|
|
||||||
out_port.format.video.eColorFormat = OMX_COLOR_FormatUnused;
|
out_port.format.video.eColorFormat = OMX_COLOR_FormatUnused;
|
||||||
|
|
||||||
OMX_CHECK(OMX_SetParameter(this->handle, OMX_IndexParamPortDefinition, (OMX_PTR) &out_port));
|
error = OMX_SetParameter(handle, OMX_IndexParamPortDefinition, (OMX_PTR) &out_port);
|
||||||
|
if (error != OMX_ErrorNone) {
|
||||||
|
LOGE("Error setting output port parameters: 0x%08x", error);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
OMX_CHECK(OMX_GetParameter(this->handle, OMX_IndexParamPortDefinition, (OMX_PTR) &out_port));
|
error = OMX_GetParameter(handle, OMX_IndexParamPortDefinition, (OMX_PTR) &out_port);
|
||||||
this->out_buf_headers.resize(out_port.nBufferCountActual);
|
if (error != OMX_ErrorNone) {
|
||||||
|
LOGE("Error getting updated output port parameters: 0x%08x", error);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
out_buf_headers.resize(out_port.nBufferCountActual);
|
||||||
|
|
||||||
OMX_VIDEO_PARAM_BITRATETYPE bitrate_type = {0};
|
OMX_VIDEO_PARAM_BITRATETYPE bitrate_type = {0};
|
||||||
bitrate_type.nSize = sizeof(bitrate_type);
|
bitrate_type.nSize = sizeof(bitrate_type);
|
||||||
bitrate_type.nPortIndex = (OMX_U32) PORT_INDEX_OUT;
|
bitrate_type.nPortIndex = (OMX_U32) PORT_INDEX_OUT;
|
||||||
OMX_CHECK(OMX_GetParameter(this->handle, OMX_IndexParamVideoBitrate, (OMX_PTR) &bitrate_type));
|
OMX_CHECK(OMX_GetParameter(handle, OMX_IndexParamVideoBitrate, (OMX_PTR) &bitrate_type));
|
||||||
bitrate_type.eControlRate = OMX_Video_ControlRateVariable;
|
bitrate_type.eControlRate = OMX_Video_ControlRateVariable;
|
||||||
bitrate_type.nTargetBitrate = bitrate;
|
bitrate_type.nTargetBitrate = bitrate;
|
||||||
|
|
||||||
OMX_CHECK(OMX_SetParameter(this->handle, OMX_IndexParamVideoBitrate, (OMX_PTR) &bitrate_type));
|
OMX_CHECK(OMX_SetParameter(handle, OMX_IndexParamVideoBitrate, (OMX_PTR) &bitrate_type));
|
||||||
|
|
||||||
if (h265) {
|
// setup h264
|
||||||
// setup HEVC
|
OMX_VIDEO_PARAM_AVCTYPE avc = {0};
|
||||||
#ifndef QCOM2
|
avc.nSize = sizeof(avc);
|
||||||
OMX_VIDEO_PARAM_HEVCTYPE hevc_type = {0};
|
avc.nPortIndex = (OMX_U32) PORT_INDEX_OUT;
|
||||||
OMX_INDEXTYPE index_type = (OMX_INDEXTYPE) OMX_IndexParamVideoHevc;
|
OMX_CHECK(OMX_GetParameter(handle, OMX_IndexParamVideoAvc, &avc));
|
||||||
#else
|
|
||||||
OMX_VIDEO_PARAM_PROFILELEVELTYPE hevc_type = {0};
|
|
||||||
OMX_INDEXTYPE index_type = OMX_IndexParamVideoProfileLevelCurrent;
|
|
||||||
#endif
|
|
||||||
hevc_type.nSize = sizeof(hevc_type);
|
|
||||||
hevc_type.nPortIndex = (OMX_U32) PORT_INDEX_OUT;
|
|
||||||
OMX_CHECK(OMX_GetParameter(this->handle, index_type, (OMX_PTR) &hevc_type));
|
|
||||||
|
|
||||||
hevc_type.eProfile = OMX_VIDEO_HEVCProfileMain;
|
avc.nBFrames = 0;
|
||||||
hevc_type.eLevel = OMX_VIDEO_HEVCHighTierLevel5;
|
avc.nPFrames = 15;
|
||||||
|
|
||||||
OMX_CHECK(OMX_SetParameter(this->handle, index_type, (OMX_PTR) &hevc_type));
|
avc.eProfile = OMX_VIDEO_AVCProfileHigh;
|
||||||
} else {
|
avc.eLevel = OMX_VIDEO_AVCLevel31;
|
||||||
// setup h264
|
|
||||||
OMX_VIDEO_PARAM_AVCTYPE avc = { 0 };
|
|
||||||
avc.nSize = sizeof(avc);
|
|
||||||
avc.nPortIndex = (OMX_U32) PORT_INDEX_OUT;
|
|
||||||
OMX_CHECK(OMX_GetParameter(this->handle, OMX_IndexParamVideoAvc, &avc));
|
|
||||||
|
|
||||||
avc.nBFrames = 0;
|
avc.nAllowedPictureTypes |= OMX_VIDEO_PictureTypeB;
|
||||||
avc.nPFrames = 15;
|
avc.eLoopFilterMode = OMX_VIDEO_AVCLoopFilterEnable;
|
||||||
|
|
||||||
avc.eProfile = OMX_VIDEO_AVCProfileHigh;
|
avc.nRefFrames = 1;
|
||||||
avc.eLevel = OMX_VIDEO_AVCLevel31;
|
avc.bUseHadamard = OMX_TRUE;
|
||||||
|
avc.bEntropyCodingCABAC = OMX_TRUE;
|
||||||
|
avc.bWeightedPPrediction = OMX_TRUE;
|
||||||
|
avc.bconstIpred = OMX_TRUE;
|
||||||
|
|
||||||
avc.nAllowedPictureTypes |= OMX_VIDEO_PictureTypeB;
|
OMX_CHECK(OMX_SetParameter(handle, OMX_IndexParamVideoAvc, &avc));
|
||||||
avc.eLoopFilterMode = OMX_VIDEO_AVCLoopFilterEnable;
|
|
||||||
|
|
||||||
avc.nRefFrames = 1;
|
OMX_CHECK(OMX_SendCommand(handle, OMX_CommandStateSet, OMX_StateIdle, NULL));
|
||||||
avc.bUseHadamard = OMX_TRUE;
|
|
||||||
avc.bEntropyCodingCABAC = OMX_TRUE;
|
|
||||||
avc.bWeightedPPrediction = OMX_TRUE;
|
|
||||||
avc.bconstIpred = OMX_TRUE;
|
|
||||||
|
|
||||||
OMX_CHECK(OMX_SetParameter(this->handle, OMX_IndexParamVideoAvc, &avc));
|
for (OMX_BUFFERHEADERTYPE* &buf : in_buf_headers) {
|
||||||
|
OMX_CHECK(OMX_AllocateBuffer(handle, &buf, PORT_INDEX_IN, this, in_port.nBufferSize));
|
||||||
}
|
}
|
||||||
|
|
||||||
OMX_CHECK(OMX_SendCommand(this->handle, OMX_CommandStateSet, OMX_StateIdle, NULL));
|
for (OMX_BUFFERHEADERTYPE* &buf : out_buf_headers) {
|
||||||
|
OMX_CHECK(OMX_AllocateBuffer(handle, &buf, PORT_INDEX_OUT, this, out_port.nBufferSize));
|
||||||
for (auto &buf : this->in_buf_headers) {
|
|
||||||
OMX_CHECK(OMX_AllocateBuffer(this->handle, &buf, PORT_INDEX_IN, this,
|
|
||||||
in_port.nBufferSize));
|
|
||||||
}
|
|
||||||
|
|
||||||
for (auto &buf : this->out_buf_headers) {
|
|
||||||
OMX_CHECK(OMX_AllocateBuffer(this->handle, &buf, PORT_INDEX_OUT, this,
|
|
||||||
out_port.nBufferSize));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
wait_for_state(OMX_StateIdle);
|
wait_for_state(OMX_StateIdle);
|
||||||
|
|
||||||
OMX_CHECK(OMX_SendCommand(this->handle, OMX_CommandStateSet, OMX_StateExecuting, NULL));
|
OMX_CHECK(OMX_SendCommand(handle, OMX_CommandStateSet, OMX_StateExecuting, NULL));
|
||||||
|
|
||||||
wait_for_state(OMX_StateExecuting);
|
wait_for_state(OMX_StateExecuting);
|
||||||
|
|
||||||
// give omx all the output buffers
|
// give omx all the output buffers
|
||||||
for (auto &buf : this->out_buf_headers) {
|
for (OMX_BUFFERHEADERTYPE* &buf : out_buf_headers) {
|
||||||
OMX_CHECK(OMX_FillThisBuffer(this->handle, buf));
|
OMX_CHECK(OMX_FillThisBuffer(handle, buf));
|
||||||
}
|
}
|
||||||
|
|
||||||
// fill the input free queue
|
// fill the input free queue
|
||||||
for (auto &buf : this->in_buf_headers) {
|
for (OMX_BUFFERHEADERTYPE* &buf : in_buf_headers) {
|
||||||
this->free_in.push(buf);
|
free_in.push(buf);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void OmxEncoder::handle_out_buf(OmxEncoder *e, OMX_BUFFERHEADERTYPE *out_buf) {
|
void OmxEncoder::handle_out_buf(OmxEncoder *encoder, OMX_BUFFERHEADERTYPE *out_buf) {
|
||||||
int err;
|
int err;
|
||||||
uint8_t *buf_data = out_buf->pBuffer + out_buf->nOffset;
|
uint8_t *buf_data = out_buf->pBuffer + out_buf->nOffset;
|
||||||
|
|
||||||
if (out_buf->nFlags & OMX_BUFFERFLAG_CODECCONFIG) {
|
if (out_buf->nFlags & OMX_BUFFERFLAG_CODECCONFIG) {
|
||||||
if (e->codec_config_len < out_buf->nFilledLen) {
|
if (encoder->codec_config_len < out_buf->nFilledLen) {
|
||||||
e->codec_config = (uint8_t *)realloc(e->codec_config, out_buf->nFilledLen);
|
encoder->codec_config = (uint8_t *)realloc(encoder->codec_config, out_buf->nFilledLen);
|
||||||
}
|
}
|
||||||
e->codec_config_len = out_buf->nFilledLen;
|
encoder->codec_config_len = out_buf->nFilledLen;
|
||||||
memcpy(e->codec_config, buf_data, out_buf->nFilledLen);
|
memcpy(encoder->codec_config, buf_data, out_buf->nFilledLen);
|
||||||
#ifdef QCOM2
|
#ifdef QCOM2
|
||||||
out_buf->nTimeStamp = 0;
|
out_buf->nTimeStamp = 0;
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
if (e->of) {
|
if (encoder->of) {
|
||||||
fwrite(buf_data, out_buf->nFilledLen, 1, e->of);
|
fwrite(buf_data, out_buf->nFilledLen, 1, encoder->of);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (e->remuxing) {
|
if (!encoder->wrote_codec_config && encoder->codec_config_len > 0) {
|
||||||
if (!e->wrote_codec_config && e->codec_config_len > 0) {
|
// extradata will be freed by av_free() in avcodec_free_context()
|
||||||
// extradata will be freed by av_free() in avcodec_free_context()
|
encoder->out_stream->codecpar->extradata = (uint8_t*)av_mallocz(encoder->codec_config_len + AV_INPUT_BUFFER_PADDING_SIZE);
|
||||||
e->codec_ctx->extradata = (uint8_t*)av_mallocz(e->codec_config_len + AV_INPUT_BUFFER_PADDING_SIZE);
|
encoder->out_stream->codecpar->extradata_size = encoder->codec_config_len;
|
||||||
e->codec_ctx->extradata_size = e->codec_config_len;
|
memcpy(encoder->out_stream->codecpar->extradata, encoder->codec_config, encoder->codec_config_len);
|
||||||
memcpy(e->codec_ctx->extradata, e->codec_config, e->codec_config_len);
|
|
||||||
|
|
||||||
err = avcodec_parameters_from_context(e->out_stream->codecpar, e->codec_ctx);
|
err = avformat_write_header(encoder->ofmt_ctx, NULL);
|
||||||
assert(err >= 0);
|
assert(err >= 0);
|
||||||
err = avformat_write_header(e->ofmt_ctx, NULL);
|
|
||||||
assert(err >= 0);
|
|
||||||
|
|
||||||
e->wrote_codec_config = true;
|
encoder->wrote_codec_config = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (out_buf->nTimeStamp > 0) {
|
||||||
|
// input timestamps are in microseconds
|
||||||
|
AVRational in_timebase = {1, 1000000};
|
||||||
|
|
||||||
|
AVPacket pkt;
|
||||||
|
av_init_packet(&pkt);
|
||||||
|
pkt.data = buf_data;
|
||||||
|
pkt.size = out_buf->nFilledLen;
|
||||||
|
|
||||||
|
enum AVRounding rnd = static_cast<enum AVRounding>(AV_ROUND_NEAR_INF|AV_ROUND_PASS_MINMAX);
|
||||||
|
pkt.pts = pkt.dts = av_rescale_q_rnd(out_buf->nTimeStamp, in_timebase, encoder->out_stream->time_base, rnd);
|
||||||
|
pkt.duration = av_rescale_q(1, AVRational{1, encoder->fps}, encoder->out_stream->time_base);
|
||||||
|
|
||||||
|
if (out_buf->nFlags & OMX_BUFFERFLAG_SYNCFRAME) {
|
||||||
|
pkt.flags |= AV_PKT_FLAG_KEY;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (out_buf->nTimeStamp > 0) {
|
err = av_write_frame(encoder->ofmt_ctx, &pkt);
|
||||||
// input timestamps are in microseconds
|
if (err < 0) { LOGW("ts encoder write issue"); }
|
||||||
AVRational in_timebase = {1, 1000000};
|
|
||||||
|
|
||||||
AVPacket pkt;
|
av_packet_unref(&pkt);
|
||||||
av_init_packet(&pkt);
|
|
||||||
pkt.data = buf_data;
|
|
||||||
pkt.size = out_buf->nFilledLen;
|
|
||||||
|
|
||||||
enum AVRounding rnd = static_cast<enum AVRounding>(AV_ROUND_NEAR_INF|AV_ROUND_PASS_MINMAX);
|
|
||||||
pkt.pts = pkt.dts = av_rescale_q_rnd(out_buf->nTimeStamp, in_timebase, e->ofmt_ctx->streams[0]->time_base, rnd);
|
|
||||||
pkt.duration = av_rescale_q(50*1000, in_timebase, e->ofmt_ctx->streams[0]->time_base);
|
|
||||||
|
|
||||||
if (out_buf->nFlags & OMX_BUFFERFLAG_SYNCFRAME) {
|
|
||||||
pkt.flags |= AV_PKT_FLAG_KEY;
|
|
||||||
}
|
|
||||||
|
|
||||||
err = av_write_frame(e->ofmt_ctx, &pkt);
|
|
||||||
if (err < 0) { LOGW("ts encoder write issue"); }
|
|
||||||
|
|
||||||
av_free_packet(&pkt);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// give omx back the buffer
|
// give omx back the buffer
|
||||||
@@ -508,59 +340,53 @@ void OmxEncoder::handle_out_buf(OmxEncoder *e, OMX_BUFFERHEADERTYPE *out_buf) {
|
|||||||
out_buf->nTimeStamp = 0;
|
out_buf->nTimeStamp = 0;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
OMX_CHECK(OMX_FillThisBuffer(e->handle, out_buf));
|
OMX_CHECK(OMX_FillThisBuffer(encoder->handle, out_buf));
|
||||||
}
|
}
|
||||||
|
|
||||||
int OmxEncoder::encode_frame_rgba(const uint8_t *ptr, int in_width, int in_height, uint64_t ts) {
|
int OmxEncoder::encode_frame_rgba(const uint8_t *ptr, int in_width, int in_height, uint64_t ts) {
|
||||||
int err;
|
if (!is_open) {
|
||||||
if (!this->is_open) {
|
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// this sometimes freezes... put it outside the encoder lock so we can still trigger rotates...
|
|
||||||
// THIS IS A REALLY BAD IDEA, but apparently the race has to happen 30 times to trigger this
|
|
||||||
OMX_BUFFERHEADERTYPE* in_buf = nullptr;
|
OMX_BUFFERHEADERTYPE* in_buf = nullptr;
|
||||||
while (!this->free_in.try_pop(in_buf, 20)) {
|
while (!free_in.try_pop(in_buf, 20)) {
|
||||||
if (do_exit) {
|
if (do_exit) {
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int ret = this->counter;
|
int ret = counter;
|
||||||
|
|
||||||
uint8_t *in_buf_ptr = in_buf->pBuffer;
|
uint8_t *in_buf_ptr = in_buf->pBuffer;
|
||||||
|
|
||||||
uint8_t *in_y_ptr = in_buf_ptr;
|
uint8_t *in_y_ptr = in_buf_ptr;
|
||||||
int in_y_stride = VENUS_Y_STRIDE(COLOR_FMT_NV12, this->width);
|
int in_y_stride = VENUS_Y_STRIDE(COLOR_FMT_NV12, width);
|
||||||
int in_uv_stride = VENUS_UV_STRIDE(COLOR_FMT_NV12, this->width);
|
int in_uv_stride = VENUS_UV_STRIDE(COLOR_FMT_NV12, width);
|
||||||
uint8_t *in_uv_ptr = in_buf_ptr + (in_y_stride * VENUS_Y_SCANLINES(COLOR_FMT_NV12, this->height));
|
uint8_t *in_uv_ptr = in_buf_ptr + (in_y_stride * VENUS_Y_SCANLINES(COLOR_FMT_NV12, height));
|
||||||
|
|
||||||
err = ABGRToNV12(ptr, this->width*4,
|
int err = ABGRToNV12(ptr, width * 4, in_y_ptr, in_y_stride, in_uv_ptr, in_uv_stride, width, height);
|
||||||
in_y_ptr, in_y_stride,
|
|
||||||
in_uv_ptr, in_uv_stride,
|
|
||||||
this->width, this->height);
|
|
||||||
assert(err == 0);
|
assert(err == 0);
|
||||||
|
|
||||||
in_buf->nFilledLen = VENUS_BUFFER_SIZE(COLOR_FMT_NV12, this->width, this->height);
|
in_buf->nFilledLen = VENUS_BUFFER_SIZE(COLOR_FMT_NV12, width, height);
|
||||||
in_buf->nFlags = OMX_BUFFERFLAG_ENDOFFRAME;
|
in_buf->nFlags = OMX_BUFFERFLAG_ENDOFFRAME;
|
||||||
in_buf->nOffset = 0;
|
in_buf->nOffset = 0;
|
||||||
in_buf->nTimeStamp = ts/1000LL; // OMX_TICKS, in microseconds
|
in_buf->nTimeStamp = ts / 1000LL; // OMX_TICKS, in microseconds
|
||||||
this->last_t = in_buf->nTimeStamp;
|
last_t = in_buf->nTimeStamp;
|
||||||
|
|
||||||
OMX_CHECK(OMX_EmptyThisBuffer(this->handle, in_buf));
|
OMX_CHECK(OMX_EmptyThisBuffer(handle, in_buf));
|
||||||
|
|
||||||
// pump output
|
// pump output
|
||||||
while (true) {
|
while (true) {
|
||||||
OMX_BUFFERHEADERTYPE *out_buf;
|
OMX_BUFFERHEADERTYPE *out_buf;
|
||||||
if (!this->done_out.try_pop(out_buf)) {
|
if (!done_out.try_pop(out_buf)) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
handle_out_buf(this, out_buf);
|
handle_out_buf(this, out_buf);
|
||||||
}
|
}
|
||||||
|
|
||||||
this->dirty = true;
|
dirty = true;
|
||||||
|
|
||||||
this->counter++;
|
counter++;
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
@@ -568,24 +394,24 @@ int OmxEncoder::encode_frame_rgba(const uint8_t *ptr, int in_width, int in_heigh
|
|||||||
// CLEARPILOT: encode raw NV12 frames directly (no RGBA conversion needed)
|
// CLEARPILOT: encode raw NV12 frames directly (no RGBA conversion needed)
|
||||||
int OmxEncoder::encode_frame_nv12(const uint8_t *y_ptr, int y_stride, const uint8_t *uv_ptr, int uv_stride,
|
int OmxEncoder::encode_frame_nv12(const uint8_t *y_ptr, int y_stride, const uint8_t *uv_ptr, int uv_stride,
|
||||||
int in_width, int in_height, uint64_t ts) {
|
int in_width, int in_height, uint64_t ts) {
|
||||||
if (!this->is_open) {
|
if (!is_open) {
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
OMX_BUFFERHEADERTYPE* in_buf = nullptr;
|
OMX_BUFFERHEADERTYPE* in_buf = nullptr;
|
||||||
while (!this->free_in.try_pop(in_buf, 20)) {
|
while (!free_in.try_pop(in_buf, 20)) {
|
||||||
if (do_exit) {
|
if (do_exit) {
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int ret = this->counter;
|
int ret = counter;
|
||||||
|
|
||||||
uint8_t *in_buf_ptr = in_buf->pBuffer;
|
uint8_t *in_buf_ptr = in_buf->pBuffer;
|
||||||
int venus_y_stride = VENUS_Y_STRIDE(COLOR_FMT_NV12, this->width);
|
int venus_y_stride = VENUS_Y_STRIDE(COLOR_FMT_NV12, width);
|
||||||
int venus_uv_stride = VENUS_UV_STRIDE(COLOR_FMT_NV12, this->width);
|
int venus_uv_stride = VENUS_UV_STRIDE(COLOR_FMT_NV12, width);
|
||||||
uint8_t *dst_y = in_buf_ptr;
|
uint8_t *dst_y = in_buf_ptr;
|
||||||
uint8_t *dst_uv = in_buf_ptr + (venus_y_stride * VENUS_Y_SCANLINES(COLOR_FMT_NV12, this->height));
|
uint8_t *dst_uv = in_buf_ptr + (venus_y_stride * VENUS_Y_SCANLINES(COLOR_FMT_NV12, height));
|
||||||
|
|
||||||
// Copy Y plane row by row (source stride may differ from VENUS stride)
|
// Copy Y plane row by row (source stride may differ from VENUS stride)
|
||||||
for (int row = 0; row < in_height; row++) {
|
for (int row = 0; row < in_height; row++) {
|
||||||
@@ -597,99 +423,103 @@ int OmxEncoder::encode_frame_nv12(const uint8_t *y_ptr, int y_stride, const uint
|
|||||||
memcpy(dst_uv + row * venus_uv_stride, uv_ptr + row * uv_stride, in_width);
|
memcpy(dst_uv + row * venus_uv_stride, uv_ptr + row * uv_stride, in_width);
|
||||||
}
|
}
|
||||||
|
|
||||||
in_buf->nFilledLen = VENUS_BUFFER_SIZE(COLOR_FMT_NV12, this->width, this->height);
|
in_buf->nFilledLen = VENUS_BUFFER_SIZE(COLOR_FMT_NV12, width, height);
|
||||||
in_buf->nFlags = OMX_BUFFERFLAG_ENDOFFRAME;
|
in_buf->nFlags = OMX_BUFFERFLAG_ENDOFFRAME;
|
||||||
in_buf->nOffset = 0;
|
in_buf->nOffset = 0;
|
||||||
in_buf->nTimeStamp = ts / 1000LL;
|
in_buf->nTimeStamp = ts / 1000LL;
|
||||||
this->last_t = in_buf->nTimeStamp;
|
last_t = in_buf->nTimeStamp;
|
||||||
|
|
||||||
OMX_CHECK(OMX_EmptyThisBuffer(this->handle, in_buf));
|
OMX_CHECK(OMX_EmptyThisBuffer(handle, in_buf));
|
||||||
|
|
||||||
while (true) {
|
while (true) {
|
||||||
OMX_BUFFERHEADERTYPE *out_buf;
|
OMX_BUFFERHEADERTYPE *out_buf;
|
||||||
if (!this->done_out.try_pop(out_buf)) {
|
if (!done_out.try_pop(out_buf)) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
handle_out_buf(this, out_buf);
|
handle_out_buf(this, out_buf);
|
||||||
}
|
}
|
||||||
|
|
||||||
this->dirty = true;
|
dirty = true;
|
||||||
this->counter++;
|
counter++;
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
void OmxEncoder::encoder_open(const char* filename) {
|
void OmxEncoder::encoder_open(const char* filename) {
|
||||||
int err;
|
if (!filename || strlen(filename) == 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (strlen(filename) + path.size() + 2 > sizeof(vid_path)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
struct stat st = {0};
|
struct stat st = {0};
|
||||||
if (stat(this->path.c_str(), &st) == -1) {
|
if (stat(path.c_str(), &st) == -1) {
|
||||||
mkdir(this->path.c_str(), 0755);
|
if (mkdir(path.c_str(), 0755) == -1) {
|
||||||
}
|
return;
|
||||||
|
|
||||||
snprintf(this->vid_path, sizeof(this->vid_path), "%s/%s", this->path.c_str(), filename);
|
|
||||||
printf("encoder_open %s remuxing:%d\n", this->vid_path, this->remuxing);
|
|
||||||
|
|
||||||
if (this->remuxing) {
|
|
||||||
avformat_alloc_output_context2(&this->ofmt_ctx, NULL, NULL, this->vid_path);
|
|
||||||
assert(this->ofmt_ctx);
|
|
||||||
|
|
||||||
this->out_stream = avformat_new_stream(this->ofmt_ctx, NULL);
|
|
||||||
assert(this->out_stream);
|
|
||||||
|
|
||||||
// set codec correctly
|
|
||||||
av_register_all();
|
|
||||||
|
|
||||||
AVCodec *codec = NULL;
|
|
||||||
codec = avcodec_find_encoder(AV_CODEC_ID_H264);
|
|
||||||
assert(codec);
|
|
||||||
|
|
||||||
this->codec_ctx = avcodec_alloc_context3(codec);
|
|
||||||
assert(this->codec_ctx);
|
|
||||||
this->codec_ctx->width = this->width;
|
|
||||||
this->codec_ctx->height = this->height;
|
|
||||||
this->codec_ctx->pix_fmt = AV_PIX_FMT_YUV420P;
|
|
||||||
this->codec_ctx->time_base = (AVRational){ 1, this->fps };
|
|
||||||
|
|
||||||
err = avio_open(&this->ofmt_ctx->pb, this->vid_path, AVIO_FLAG_WRITE);
|
|
||||||
assert(err >= 0);
|
|
||||||
|
|
||||||
this->wrote_codec_config = false;
|
|
||||||
} else {
|
|
||||||
this->of = fopen(this->vid_path, "wb");
|
|
||||||
assert(this->of);
|
|
||||||
#ifndef QCOM2
|
|
||||||
if (this->codec_config_len > 0) {
|
|
||||||
fwrite(this->codec_config, this->codec_config_len, 1, this->of);
|
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// create camera lock file
|
snprintf(vid_path, sizeof(vid_path), "%s/%s", path.c_str(), filename);
|
||||||
snprintf(this->lock_path, sizeof(this->lock_path), "%s/%s.lock", this->path.c_str(), filename);
|
|
||||||
int lock_fd = HANDLE_EINTR(open(this->lock_path, O_RDWR | O_CREAT, 0664));
|
if (avformat_alloc_output_context2(&ofmt_ctx, NULL, NULL, vid_path) < 0 || !ofmt_ctx) {
|
||||||
assert(lock_fd >= 0);
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
out_stream = avformat_new_stream(ofmt_ctx, NULL);
|
||||||
|
if (!out_stream) {
|
||||||
|
avformat_free_context(ofmt_ctx);
|
||||||
|
ofmt_ctx = nullptr;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
out_stream->time_base = AVRational{1, fps};
|
||||||
|
|
||||||
|
out_stream->codecpar->codec_id = AV_CODEC_ID_H264;
|
||||||
|
out_stream->codecpar->codec_type = AVMEDIA_TYPE_VIDEO;
|
||||||
|
out_stream->codecpar->width = width;
|
||||||
|
out_stream->codecpar->height = height;
|
||||||
|
|
||||||
|
int err = avio_open(&ofmt_ctx->pb, vid_path, AVIO_FLAG_WRITE);
|
||||||
|
if (err < 0) {
|
||||||
|
avformat_free_context(ofmt_ctx);
|
||||||
|
ofmt_ctx = nullptr;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
wrote_codec_config = false;
|
||||||
|
|
||||||
|
snprintf(lock_path, sizeof(lock_path), "%s/%s.lock", path.c_str(), filename);
|
||||||
|
int lock_fd = HANDLE_EINTR(open(lock_path, O_RDWR | O_CREAT, 0664));
|
||||||
|
if (lock_fd < 0) {
|
||||||
|
avio_closep(&ofmt_ctx->pb);
|
||||||
|
avformat_free_context(ofmt_ctx);
|
||||||
|
ofmt_ctx = nullptr;
|
||||||
|
return;
|
||||||
|
}
|
||||||
close(lock_fd);
|
close(lock_fd);
|
||||||
|
|
||||||
this->is_open = true;
|
is_open = true;
|
||||||
this->counter = 0;
|
counter = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void OmxEncoder::encoder_close() {
|
void OmxEncoder::encoder_close() {
|
||||||
if (this->is_open) {
|
if (!is_open) return;
|
||||||
if (this->dirty) {
|
|
||||||
// drain output only if there could be frames in the encoder
|
|
||||||
|
|
||||||
OMX_BUFFERHEADERTYPE* in_buf = this->free_in.pop();
|
if (dirty) {
|
||||||
|
OMX_BUFFERHEADERTYPE* in_buf = free_in.pop();
|
||||||
|
if (in_buf) {
|
||||||
in_buf->nFilledLen = 0;
|
in_buf->nFilledLen = 0;
|
||||||
in_buf->nOffset = 0;
|
in_buf->nOffset = 0;
|
||||||
in_buf->nFlags = OMX_BUFFERFLAG_EOS;
|
in_buf->nFlags = OMX_BUFFERFLAG_EOS;
|
||||||
in_buf->nTimeStamp = this->last_t + 1000000LL/this->fps;
|
in_buf->nTimeStamp = last_t + 1000000LL / fps;
|
||||||
|
|
||||||
OMX_CHECK(OMX_EmptyThisBuffer(this->handle, in_buf));
|
OMX_CHECK(OMX_EmptyThisBuffer(handle, in_buf));
|
||||||
|
|
||||||
while (true) {
|
while (true) {
|
||||||
OMX_BUFFERHEADERTYPE *out_buf = this->done_out.pop();
|
OMX_BUFFERHEADERTYPE *out_buf = done_out.pop();
|
||||||
|
if (!out_buf) break;
|
||||||
|
|
||||||
handle_out_buf(this, out_buf);
|
handle_out_buf(this, out_buf);
|
||||||
|
|
||||||
@@ -697,55 +527,112 @@ void OmxEncoder::encoder_close() {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
this->dirty = false;
|
|
||||||
}
|
}
|
||||||
|
dirty = false;
|
||||||
if (this->remuxing) {
|
}
|
||||||
av_write_trailer(this->ofmt_ctx);
|
|
||||||
avcodec_free_context(&this->codec_ctx);
|
if (out_stream) {
|
||||||
avio_closep(&this->ofmt_ctx->pb);
|
out_stream->nb_frames = counter;
|
||||||
avformat_free_context(this->ofmt_ctx);
|
out_stream->duration = av_rescale_q(counter, AVRational{1, fps}, out_stream->time_base);
|
||||||
} else {
|
}
|
||||||
fclose(this->of);
|
|
||||||
this->of = nullptr;
|
if (ofmt_ctx) {
|
||||||
}
|
av_write_trailer(ofmt_ctx);
|
||||||
unlink(this->lock_path);
|
ofmt_ctx->duration = out_stream ? out_stream->duration : 0;
|
||||||
|
avio_closep(&ofmt_ctx->pb);
|
||||||
|
avformat_free_context(ofmt_ctx);
|
||||||
|
ofmt_ctx = nullptr;
|
||||||
|
out_stream = nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (lock_path[0] != '\0') {
|
||||||
|
unlink(lock_path);
|
||||||
|
}
|
||||||
|
|
||||||
|
is_open = false;
|
||||||
|
|
||||||
|
// Remux with faststart for streaming/seeking support
|
||||||
|
if (strlen(vid_path) > 0) {
|
||||||
|
char fixed_path[1024];
|
||||||
|
snprintf(fixed_path, sizeof(fixed_path), "%s.fixed.mp4", vid_path);
|
||||||
|
|
||||||
|
char cmd[2048];
|
||||||
|
snprintf(cmd, sizeof(cmd), "ffmpeg -y -i \"%s\" -c copy -movflags +faststart \"%s\" && mv \"%s\" \"%s\"",
|
||||||
|
vid_path, fixed_path, fixed_path, vid_path);
|
||||||
|
|
||||||
|
int ret = system(cmd);
|
||||||
|
if (ret != 0) {
|
||||||
|
LOGW("ffmpeg faststart remux failed with exit code %d", ret);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
this->is_open = false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
OmxEncoder::~OmxEncoder() {
|
OmxEncoder::~OmxEncoder() {
|
||||||
assert(!this->is_open);
|
if (is_open) {
|
||||||
|
LOGE("OmxEncoder closed with is_open=true, calling encoder_close()");
|
||||||
OMX_CHECK(OMX_SendCommand(this->handle, OMX_CommandStateSet, OMX_StateIdle, NULL));
|
encoder_close();
|
||||||
|
|
||||||
wait_for_state(OMX_StateIdle);
|
|
||||||
|
|
||||||
OMX_CHECK(OMX_SendCommand(this->handle, OMX_CommandStateSet, OMX_StateLoaded, NULL));
|
|
||||||
|
|
||||||
for (auto &buf : this->in_buf_headers) {
|
|
||||||
OMX_CHECK(OMX_FreeBuffer(this->handle, PORT_INDEX_IN, buf));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for (auto &buf : this->out_buf_headers) {
|
if (!handle) {
|
||||||
OMX_CHECK(OMX_FreeBuffer(this->handle, PORT_INDEX_OUT, buf));
|
LOGE("OMX handle is null in destructor, skipping teardown.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
OMX_ERRORTYPE err;
|
||||||
|
|
||||||
|
err = OMX_SendCommand(handle, OMX_CommandStateSet, OMX_StateIdle, NULL);
|
||||||
|
if (err != OMX_ErrorNone) {
|
||||||
|
LOGE("Failed to set OMX state to Idle: %x", err);
|
||||||
|
} else {
|
||||||
|
wait_for_state(OMX_StateIdle);
|
||||||
|
}
|
||||||
|
|
||||||
|
err = OMX_SendCommand(handle, OMX_CommandStateSet, OMX_StateLoaded, NULL);
|
||||||
|
if (err != OMX_ErrorNone) {
|
||||||
|
LOGE("Failed to set OMX state to Loaded: %x", err);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (OMX_BUFFERHEADERTYPE *buf : in_buf_headers) {
|
||||||
|
if (buf) {
|
||||||
|
err = OMX_FreeBuffer(handle, PORT_INDEX_IN, buf);
|
||||||
|
if (err != OMX_ErrorNone) {
|
||||||
|
LOGE("Failed to free input buffer: %x", err);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (OMX_BUFFERHEADERTYPE *buf : out_buf_headers) {
|
||||||
|
if (buf) {
|
||||||
|
err = OMX_FreeBuffer(handle, PORT_INDEX_OUT, buf);
|
||||||
|
if (err != OMX_ErrorNone) {
|
||||||
|
LOGE("Failed to free output buffer: %x", err);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
wait_for_state(OMX_StateLoaded);
|
wait_for_state(OMX_StateLoaded);
|
||||||
|
|
||||||
OMX_CHECK(OMX_FreeHandle(this->handle));
|
err = OMX_FreeHandle(handle);
|
||||||
|
if (err != OMX_ErrorNone) {
|
||||||
|
LOGE("Failed to free OMX handle: %x", err);
|
||||||
|
}
|
||||||
|
|
||||||
|
handle = nullptr;
|
||||||
|
|
||||||
|
err = OMX_Deinit();
|
||||||
|
if (err != OMX_ErrorNone) {
|
||||||
|
LOGE("OMX_Deinit failed: %x", err);
|
||||||
|
}
|
||||||
|
|
||||||
OMX_BUFFERHEADERTYPE *out_buf;
|
OMX_BUFFERHEADERTYPE *out_buf;
|
||||||
while (this->free_in.try_pop(out_buf));
|
while (free_in.try_pop(out_buf));
|
||||||
while (this->done_out.try_pop(out_buf));
|
while (done_out.try_pop(out_buf));
|
||||||
|
|
||||||
if (this->codec_config) {
|
if (codec_config) {
|
||||||
free(this->codec_config);
|
free(codec_config);
|
||||||
|
codec_config = nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this->downscale) {
|
in_buf_headers.clear();
|
||||||
free(this->y_ptr2);
|
out_buf_headers.clear();
|
||||||
free(this->u_ptr2);
|
|
||||||
free(this->v_ptr2);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,10 +12,10 @@ extern "C" {
|
|||||||
|
|
||||||
#include "common/queue.h"
|
#include "common/queue.h"
|
||||||
|
|
||||||
// OmxEncoder, lossey codec using hardware hevc
|
// OmxEncoder, lossey codec using hardware H.264
|
||||||
class OmxEncoder {
|
class OmxEncoder {
|
||||||
public:
|
public:
|
||||||
OmxEncoder(const char* path, int width, int height, int fps, int bitrate, bool h265, bool downscale);
|
OmxEncoder(const char* path, int width, int height, int fps, int bitrate);
|
||||||
~OmxEncoder();
|
~OmxEncoder();
|
||||||
|
|
||||||
int encode_frame_rgba(const uint8_t *ptr, int in_width, int in_height, uint64_t ts);
|
int encode_frame_rgba(const uint8_t *ptr, int in_width, int in_height, uint64_t ts);
|
||||||
@@ -44,31 +44,26 @@ private:
|
|||||||
int counter = 0;
|
int counter = 0;
|
||||||
|
|
||||||
std::string path;
|
std::string path;
|
||||||
FILE *of;
|
FILE *of = nullptr;
|
||||||
|
|
||||||
size_t codec_config_len;
|
size_t codec_config_len = 0;
|
||||||
uint8_t *codec_config = NULL;
|
uint8_t *codec_config = nullptr;
|
||||||
bool wrote_codec_config;
|
bool wrote_codec_config = false;
|
||||||
|
|
||||||
std::mutex state_lock;
|
std::mutex state_lock;
|
||||||
std::condition_variable state_cv;
|
std::condition_variable state_cv;
|
||||||
OMX_STATETYPE state = OMX_StateLoaded;
|
OMX_STATETYPE state = OMX_StateLoaded;
|
||||||
|
|
||||||
OMX_HANDLETYPE handle;
|
OMX_HANDLETYPE handle = nullptr;
|
||||||
|
|
||||||
std::vector<OMX_BUFFERHEADERTYPE *> in_buf_headers;
|
std::vector<OMX_BUFFERHEADERTYPE *> in_buf_headers;
|
||||||
std::vector<OMX_BUFFERHEADERTYPE *> out_buf_headers;
|
std::vector<OMX_BUFFERHEADERTYPE *> out_buf_headers;
|
||||||
|
|
||||||
uint64_t last_t;
|
uint64_t last_t = 0;
|
||||||
|
|
||||||
SafeQueue<OMX_BUFFERHEADERTYPE *> free_in;
|
SafeQueue<OMX_BUFFERHEADERTYPE *> free_in;
|
||||||
SafeQueue<OMX_BUFFERHEADERTYPE *> done_out;
|
SafeQueue<OMX_BUFFERHEADERTYPE *> done_out;
|
||||||
|
|
||||||
AVFormatContext *ofmt_ctx;
|
AVFormatContext *ofmt_ctx = nullptr;
|
||||||
AVCodecContext *codec_ctx;
|
AVStream *out_stream = nullptr;
|
||||||
AVStream *out_stream;
|
|
||||||
bool remuxing;
|
|
||||||
|
|
||||||
bool downscale;
|
|
||||||
uint8_t *y_ptr2, *u_ptr2, *v_ptr2;
|
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -27,7 +27,7 @@ ScreenRecorder::ScreenRecorder(QWidget *parent) : QPushButton(parent), image_que
|
|||||||
|
|
||||||
void ScreenRecorder::initializeEncoder() {
|
void ScreenRecorder::initializeEncoder() {
|
||||||
const std::string path = "/data/media/0/videos";
|
const std::string path = "/data/media/0/videos";
|
||||||
encoder = std::make_unique<OmxEncoder>(path.c_str(), recording_width, recording_height, 60, 2 * 1024 * 1024, false, false);
|
encoder = std::make_unique<OmxEncoder>(path.c_str(), recording_width, recording_height, 60, 2 * 1024 * 1024);
|
||||||
}
|
}
|
||||||
|
|
||||||
ScreenRecorder::~ScreenRecorder() {
|
ScreenRecorder::~ScreenRecorder() {
|
||||||
|
|||||||
@@ -87,8 +87,19 @@ def manager_init(frogpilot_functions) -> None:
|
|||||||
params_memory = Params("/dev/shm/params")
|
params_memory = Params("/dev/shm/params")
|
||||||
params_memory.put("TelemetryEnabled", "0")
|
params_memory.put("TelemetryEnabled", "0")
|
||||||
params_memory.put("VpnEnabled", "1")
|
params_memory.put("VpnEnabled", "1")
|
||||||
|
params_memory.put("DashcamFrames", "0")
|
||||||
|
params_memory.put("DashcamState", "stopped")
|
||||||
params_memory.put("ModelStandby", "0")
|
params_memory.put("ModelStandby", "0")
|
||||||
params_memory.put("ModelStandbyTs", "0")
|
params_memory.put("ModelStandbyTs", "0")
|
||||||
|
params_memory.put("CarIsMetric", "0")
|
||||||
|
params_memory.put("ClearpilotSpeedDisplay", "")
|
||||||
|
params_memory.put("ClearpilotSpeedLimitDisplay", "0")
|
||||||
|
params_memory.put("ClearpilotHasSpeed", "0")
|
||||||
|
params_memory.put("ClearpilotIsMetric", "0")
|
||||||
|
params_memory.put("ClearpilotSpeedUnit", "mph")
|
||||||
|
params_memory.put("ClearpilotCruiseWarning", "")
|
||||||
|
params_memory.put("ClearpilotCruiseWarningSpeed", "")
|
||||||
|
params_memory.put("ClearpilotPlayDing", "0")
|
||||||
params.clear_all(ParamKeyType.CLEAR_ON_ONROAD_TRANSITION)
|
params.clear_all(ParamKeyType.CLEAR_ON_ONROAD_TRANSITION)
|
||||||
params.clear_all(ParamKeyType.CLEAR_ON_OFFROAD_TRANSITION)
|
params.clear_all(ParamKeyType.CLEAR_ON_OFFROAD_TRANSITION)
|
||||||
if is_release_branch():
|
if is_release_branch():
|
||||||
|
|||||||
@@ -248,7 +248,8 @@ def main(demo=False):
|
|||||||
lat_active = sm['carControl'].latActive
|
lat_active = sm['carControl'].latActive
|
||||||
lane_changing = params_memory.get_bool("no_lat_lane_change")
|
lane_changing = params_memory.get_bool("no_lat_lane_change")
|
||||||
standstill = sm['carState'].standstill
|
standstill = sm['carState'].standstill
|
||||||
full_rate = lat_active or lane_changing
|
calibrating = sm['liveCalibration'].calStatus != log.LiveCalibrationData.Status.calibrated
|
||||||
|
full_rate = lat_active or lane_changing or calibrating
|
||||||
|
|
||||||
# Standby transitions (standstill only)
|
# Standby transitions (standstill only)
|
||||||
should_standby = standstill and not full_rate
|
should_standby = standstill and not full_rate
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
#!/usr/bin/env python3
|
#!/usr/bin/env python3
|
||||||
|
import os
|
||||||
from abc import ABC, abstractmethod
|
from abc import ABC, abstractmethod
|
||||||
|
|
||||||
from openpilot.common.realtime import DT_TRML
|
from openpilot.common.realtime import DT_TRML
|
||||||
@@ -6,6 +7,8 @@ from openpilot.common.numpy_fast import interp
|
|||||||
from openpilot.common.swaglog import cloudlog
|
from openpilot.common.swaglog import cloudlog
|
||||||
from openpilot.selfdrive.controls.lib.pid import PIDController
|
from openpilot.selfdrive.controls.lib.pid import PIDController
|
||||||
|
|
||||||
|
BENCH_MODE = os.environ.get("BENCH_MODE") == "1"
|
||||||
|
|
||||||
class BaseFanController(ABC):
|
class BaseFanController(ABC):
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def update(self, cur_temp: float, ignition: bool, standstill: bool = False) -> int:
|
def update(self, cur_temp: float, ignition: bool, standstill: bool = False) -> int:
|
||||||
@@ -21,10 +24,14 @@ class TiciFanController(BaseFanController):
|
|||||||
self.controller = PIDController(k_p=0, k_i=4e-3, k_f=1, rate=(1 / DT_TRML))
|
self.controller = PIDController(k_p=0, k_i=4e-3, k_f=1, rate=(1 / DT_TRML))
|
||||||
|
|
||||||
def update(self, cur_temp: float, ignition: bool, standstill: bool = False) -> int:
|
def update(self, cur_temp: float, ignition: bool, standstill: bool = False) -> int:
|
||||||
# CLEARPILOT: at standstill below 74°C, clamp to 0-10% (near silent)
|
# CLEARPILOT: bench mode always uses normal onroad fan range (30-100%)
|
||||||
|
if BENCH_MODE and ignition:
|
||||||
|
self.controller.neg_limit = -100
|
||||||
|
self.controller.pos_limit = -30
|
||||||
|
# CLEARPILOT: at standstill below 74°C, clamp to 0-30% (quiet)
|
||||||
# at standstill above 74°C, allow full 0-100% range
|
# at standstill above 74°C, allow full 0-100% range
|
||||||
if ignition and standstill and cur_temp < 74:
|
elif ignition and standstill and cur_temp < 74:
|
||||||
self.controller.neg_limit = -10
|
self.controller.neg_limit = -30
|
||||||
self.controller.pos_limit = 0
|
self.controller.pos_limit = 0
|
||||||
elif ignition and standstill:
|
elif ignition and standstill:
|
||||||
self.controller.neg_limit = -100
|
self.controller.neg_limit = -100
|
||||||
|
|||||||
@@ -85,15 +85,16 @@ void HomeWindow::updateState(const UIState &s) {
|
|||||||
showDriverView(s.scene.driver_camera_timer >= 10, true);
|
showDriverView(s.scene.driver_camera_timer >= 10, true);
|
||||||
|
|
||||||
// CLEARPILOT: show splash screen when onroad but in park
|
// CLEARPILOT: show splash screen when onroad but in park
|
||||||
// In nightrider mode (states 1,4), stay on onroad view in park — only offroad transition shows splash
|
|
||||||
bool parked = s.scene.parked;
|
bool parked = s.scene.parked;
|
||||||
int screenMode = paramsMemory.getInt("ScreenDisplayMode");
|
int screenMode = paramsMemory.getInt("ScreenDisplayMode");
|
||||||
bool nightrider = (screenMode == 1 || screenMode == 4);
|
bool nightrider = (screenMode == 1 || screenMode == 4);
|
||||||
|
|
||||||
if (parked && !was_parked_onroad) {
|
if (parked && !was_parked_onroad) {
|
||||||
if (!nightrider) {
|
LOGW("CLP UI: park transition -> showing splash");
|
||||||
LOGW("CLP UI: park transition -> showing splash");
|
slayout->setCurrentWidget(ready);
|
||||||
slayout->setCurrentWidget(ready);
|
// If we were in nightrider mode, switch to screen off
|
||||||
|
if (nightrider) {
|
||||||
|
paramsMemory.putInt("ScreenDisplayMode", 3);
|
||||||
}
|
}
|
||||||
} else if (!parked && was_parked_onroad) {
|
} else if (!parked && was_parked_onroad) {
|
||||||
LOGW("CLP UI: drive transition -> showing onroad");
|
LOGW("CLP UI: drive transition -> showing onroad");
|
||||||
|
|||||||
@@ -355,14 +355,15 @@ void AnnotatedCameraWidget::updateState(const UIState &s) {
|
|||||||
setSpeed *= KM_TO_MILE;
|
setSpeed *= KM_TO_MILE;
|
||||||
}
|
}
|
||||||
|
|
||||||
// CLEARPILOT: use GPS speed if available, hide speed display if no GPS fix
|
// CLEARPILOT: read speed and speed limit from params (written by speed_logic.py ~2Hz)
|
||||||
has_gps_speed = sm.valid("gpsLocation") && sm["gpsLocation"].getGpsLocation().getHasFix();
|
clpParamFrame++;
|
||||||
if (has_gps_speed) {
|
if (clpParamFrame % 10 == 0) { // ~2Hz at 20Hz UI update rate
|
||||||
float gps_speed_ms = sm["gpsLocation"].getGpsLocation().getSpeed();
|
clpHasSpeed = paramsMemory.get("ClearpilotHasSpeed") == "1";
|
||||||
speed = std::max<float>(0.0, gps_speed_ms);
|
clpSpeedDisplay = QString::fromStdString(paramsMemory.get("ClearpilotSpeedDisplay"));
|
||||||
speed *= s.scene.is_metric ? MS_TO_KPH : MS_TO_MPH;
|
clpSpeedLimitDisplay = QString::fromStdString(paramsMemory.get("ClearpilotSpeedLimitDisplay"));
|
||||||
} else {
|
clpSpeedUnit = QString::fromStdString(paramsMemory.get("ClearpilotSpeedUnit"));
|
||||||
speed = 0.0;
|
clpCruiseWarning = QString::fromStdString(paramsMemory.get("ClearpilotCruiseWarning"));
|
||||||
|
clpCruiseWarningSpeed = QString::fromStdString(paramsMemory.get("ClearpilotCruiseWarningSpeed"));
|
||||||
}
|
}
|
||||||
|
|
||||||
// auto speed_limit_sign = nav_instruction.getSpeedLimitSign();
|
// auto speed_limit_sign = nav_instruction.getSpeedLimitSign();
|
||||||
@@ -431,7 +432,7 @@ void AnnotatedCameraWidget::drawHud(QPainter &p) {
|
|||||||
|
|
||||||
// QString speedLimitStr = (speedLimit > 1) ? QString::number(std::nearbyint(speedLimit)) : "–";
|
// QString speedLimitStr = (speedLimit > 1) ? QString::number(std::nearbyint(speedLimit)) : "–";
|
||||||
// QString speedLimitOffsetStr = slcSpeedLimitOffset == 0 ? "–" : QString::number(slcSpeedLimitOffset, 'f', 0).prepend(slcSpeedLimitOffset > 0 ? "+" : "");
|
// QString speedLimitOffsetStr = slcSpeedLimitOffset == 0 ? "–" : QString::number(slcSpeedLimitOffset, 'f', 0).prepend(slcSpeedLimitOffset > 0 ? "+" : "");
|
||||||
QString speedStr = QString::number(std::nearbyint(speed));
|
// QString speedStr = QString::number(std::nearbyint(speed));
|
||||||
QString setSpeedStr = is_cruise_set ? QString::number(std::nearbyint(setSpeed - cruiseAdjustment)) : "–";
|
QString setSpeedStr = is_cruise_set ? QString::number(std::nearbyint(setSpeed - cruiseAdjustment)) : "–";
|
||||||
|
|
||||||
p.restore();
|
p.restore();
|
||||||
@@ -455,14 +456,18 @@ void AnnotatedCameraWidget::drawHud(QPainter &p) {
|
|||||||
// Todo: lead speed
|
// Todo: lead speed
|
||||||
// Todo: Experimental speed
|
// Todo: Experimental speed
|
||||||
|
|
||||||
// CLEARPILOT: show GPS speed, hide when no fix
|
// CLEARPILOT: show speed from speed_logic params, hide when no speed or speed=0
|
||||||
if (has_gps_speed && !scene.hide_speed) {
|
if (clpHasSpeed && !clpSpeedDisplay.isEmpty() && !scene.hide_speed) {
|
||||||
p.setFont(InterFont(140, QFont::Bold));
|
p.setFont(InterFont(140, QFont::Bold));
|
||||||
drawText(p, rect().center().x(), 210, speedStr);
|
drawText(p, rect().center().x(), 210, clpSpeedDisplay);
|
||||||
p.setFont(InterFont(50));
|
p.setFont(InterFont(50));
|
||||||
drawText(p, rect().center().x(), 290, speedUnit, 200);
|
drawText(p, rect().center().x(), 290, clpSpeedUnit, 200);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// CLEARPILOT: speed limit sign in lower-left, cruise warning above it
|
||||||
|
drawSpeedLimitSign(p);
|
||||||
|
drawCruiseWarningSign(p);
|
||||||
|
|
||||||
// Draw FrogPilot widgets
|
// Draw FrogPilot widgets
|
||||||
paintFrogPilotWidgets(p);
|
paintFrogPilotWidgets(p);
|
||||||
}
|
}
|
||||||
@@ -556,6 +561,164 @@ void AnnotatedCameraWidget::drawSpeedWidget(QPainter &p, int x, int y, const QSt
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void AnnotatedCameraWidget::drawSpeedLimitSign(QPainter &p) {
|
||||||
|
// Hide when no speed limit or speed limit is 0
|
||||||
|
if (clpSpeedLimitDisplay.isEmpty() || clpSpeedLimitDisplay == "0") return;
|
||||||
|
|
||||||
|
p.save();
|
||||||
|
|
||||||
|
const int signW = 189;
|
||||||
|
const int signH = 239;
|
||||||
|
const int margin = 20;
|
||||||
|
const int borderW = 6;
|
||||||
|
const int innerBorderW = 4;
|
||||||
|
const int innerMargin = 10;
|
||||||
|
const int cornerR = 15;
|
||||||
|
|
||||||
|
// Position: 20px from lower-left corner
|
||||||
|
QRect signRect(margin, height() - signH - margin, signW, signH);
|
||||||
|
|
||||||
|
if (nightriderMode) {
|
||||||
|
// Nightrider: black background, light gray-blue border and text
|
||||||
|
QColor borderColor(160, 180, 210);
|
||||||
|
QColor textColor(160, 180, 210);
|
||||||
|
|
||||||
|
// Outer border
|
||||||
|
p.setPen(QPen(borderColor, borderW));
|
||||||
|
p.setBrush(QColor(0, 0, 0, 220));
|
||||||
|
p.drawRoundedRect(signRect, cornerR, cornerR);
|
||||||
|
|
||||||
|
// Inner border
|
||||||
|
QRect innerRect = signRect.adjusted(innerMargin, innerMargin, -innerMargin, -innerMargin);
|
||||||
|
p.setPen(QPen(borderColor, innerBorderW));
|
||||||
|
p.setBrush(Qt::NoBrush);
|
||||||
|
p.drawRoundedRect(innerRect, cornerR - 4, cornerR - 4);
|
||||||
|
|
||||||
|
// "SPEED" text
|
||||||
|
p.setPen(textColor);
|
||||||
|
p.setFont(InterFont(30, QFont::Bold));
|
||||||
|
p.drawText(innerRect.adjusted(0, 15, 0, 0), Qt::AlignTop | Qt::AlignHCenter, "SPEED");
|
||||||
|
|
||||||
|
// "LIMIT" text
|
||||||
|
p.setFont(InterFont(30, QFont::Bold));
|
||||||
|
p.drawText(innerRect.adjusted(0, 48, 0, 0), Qt::AlignTop | Qt::AlignHCenter, "LIMIT");
|
||||||
|
|
||||||
|
// Speed limit number
|
||||||
|
p.setFont(InterFont(90, QFont::Bold));
|
||||||
|
p.drawText(innerRect.adjusted(0, 42, 0, 0), Qt::AlignCenter, clpSpeedLimitDisplay);
|
||||||
|
} else {
|
||||||
|
// Normal: white background, black border and text
|
||||||
|
QColor borderColor(0, 0, 0);
|
||||||
|
QColor textColor(0, 0, 0);
|
||||||
|
|
||||||
|
// Outer border
|
||||||
|
p.setPen(QPen(borderColor, borderW));
|
||||||
|
p.setBrush(QColor(255, 255, 255, 240));
|
||||||
|
p.drawRoundedRect(signRect, cornerR, cornerR);
|
||||||
|
|
||||||
|
// Inner border
|
||||||
|
QRect innerRect = signRect.adjusted(innerMargin, innerMargin, -innerMargin, -innerMargin);
|
||||||
|
p.setPen(QPen(borderColor, innerBorderW));
|
||||||
|
p.setBrush(Qt::NoBrush);
|
||||||
|
p.drawRoundedRect(innerRect, cornerR - 4, cornerR - 4);
|
||||||
|
|
||||||
|
// "SPEED" text
|
||||||
|
p.setPen(textColor);
|
||||||
|
p.setFont(InterFont(30, QFont::Bold));
|
||||||
|
p.drawText(innerRect.adjusted(0, 15, 0, 0), Qt::AlignTop | Qt::AlignHCenter, "SPEED");
|
||||||
|
|
||||||
|
// "LIMIT" text
|
||||||
|
p.setFont(InterFont(30, QFont::Bold));
|
||||||
|
p.drawText(innerRect.adjusted(0, 48, 0, 0), Qt::AlignTop | Qt::AlignHCenter, "LIMIT");
|
||||||
|
|
||||||
|
// Speed limit number
|
||||||
|
p.setFont(InterFont(90, QFont::Bold));
|
||||||
|
p.drawText(innerRect.adjusted(0, 42, 0, 0), Qt::AlignCenter, clpSpeedLimitDisplay);
|
||||||
|
}
|
||||||
|
|
||||||
|
p.restore();
|
||||||
|
}
|
||||||
|
|
||||||
|
void AnnotatedCameraWidget::drawCruiseWarningSign(QPainter &p) {
|
||||||
|
// Only show when there's an active warning and the speed limit sign is visible
|
||||||
|
if (clpCruiseWarning.isEmpty() || clpCruiseWarningSpeed.isEmpty()) return;
|
||||||
|
if (clpSpeedLimitDisplay.isEmpty() || clpSpeedLimitDisplay == "0") return;
|
||||||
|
|
||||||
|
bool isOver = (clpCruiseWarning == "over");
|
||||||
|
if (!isOver && clpCruiseWarning != "under") return;
|
||||||
|
|
||||||
|
p.save();
|
||||||
|
|
||||||
|
// Same dimensions as speed limit sign
|
||||||
|
const int signW = 189;
|
||||||
|
const int signH = 239;
|
||||||
|
const int margin = 20;
|
||||||
|
const int borderW = 6;
|
||||||
|
const int innerBorderW = 4;
|
||||||
|
const int innerMargin = 10;
|
||||||
|
const int cornerR = 15;
|
||||||
|
const int gap = 20;
|
||||||
|
|
||||||
|
// Position: directly above the speed limit sign
|
||||||
|
int speedLimitY = height() - signH - margin;
|
||||||
|
QRect signRect(margin, speedLimitY - signH - gap, signW, signH);
|
||||||
|
|
||||||
|
if (nightriderMode) {
|
||||||
|
// Nightrider: black background with colored border/text
|
||||||
|
QColor accentColor = isOver ? QColor(220, 50, 50) : QColor(50, 180, 80);
|
||||||
|
|
||||||
|
p.setPen(QPen(accentColor, borderW));
|
||||||
|
p.setBrush(QColor(0, 0, 0, 220));
|
||||||
|
p.drawRoundedRect(signRect, cornerR, cornerR);
|
||||||
|
|
||||||
|
QRect innerRect = signRect.adjusted(innerMargin, innerMargin, -innerMargin, -innerMargin);
|
||||||
|
p.setPen(QPen(accentColor, innerBorderW));
|
||||||
|
p.setBrush(Qt::NoBrush);
|
||||||
|
p.drawRoundedRect(innerRect, cornerR - 4, cornerR - 4);
|
||||||
|
|
||||||
|
// "CRUISE" text
|
||||||
|
p.setPen(accentColor);
|
||||||
|
p.setFont(InterFont(26, QFont::Bold));
|
||||||
|
p.drawText(innerRect.adjusted(0, 15, 0, 0), Qt::AlignTop | Qt::AlignHCenter, "CRUISE");
|
||||||
|
|
||||||
|
// "SET" text
|
||||||
|
p.setFont(InterFont(26, QFont::Bold));
|
||||||
|
p.drawText(innerRect.adjusted(0, 45, 0, 0), Qt::AlignTop | Qt::AlignHCenter, "SET");
|
||||||
|
|
||||||
|
// Cruise speed number
|
||||||
|
p.setFont(InterFont(90, QFont::Bold));
|
||||||
|
p.drawText(innerRect.adjusted(0, 42, 0, 0), Qt::AlignCenter, clpCruiseWarningSpeed);
|
||||||
|
} else {
|
||||||
|
// Normal: colored background with white border/text
|
||||||
|
QColor bgColor = isOver ? QColor(200, 30, 30, 240) : QColor(40, 160, 60, 240);
|
||||||
|
QColor fgColor(255, 255, 255);
|
||||||
|
|
||||||
|
p.setPen(QPen(fgColor, borderW));
|
||||||
|
p.setBrush(bgColor);
|
||||||
|
p.drawRoundedRect(signRect, cornerR, cornerR);
|
||||||
|
|
||||||
|
QRect innerRect = signRect.adjusted(innerMargin, innerMargin, -innerMargin, -innerMargin);
|
||||||
|
p.setPen(QPen(fgColor, innerBorderW));
|
||||||
|
p.setBrush(Qt::NoBrush);
|
||||||
|
p.drawRoundedRect(innerRect, cornerR - 4, cornerR - 4);
|
||||||
|
|
||||||
|
// "CRUISE" text
|
||||||
|
p.setPen(fgColor);
|
||||||
|
p.setFont(InterFont(26, QFont::Bold));
|
||||||
|
p.drawText(innerRect.adjusted(0, 15, 0, 0), Qt::AlignTop | Qt::AlignHCenter, "CRUISE");
|
||||||
|
|
||||||
|
// "SET" text
|
||||||
|
p.setFont(InterFont(26, QFont::Bold));
|
||||||
|
p.drawText(innerRect.adjusted(0, 45, 0, 0), Qt::AlignTop | Qt::AlignHCenter, "SET");
|
||||||
|
|
||||||
|
// Cruise speed number
|
||||||
|
p.setFont(InterFont(90, QFont::Bold));
|
||||||
|
p.drawText(innerRect.adjusted(0, 42, 0, 0), Qt::AlignCenter, clpCruiseWarningSpeed);
|
||||||
|
}
|
||||||
|
|
||||||
|
p.restore();
|
||||||
|
}
|
||||||
|
|
||||||
void AnnotatedCameraWidget::drawText(QPainter &p, int x, int y, const QString &text, int alpha) {
|
void AnnotatedCameraWidget::drawText(QPainter &p, int x, int y, const QString &text, int alpha) {
|
||||||
QRect real_rect = p.fontMetrics().boundingRect(text);
|
QRect real_rect = p.fontMetrics().boundingRect(text);
|
||||||
real_rect.moveCenter({x, y - real_rect.height() / 2});
|
real_rect.moveCenter({x, y - real_rect.height() / 2});
|
||||||
|
|||||||
@@ -51,6 +51,8 @@ public:
|
|||||||
private:
|
private:
|
||||||
void drawText(QPainter &p, int x, int y, const QString &text, int alpha = 255);
|
void drawText(QPainter &p, int x, int y, const QString &text, int alpha = 255);
|
||||||
void drawSpeedWidget(QPainter &p, int x, int y, const QString &title, const QString &speedLimitStr, QColor colorSpeed, int width = 176);
|
void drawSpeedWidget(QPainter &p, int x, int y, const QString &title, const QString &speedLimitStr, QColor colorSpeed, int width = 176);
|
||||||
|
void drawSpeedLimitSign(QPainter &p);
|
||||||
|
void drawCruiseWarningSign(QPainter &p);
|
||||||
|
|
||||||
QVBoxLayout *main_layout;
|
QVBoxLayout *main_layout;
|
||||||
QPixmap dm_img;
|
QPixmap dm_img;
|
||||||
@@ -59,6 +61,16 @@ private:
|
|||||||
bool nightriderMode = false;
|
bool nightriderMode = false;
|
||||||
int displayMode = 0;
|
int displayMode = 0;
|
||||||
QString speedUnit;
|
QString speedUnit;
|
||||||
|
|
||||||
|
// ClearPilot speed state (from params_memory, updated ~2Hz)
|
||||||
|
bool clpHasSpeed = false;
|
||||||
|
QString clpSpeedDisplay;
|
||||||
|
QString clpSpeedLimitDisplay;
|
||||||
|
QString clpSpeedUnit;
|
||||||
|
QString clpCruiseWarning;
|
||||||
|
QString clpCruiseWarningSpeed;
|
||||||
|
int clpParamFrame = 0;
|
||||||
|
|
||||||
float setSpeed;
|
float setSpeed;
|
||||||
float speedLimit;
|
float speedLimit;
|
||||||
bool is_cruise_set = false;
|
bool is_cruise_set = false;
|
||||||
|
|||||||
Binary file not shown.
@@ -171,6 +171,13 @@ bool MainWindow::eventFilter(QObject *obj, QEvent *event) {
|
|||||||
case QEvent::TouchEnd:
|
case QEvent::TouchEnd:
|
||||||
case QEvent::MouseButtonPress:
|
case QEvent::MouseButtonPress:
|
||||||
case QEvent::MouseMove: {
|
case QEvent::MouseMove: {
|
||||||
|
// CLEARPILOT: tap while screen-off (mode 3) -> wake to auto-normal (mode 0)
|
||||||
|
if (!device()->isAwake()) {
|
||||||
|
Params pmem{"/dev/shm/params"};
|
||||||
|
if (pmem.getInt("ScreenDisplayMode") == 3) {
|
||||||
|
pmem.putInt("ScreenDisplayMode", 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
// ignore events when device is awakened by resetInteractiveTimeout
|
// ignore events when device is awakened by resetInteractiveTimeout
|
||||||
ignore = !device()->isAwake();
|
ignore = !device()->isAwake();
|
||||||
device()->resetInteractiveTimeout(uiState()->scene.screen_timeout, uiState()->scene.screen_timeout_onroad);
|
device()->resetInteractiveTimeout(uiState()->scene.screen_timeout, uiState()->scene.screen_timeout_onroad);
|
||||||
@@ -256,6 +263,11 @@ static StatusWindow::StatusData collectStatus() {
|
|||||||
// Telemetry
|
// Telemetry
|
||||||
d.telemetry = readFile("/data/params/d/TelemetryEnabled");
|
d.telemetry = readFile("/data/params/d/TelemetryEnabled");
|
||||||
|
|
||||||
|
// Dashcam
|
||||||
|
d.dashcam_state = readFile("/dev/shm/params/d/DashcamState");
|
||||||
|
if (d.dashcam_state.isEmpty()) d.dashcam_state = "stopped";
|
||||||
|
d.dashcam_frames = readFile("/dev/shm/params/d/DashcamFrames");
|
||||||
|
|
||||||
// Panda: checked on UI thread in applyResults() via scene.pandaType
|
// Panda: checked on UI thread in applyResults() via scene.pandaType
|
||||||
|
|
||||||
return d;
|
return d;
|
||||||
@@ -298,6 +310,7 @@ StatusWindow::StatusWindow(QWidget *parent) : QFrame(parent) {
|
|||||||
vpn_label = makeRow("VPN");
|
vpn_label = makeRow("VPN");
|
||||||
gps_label = makeRow("GPS");
|
gps_label = makeRow("GPS");
|
||||||
telemetry_label = makeRow("Telemetry");
|
telemetry_label = makeRow("Telemetry");
|
||||||
|
dashcam_label = makeRow("Dashcam");
|
||||||
|
|
||||||
layout->addStretch();
|
layout->addStretch();
|
||||||
|
|
||||||
@@ -369,6 +382,19 @@ void StatusWindow::applyResults() {
|
|||||||
telemetry_label->setText("Disabled");
|
telemetry_label->setText("Disabled");
|
||||||
telemetry_label->setStyleSheet("color: grey; font-size: 38px;");
|
telemetry_label->setStyleSheet("color: grey; font-size: 38px;");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (d.dashcam_state == "recording") {
|
||||||
|
QString text = "Recording";
|
||||||
|
if (!d.dashcam_frames.isEmpty() && d.dashcam_frames != "0") text += " (" + d.dashcam_frames + " frames)";
|
||||||
|
dashcam_label->setText(text);
|
||||||
|
dashcam_label->setStyleSheet("color: #17c44d; font-size: 38px;");
|
||||||
|
} else if (d.dashcam_state == "waiting") {
|
||||||
|
dashcam_label->setText("Waiting");
|
||||||
|
dashcam_label->setStyleSheet("color: #ffaa00; font-size: 38px;");
|
||||||
|
} else {
|
||||||
|
dashcam_label->setText("Stopped");
|
||||||
|
dashcam_label->setStyleSheet("color: #ff4444; font-size: 38px;");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void StatusWindow::mousePressEvent(QMouseEvent *e) {
|
void StatusWindow::mousePressEvent(QMouseEvent *e) {
|
||||||
|
|||||||
@@ -20,6 +20,7 @@ public:
|
|||||||
struct StatusData {
|
struct StatusData {
|
||||||
QString time, storage, ram, load, temp, fan, ip, wifi;
|
QString time, storage, ram, load, temp, fan, ip, wifi;
|
||||||
QString vpn_status, vpn_ip, gps, telemetry;
|
QString vpn_status, vpn_ip, gps, telemetry;
|
||||||
|
QString dashcam_state, dashcam_frames;
|
||||||
float temp_c = 0;
|
float temp_c = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -49,6 +50,7 @@ private:
|
|||||||
QLabel *gps_label;
|
QLabel *gps_label;
|
||||||
QLabel *time_label;
|
QLabel *time_label;
|
||||||
QLabel *telemetry_label;
|
QLabel *telemetry_label;
|
||||||
|
QLabel *dashcam_label;
|
||||||
QLabel *panda_label;
|
QLabel *panda_label;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -93,6 +93,27 @@ class Soundd:
|
|||||||
|
|
||||||
self.spl_filter_weighted = FirstOrderFilter(0, 2.5, FILTER_DT, initialized=False)
|
self.spl_filter_weighted = FirstOrderFilter(0, 2.5, FILTER_DT, initialized=False)
|
||||||
|
|
||||||
|
# ClearPilot ding (plays independently of alerts)
|
||||||
|
self.ding_sound = None
|
||||||
|
self.ding_frame = 0
|
||||||
|
self.ding_playing = False
|
||||||
|
self.ding_check_counter = 0
|
||||||
|
self._load_ding()
|
||||||
|
|
||||||
|
def _load_ding(self):
|
||||||
|
ding_path = BASEDIR + "/selfdrive/clearpilot/sounds/ding.wav"
|
||||||
|
try:
|
||||||
|
wavefile = wave.open(ding_path, 'r')
|
||||||
|
assert wavefile.getnchannels() == 1
|
||||||
|
assert wavefile.getsampwidth() == 2
|
||||||
|
assert wavefile.getframerate() == SAMPLE_RATE
|
||||||
|
length = wavefile.getnframes()
|
||||||
|
self.ding_sound = np.frombuffer(wavefile.readframes(length), dtype=np.int16).astype(np.float32) / (2**16/2)
|
||||||
|
cloudlog.info(f"ClearPilot ding loaded: {length} frames")
|
||||||
|
except Exception as e:
|
||||||
|
cloudlog.error(f"Failed to load ding sound: {e}")
|
||||||
|
self.ding_sound = None
|
||||||
|
|
||||||
def load_sounds(self):
|
def load_sounds(self):
|
||||||
self.loaded_sounds: dict[int, np.ndarray] = {}
|
self.loaded_sounds: dict[int, np.ndarray] = {}
|
||||||
|
|
||||||
@@ -137,7 +158,20 @@ class Soundd:
|
|||||||
written_frames += frames_to_write
|
written_frames += frames_to_write
|
||||||
self.current_sound_frame += frames_to_write
|
self.current_sound_frame += frames_to_write
|
||||||
|
|
||||||
return ret * self.current_volume
|
ret = ret * self.current_volume
|
||||||
|
|
||||||
|
# Mix in ClearPilot ding (independent of alerts, always max volume)
|
||||||
|
if self.ding_playing and self.ding_sound is not None:
|
||||||
|
ding_remaining = len(self.ding_sound) - self.ding_frame
|
||||||
|
if ding_remaining > 0:
|
||||||
|
frames_to_write = min(ding_remaining, frames)
|
||||||
|
ret[:frames_to_write] += self.ding_sound[self.ding_frame:self.ding_frame + frames_to_write] * MAX_VOLUME
|
||||||
|
self.ding_frame += frames_to_write
|
||||||
|
else:
|
||||||
|
self.ding_playing = False
|
||||||
|
self.ding_frame = 0
|
||||||
|
|
||||||
|
return ret
|
||||||
|
|
||||||
def callback(self, data_out: np.ndarray, frames: int, time, status) -> None:
|
def callback(self, data_out: np.ndarray, frames: int, time, status) -> None:
|
||||||
if status:
|
if status:
|
||||||
@@ -197,6 +231,14 @@ class Soundd:
|
|||||||
|
|
||||||
self.get_audible_alert(sm)
|
self.get_audible_alert(sm)
|
||||||
|
|
||||||
|
# ClearPilot: check for ding trigger at ~2Hz
|
||||||
|
self.ding_check_counter += 1
|
||||||
|
if self.ding_check_counter % 10 == 0 and self.ding_sound is not None:
|
||||||
|
if self.params_memory.get("ClearpilotPlayDing") == b"1":
|
||||||
|
self.params_memory.put("ClearpilotPlayDing", "0")
|
||||||
|
self.ding_playing = True
|
||||||
|
self.ding_frame = 0
|
||||||
|
|
||||||
rk.keep_time()
|
rk.keep_time()
|
||||||
|
|
||||||
assert stream.active
|
assert stream.active
|
||||||
|
|||||||
@@ -565,7 +565,10 @@ void Device::updateWakefulness(const UIState &s) {
|
|||||||
emit interactiveTimeout();
|
emit interactiveTimeout();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (s.scene.screen_brightness_onroad != 0) {
|
// CLEARPILOT: ScreenDisplayMode 3 = screen off — override awake state
|
||||||
|
if (paramsMemory.getInt("ScreenDisplayMode") == 3) {
|
||||||
|
setAwake(false);
|
||||||
|
} else if (s.scene.screen_brightness_onroad != 0) {
|
||||||
setAwake(s.scene.ignition || interactive_timeout > 0);
|
setAwake(s.scene.ignition || interactive_timeout > 0);
|
||||||
} else {
|
} else {
|
||||||
setAwake(interactive_timeout > 0);
|
setAwake(interactive_timeout > 0);
|
||||||
|
|||||||
@@ -9,6 +9,35 @@ echo -n 1 > /data/params/d/SshEnabled
|
|||||||
sudo systemctl enable ssh 2>/dev/null
|
sudo systemctl enable ssh 2>/dev/null
|
||||||
sudo systemctl start ssh
|
sudo systemctl start ssh
|
||||||
|
|
||||||
|
# Decrypt and install SSH identity keys for root (git auth)
|
||||||
|
serial=$(sed 's/.*androidboot.serialno=\([^ ]*\).*/\1/' /proc/cmdline)
|
||||||
|
ssh_dir="/root/.ssh"
|
||||||
|
if [[ $serial == 3889765b ]] && [[ ! -f "$ssh_dir/id_ed25519" || ! -f "$ssh_dir/id_ed25519.pub" ]]; then
|
||||||
|
echo "Decrypting SSH identity keys for root (serial=$serial)..."
|
||||||
|
tmpdir=$(mktemp -d)
|
||||||
|
bash /data/openpilot/system/clearpilot/tools/decrypt /data/openpilot/system/clearpilot/dev/id_ed25519.cpt "$tmpdir/id_ed25519"
|
||||||
|
bash /data/openpilot/system/clearpilot/tools/decrypt /data/openpilot/system/clearpilot/dev/id_ed25519.pub.cpt "$tmpdir/id_ed25519.pub"
|
||||||
|
sudo mkdir -p "$ssh_dir"
|
||||||
|
sudo cp "$tmpdir/id_ed25519" "$tmpdir/id_ed25519.pub" "$ssh_dir/"
|
||||||
|
rm -rf "$tmpdir"
|
||||||
|
sudo chmod 700 "$ssh_dir"
|
||||||
|
sudo chmod 600 "$ssh_dir/id_ed25519"
|
||||||
|
sudo chmod 644 "$ssh_dir/id_ed25519.pub"
|
||||||
|
echo "SSH identity keys installed to $ssh_dir"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Ensure root SSH config has git.hanson.xyz entry
|
||||||
|
if ! grep -q "Host git.hanson.xyz" "$ssh_dir/config" 2>/dev/null; then
|
||||||
|
sudo tee -a "$ssh_dir/config" > /dev/null <<'SSHCFG'
|
||||||
|
|
||||||
|
Host git.hanson.xyz
|
||||||
|
IdentityFile /root/.ssh/id_ed25519
|
||||||
|
StrictHostKeyChecking no
|
||||||
|
SSHCFG
|
||||||
|
sudo chmod 600 "$ssh_dir/config"
|
||||||
|
echo "SSH config updated for git.hanson.xyz"
|
||||||
|
fi
|
||||||
|
|
||||||
# Always ensure WiFi radio is on
|
# Always ensure WiFi radio is on
|
||||||
nmcli radio wifi on 2>/dev/null
|
nmcli radio wifi on 2>/dev/null
|
||||||
|
|
||||||
|
|||||||
@@ -38,27 +38,13 @@ sudo apt-get install -y nodejs
|
|||||||
mount -o rw,remount /
|
mount -o rw,remount /
|
||||||
echo "Installing Claude Code..."
|
echo "Installing Claude Code..."
|
||||||
curl -fsSL https://claude.ai/install.sh | bash
|
curl -fsSL https://claude.ai/install.sh | bash
|
||||||
echo 'export PATH="$HOME/.local/bin:$PATH"' >> ~/.bashrc && source ~/.bashrc
|
cat > /usr/local/bin/claude <<'WRAPPER'
|
||||||
|
#!/bin/bash
|
||||||
|
sudo mount -o rw,remount /
|
||||||
|
exec /root/.local/bin/claude "$@"
|
||||||
|
WRAPPER
|
||||||
|
chmod +x /usr/local/bin/claude
|
||||||
echo "Packages installed"
|
echo "Packages installed"
|
||||||
|
|
||||||
# Decrypt and install SSH identity keys (for git auth)
|
|
||||||
# Uses hardware serial from /proc/cmdline as device identity and decryption key
|
|
||||||
serial=$(sed 's/.*androidboot.serialno=\([^ ]*\).*/\1/' /proc/cmdline)
|
|
||||||
ssh_dir="/data/ssh/.ssh"
|
|
||||||
if [[ $serial == 3889765b ]] && [[ ! -f "$ssh_dir/id_ed25519" || ! -f "$ssh_dir/id_ed25519.pub" ]]; then
|
|
||||||
echo "Decrypting SSH identity keys (serial=$serial)..."
|
|
||||||
tmpdir=$(mktemp -d)
|
|
||||||
bash /data/openpilot/system/clearpilot/tools/decrypt /data/openpilot/system/clearpilot/dev/id_ed25519.cpt "$tmpdir/id_ed25519"
|
|
||||||
bash /data/openpilot/system/clearpilot/tools/decrypt /data/openpilot/system/clearpilot/dev/id_ed25519.pub.cpt "$tmpdir/id_ed25519.pub"
|
|
||||||
mkdir -p "$ssh_dir"
|
|
||||||
cp "$tmpdir/id_ed25519" "$tmpdir/id_ed25519.pub" "$ssh_dir/"
|
|
||||||
rm -rf "$tmpdir"
|
|
||||||
chmod 700 "$ssh_dir"
|
|
||||||
chmod 600 "$ssh_dir/id_ed25519"
|
|
||||||
chmod 644 "$ssh_dir/id_ed25519.pub"
|
|
||||||
echo "SSH identity keys installed to $ssh_dir"
|
|
||||||
fi
|
|
||||||
|
|
||||||
# 4. Ensure git remote uses SSH (not HTTPS)
|
# 4. Ensure git remote uses SSH (not HTTPS)
|
||||||
cd /data/openpilot
|
cd /data/openpilot
|
||||||
EXPECTED_REMOTE="git@git.hanson.xyz:brianhansonxyz/clearpilot.git"
|
EXPECTED_REMOTE="git@git.hanson.xyz:brianhansonxyz/clearpilot.git"
|
||||||
|
|||||||
Reference in New Issue
Block a user