887b9c9e12
Adds a second controlsd variant that runs while ignition is on but the car is in Park. It only listens to CAN and publishes carState — no model, no planner, no lateral/long control, no actuator commands — so modeld, locationd, calibrationd, plannerd, radard, paramsd, torqued, dmonitoring*, soundd, loggerd all stay stopped while parked. Manager swaps between the two via mutually-exclusive predicates: - controlsd_parked: ignition AND not started - controlsd (full): started (= ignition AND not_parked) Thermald owns the swap. It already subscribes to carState; we add a new onroad condition `not_parked` derived from gearShifter, with a 1.5s hysteresis on going into parked (R/P/D thrash protection) and zero hysteresis on going out (instant wake on shift to D/R/N). At boot we assume parked so the heavy stack waits for carState to confirm gear has actually left Park. Manager predicates can only see persistent Params, not pandaStates, so thermald exposes ignition as a new IgnitionOn param (edge-written). Reverse is treated as not-parked — driver is moving. Files: - selfdrive/controls/controlsd_parked.py (new, ~50 lines) - selfdrive/thermald/thermald.py: not_parked condition + IgnitionOn - selfdrive/manager/process_config.py: parked_only predicate + entry - selfdrive/manager/manager.py: seed IgnitionOn=False - common/params.cc: register IgnitionOn The full controlsd is unchanged. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
55 lines
1.9 KiB
Python
55 lines
1.9 KiB
Python
#!/usr/bin/env python3
|
|
"""
|
|
CLEARPILOT: minimal controlsd variant that runs while ignition is on but the
|
|
car is in Park. Keeps CAN parsing and carState publishing alive (so thermald
|
|
can see gearShifter and decide when to swap us out for the full controlsd),
|
|
but skips all of the heavy onroad work — no model, no planner, no lateral or
|
|
longitudinal control, no actuator commands.
|
|
|
|
Manager swaps between this and the full controlsd via predicate flips:
|
|
- this runs when: ignition AND not started
|
|
- full runs when: started (which requires ignition AND not_parked)
|
|
|
|
The two are mutually exclusive — only one publishes carState at a time.
|
|
"""
|
|
from types import SimpleNamespace
|
|
|
|
from openpilot.common.realtime import Priority, config_realtime_process
|
|
from openpilot.selfdrive.car.card import CarD
|
|
|
|
|
|
def _make_default_frogpilot_variables() -> SimpleNamespace:
|
|
"""Safe defaults for fields read inside CarInterface.update / CarState.update.
|
|
|
|
We're not actuating anything here; these only need to keep the update path
|
|
from raising AttributeError. False/0 across the board is the safe baseline."""
|
|
fv = SimpleNamespace()
|
|
fv.conditional_experimental_mode = False
|
|
fv.experimental_mode_via_distance = False
|
|
fv.traffic_mode = False
|
|
fv.sport_plus = False
|
|
fv.long_pitch = False
|
|
fv.no_lat_lane_change = False
|
|
return fv
|
|
|
|
|
|
def main():
|
|
config_realtime_process(4, Priority.CTRL_HIGH)
|
|
|
|
# CarD's __init__ blocks until it sees CAN + a pandaState, then calls get_car
|
|
# to fingerprint and write CarParams. Same path the full controlsd takes.
|
|
card = CarD()
|
|
card.initialize()
|
|
|
|
fv = _make_default_frogpilot_variables()
|
|
|
|
# state_update drains CAN, parses carState, publishes carState/carOutput/carParams.
|
|
# Internally blocks via drain_sock_raw(wait_for_one=True), so the loop is
|
|
# naturally paced by CAN traffic — no extra sleep needed.
|
|
while True:
|
|
card.state_update(fv)
|
|
|
|
|
|
if __name__ == "__main__":
|
|
main()
|