Files
StarPilot/selfdrive/controls/tests/test_leads.py
T
firestar5683 8a4d2b558b radar vs voacc
2026-06-14 17:06:52 -05:00

104 lines
3.7 KiB
Python

from types import SimpleNamespace
import cereal.messaging as messaging
from cereal import log
from opendbc.car.toyota.values import CAR as TOYOTA
from openpilot.selfdrive.test.process_replay import replay_process_with_name
from openpilot.selfdrive.controls.radard import (
g90_low_speed_radar_lead_sane,
g90_radar_lead_lateral_sane,
match_vision_to_track,
)
class TestLeads:
@staticmethod
def make_track(d_rel: float, y_rel: float, v_rel: float, cnt: int = 5):
return SimpleNamespace(dRel=d_rel, yRel=y_rel, vRel=v_rel, cnt=cnt)
@staticmethod
def make_lead(x: float, y: float, v: float, x_std: float = 1.5, y_std: float = 0.3, v_std: float = 1.0):
return SimpleNamespace(x=[x], y=[y], v=[v], xStd=[x_std], yStd=[y_std], vStd=[v_std])
@staticmethod
def make_model_data():
model = log.ModelDataV2.new_message()
model.init('laneLines', 4)
model.meta.laneChangeState = 0
model.meta.laneChangeDirection = 0
return model
def test_g90_radar_filters_side_tracks(self):
side_track = SimpleNamespace(dRel=13.0, yRel=-10.38, cnt=10)
centered_track = SimpleNamespace(dRel=10.8, yRel=-0.21, cnt=5)
close_side_ghost = SimpleNamespace(dRel=2.2, yRel=2.41, cnt=10)
close_centered_track = SimpleNamespace(dRel=2.2, yRel=1.2, cnt=10)
far_low_speed_track = SimpleNamespace(dRel=15.5, yRel=0.58, cnt=5)
assert not g90_radar_lead_lateral_sane(side_track)
assert g90_radar_lead_lateral_sane(centered_track)
assert not g90_radar_lead_lateral_sane(close_side_ghost)
assert g90_radar_lead_lateral_sane(close_centered_track)
assert g90_low_speed_radar_lead_sane(centered_track, 2.0)
assert not g90_low_speed_radar_lead_sane(far_low_speed_track, 3.5)
def test_match_vision_to_track_sticky_holds_preferred_track(self):
v_ego = 21.2
lead = self.make_lead(x=37.72, y=0.07, v=20.03)
preferred_track = self.make_track(d_rel=46.95, y_rel=0.05, v_rel=-0.58, cnt=8)
tracks = {3647: preferred_track}
track = match_vision_to_track(
v_ego,
lead,
self.make_model_data(),
tracks,
SimpleNamespace(human_lane_changes=False),
preferred_track_id=3647,
)
assert track is preferred_track
def test_match_vision_to_track_sticky_rejects_bad_preferred_track(self):
v_ego = 21.2
lead = self.make_lead(x=37.72, y=0.07, v=20.03)
preferred_track = self.make_track(d_rel=58.0, y_rel=2.8, v_rel=-4.5, cnt=8)
tracks = {3647: preferred_track}
track = match_vision_to_track(
v_ego,
lead,
self.make_model_data(),
tracks,
SimpleNamespace(human_lane_changes=False),
preferred_track_id=3647,
)
assert track is None
def test_radar_fault(self):
# if there's no radar-related can traffic, radard should either not respond or respond with an error
# this is tightly coupled with underlying car radar_interface implementation, but it's a good sanity check
def single_iter_pkg():
# single iter package, with meaningless cans and empty carState/modelV2
msgs = []
for _ in range(500):
can = messaging.new_message("can", 1)
cs = messaging.new_message("carState")
cp = messaging.new_message("carParams")
msgs.append(can.as_reader())
msgs.append(cs.as_reader())
msgs.append(cp.as_reader())
model = messaging.new_message("modelV2")
msgs.append(model.as_reader())
return msgs
msgs = [m for _ in range(3) for m in single_iter_pkg()]
out = replay_process_with_name("card", msgs, fingerprint=TOYOTA.TOYOTA_COROLLA_TSS2)
states = [m for m in out if m.which() == "liveTracks"]
failures = [not state.valid for state in states]
assert len(states) == 0 or all(failures)