forked from commaai/opendbc
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge commit '06c51c1db49c3b4ffb9144ba010b84ce186d5c5b'
- Loading branch information
Showing
3 changed files
with
170 additions
and
16 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,146 @@ | ||
import os | ||
import math | ||
import hypothesis.strategies as st | ||
from hypothesis import Phase, given, settings | ||
import importlib | ||
from parameterized import parameterized | ||
from collections.abc import Callable | ||
from typing import Any | ||
|
||
from opendbc.car import DT_CTRL, CanData, gen_empty_fingerprint, structs | ||
from opendbc.car.car_helpers import interfaces | ||
from opendbc.car.fingerprints import all_known_cars | ||
from opendbc.car.fw_versions import FW_VERSIONS, FW_QUERY_CONFIGS | ||
from opendbc.car.interfaces import get_interface_attr | ||
from opendbc.car.mock.values import CAR as MOCK | ||
|
||
DrawType = Callable[[st.SearchStrategy], Any] | ||
|
||
ALL_ECUS = {ecu for ecus in FW_VERSIONS.values() for ecu in ecus.keys()} | ||
ALL_ECUS |= {ecu for config in FW_QUERY_CONFIGS.values() for ecu in config.extra_ecus} | ||
|
||
ALL_REQUESTS = {tuple(r.request) for config in FW_QUERY_CONFIGS.values() for r in config.requests} | ||
|
||
MAX_EXAMPLES = int(os.environ.get('MAX_EXAMPLES', '5')) | ||
|
||
|
||
def get_fuzzy_car_interface_args(draw: DrawType) -> dict: | ||
# Fuzzy CAN fingerprints and FW versions to test more states of the CarInterface | ||
fingerprint_strategy = st.fixed_dictionaries({key: st.dictionaries(st.integers(min_value=0, max_value=0x800), | ||
st.integers(min_value=0, max_value=64)) for key in | ||
gen_empty_fingerprint()}) | ||
|
||
# only pick from possible ecus to reduce search space | ||
car_fw_strategy = st.lists(st.sampled_from(sorted(ALL_ECUS))) | ||
|
||
params_strategy = st.fixed_dictionaries({ | ||
'fingerprints': fingerprint_strategy, | ||
'car_fw': car_fw_strategy, | ||
'experimental_long': st.booleans(), | ||
}) | ||
|
||
params: dict = draw(params_strategy) | ||
params['car_fw'] = [structs.CarParams.CarFw(ecu=fw[0], address=fw[1], subAddress=fw[2] or 0, | ||
request=draw(st.sampled_from(sorted(ALL_REQUESTS)))) | ||
for fw in params['car_fw']] | ||
return params | ||
|
||
|
||
class TestCarInterfaces: | ||
# FIXME: Due to the lists used in carParams, Phase.target is very slow and will cause | ||
# many generated examples to overrun when max_examples > ~20, don't use it | ||
@parameterized.expand([(car,) for car in sorted(all_known_cars())] + [MOCK.MOCK]) | ||
@settings(max_examples=MAX_EXAMPLES, deadline=None, | ||
phases=(Phase.reuse, Phase.generate, Phase.shrink)) | ||
@given(data=st.data()) | ||
def test_car_interfaces(self, car_name, data): | ||
CarInterface, CarController, CarState = interfaces[car_name] | ||
|
||
args = get_fuzzy_car_interface_args(data.draw) | ||
|
||
car_params = CarInterface.get_params(car_name, args['fingerprints'], args['car_fw'], | ||
experimental_long=args['experimental_long'], docs=False) | ||
car_interface = CarInterface(car_params, CarController, CarState) | ||
assert car_params | ||
assert car_interface | ||
|
||
assert car_params.mass > 1 | ||
assert car_params.wheelbase > 0 | ||
# centerToFront is center of gravity to front wheels, assert a reasonable range | ||
assert car_params.wheelbase * 0.3 < car_params.centerToFront < car_params.wheelbase * 0.7 | ||
assert car_params.maxLateralAccel > 0 | ||
|
||
# Longitudinal sanity checks | ||
assert len(car_params.longitudinalTuning.kpV) == len(car_params.longitudinalTuning.kpBP) | ||
assert len(car_params.longitudinalTuning.kiV) == len(car_params.longitudinalTuning.kiBP) | ||
|
||
# Lateral sanity checks | ||
if car_params.steerControlType != structs.CarParams.SteerControlType.angle: | ||
tune = car_params.lateralTuning | ||
if tune.which() == 'pid': | ||
if car_name != MOCK.MOCK: | ||
assert not math.isnan(tune.pid.kf) and tune.pid.kf > 0 | ||
assert len(tune.pid.kpV) > 0 and len(tune.pid.kpV) == len(tune.pid.kpBP) | ||
assert len(tune.pid.kiV) > 0 and len(tune.pid.kiV) == len(tune.pid.kiBP) | ||
|
||
elif tune.which() == 'torque': | ||
assert not math.isnan(tune.torque.kf) and tune.torque.kf > 0 | ||
assert not math.isnan(tune.torque.friction) and tune.torque.friction > 0 | ||
|
||
# Run car interface | ||
# TODO: use hypothesis to generate random messages | ||
now_nanos = 0 | ||
CC = structs.CarControl() | ||
for _ in range(10): | ||
car_interface.update([]) | ||
car_interface.apply(CC, now_nanos) | ||
now_nanos += DT_CTRL * 1e9 # 10 ms | ||
|
||
CC = structs.CarControl() | ||
CC.enabled = True | ||
for _ in range(10): | ||
car_interface.update([]) | ||
car_interface.apply(CC, now_nanos) | ||
now_nanos += DT_CTRL * 1e9 # 10ms | ||
|
||
# Test radar interface | ||
RadarInterface = importlib.import_module(f'opendbc.car.{car_params.carName}.radar_interface').RadarInterface | ||
radar_interface = RadarInterface(car_params) | ||
assert radar_interface | ||
|
||
# Run radar interface once | ||
radar_interface.update([]) | ||
if not car_params.radarUnavailable and radar_interface.rcp is not None and \ | ||
hasattr(radar_interface, '_update') and hasattr(radar_interface, 'trigger_msg'): | ||
radar_interface._update([radar_interface.trigger_msg]) | ||
|
||
# Test radar fault | ||
if not car_params.radarUnavailable and radar_interface.rcp is not None: | ||
cans = [(0, [CanData(0, b'', 0) for _ in range(5)])] | ||
rr = radar_interface.update(cans) | ||
assert rr is None or len(rr.errors) > 0 | ||
|
||
def test_interface_attrs(self): | ||
"""Asserts basic behavior of interface attribute getter""" | ||
num_brands = len(get_interface_attr('CAR')) | ||
assert num_brands >= 12 | ||
|
||
# Should return value for all brands when not combining, even if attribute doesn't exist | ||
ret = get_interface_attr('FAKE_ATTR') | ||
assert len(ret) == num_brands | ||
|
||
# Make sure we can combine dicts | ||
ret = get_interface_attr('DBC', combine_brands=True) | ||
assert len(ret) >= 160 | ||
|
||
# We don't support combining non-dicts | ||
ret = get_interface_attr('CAR', combine_brands=True) | ||
assert len(ret) == 0 | ||
|
||
# If brand has None value, it shouldn't return when ignore_none=True is specified | ||
none_brands = {b for b, v in get_interface_attr('FINGERPRINTS').items() if v is None} | ||
assert len(none_brands) >= 1 | ||
|
||
ret = get_interface_attr('FINGERPRINTS', ignore_none=True) | ||
none_brands_in_ret = none_brands.intersection(ret) | ||
assert len(none_brands_in_ret) == 0, f'Brands with None values in ignore_none=True result: {none_brands_in_ret}' |