You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
110 lines
3.9 KiB
110 lines
3.9 KiB
5 years ago
|
import os
|
||
2 years ago
|
import copy
|
||
5 years ago
|
import random
|
||
|
import time
|
||
12 months ago
|
import pytest
|
||
5 years ago
|
from collections import defaultdict
|
||
2 years ago
|
from pprint import pprint
|
||
5 years ago
|
|
||
5 years ago
|
import cereal.messaging as messaging
|
||
2 years ago
|
from cereal import car, log
|
||
11 months ago
|
from openpilot.common.retry import retry
|
||
2 years ago
|
from openpilot.common.params import Params
|
||
|
from openpilot.common.timeout import Timeout
|
||
11 months ago
|
from openpilot.selfdrive.pandad import can_list_to_can_capnp
|
||
2 years ago
|
from openpilot.selfdrive.car import make_can_msg
|
||
|
from openpilot.system.hardware import TICI
|
||
|
from openpilot.selfdrive.test.helpers import phone_only, with_processes
|
||
5 years ago
|
|
||
|
|
||
11 months ago
|
@retry(attempts=3)
|
||
11 months ago
|
def setup_pandad(num_pandas):
|
||
12 months ago
|
params = Params()
|
||
|
params.put_bool("IsOnroad", False)
|
||
4 years ago
|
|
||
11 months ago
|
with Timeout(90, "pandad didn't start"):
|
||
12 months ago
|
sm = messaging.SubMaster(['pandaStates'])
|
||
|
while sm.recv_frame['pandaStates'] < 1 or len(sm['pandaStates']) == 0 or \
|
||
|
any(ps.pandaType == log.PandaState.PandaType.unknown for ps in sm['pandaStates']):
|
||
|
sm.update(1000)
|
||
4 years ago
|
|
||
12 months ago
|
found_pandas = len(sm['pandaStates'])
|
||
|
assert num_pandas == found_pandas, "connected pandas ({found_pandas}) doesn't match expected panda count ({num_pandas}). \
|
||
|
connect another panda for multipanda tests."
|
||
|
|
||
11 months ago
|
# pandad safety setting relies on these params
|
||
12 months ago
|
cp = car.CarParams.new_message()
|
||
4 years ago
|
|
||
12 months ago
|
safety_config = car.CarParams.SafetyConfig.new_message()
|
||
|
safety_config.safetyModel = car.CarParams.SafetyModel.allOutput
|
||
|
cp.safetyConfigs = [safety_config]*num_pandas
|
||
4 years ago
|
|
||
12 months ago
|
params.put_bool("IsOnroad", True)
|
||
|
params.put_bool("FirmwareQueryDone", True)
|
||
|
params.put_bool("ControlsReady", True)
|
||
|
params.put("CarParams", cp.to_bytes())
|
||
3 years ago
|
|
||
4 years ago
|
|
||
12 months ago
|
def send_random_can_messages(sendcan, count, num_pandas=1):
|
||
|
sent_msgs = defaultdict(set)
|
||
|
for _ in range(count):
|
||
|
to_send = []
|
||
|
for __ in range(random.randrange(20)):
|
||
|
bus = random.choice([b for b in range(3*num_pandas) if b % 4 != 3])
|
||
|
addr = random.randrange(1, 1<<29)
|
||
|
dat = bytes(random.getrandbits(8) for _ in range(random.randrange(1, 9)))
|
||
|
if (addr, dat) in sent_msgs[bus]:
|
||
|
continue
|
||
|
sent_msgs[bus].add((addr, dat))
|
||
|
to_send.append(make_can_msg(addr, dat, bus))
|
||
|
sendcan.send(can_list_to_can_capnp(to_send, msgtype='sendcan'))
|
||
|
return sent_msgs
|
||
4 years ago
|
|
||
|
|
||
12 months ago
|
@pytest.mark.tici
|
||
|
class TestBoarddLoopback:
|
||
|
@classmethod
|
||
|
def setup_class(cls):
|
||
|
os.environ['STARTED'] = '1'
|
||
|
os.environ['BOARDD_LOOPBACK'] = '1'
|
||
|
|
||
|
@phone_only
|
||
|
@with_processes(['pandad'])
|
||
|
def test_loopback(self):
|
||
|
num_pandas = 2 if TICI and "SINGLE_PANDA" not in os.environ else 1
|
||
11 months ago
|
setup_pandad(num_pandas)
|
||
4 years ago
|
sendcan = messaging.pub_sock('sendcan')
|
||
|
can = messaging.sub_sock('can', conflate=False, timeout=100)
|
||
2 years ago
|
sm = messaging.SubMaster(['pandaStates'])
|
||
2 years ago
|
time.sleep(0.5)
|
||
4 years ago
|
|
||
3 years ago
|
n = 200
|
||
4 years ago
|
for i in range(n):
|
||
11 months ago
|
print(f"pandad loopback {i}/{n}")
|
||
4 years ago
|
|
||
12 months ago
|
sent_msgs = send_random_can_messages(sendcan, random.randrange(20, 100), num_pandas)
|
||
4 years ago
|
|
||
2 years ago
|
sent_loopback = copy.deepcopy(sent_msgs)
|
||
|
sent_loopback.update({k+128: copy.deepcopy(v) for k, v in sent_msgs.items()})
|
||
|
sent_total = {k: len(v) for k, v in sent_loopback.items()}
|
||
|
for _ in range(100 * 5):
|
||
2 years ago
|
sm.update(0)
|
||
4 years ago
|
recvd = messaging.drain_sock(can, wait_for_one=True)
|
||
|
for msg in recvd:
|
||
|
for m in msg.can:
|
||
2 years ago
|
key = (m.address, m.dat)
|
||
|
assert key in sent_loopback[m.src], f"got unexpected msg: {m.src=} {m.address=} {m.dat=}"
|
||
|
sent_loopback[m.src].discard(key)
|
||
3 years ago
|
|
||
2 years ago
|
if all(len(v) == 0 for v in sent_loopback.values()):
|
||
3 years ago
|
break
|
||
4 years ago
|
|
||
|
# if a set isn't empty, messages got dropped
|
||
2 years ago
|
pprint(sent_msgs)
|
||
|
pprint(sent_loopback)
|
||
2 years ago
|
print({k: len(x) for k, x in sent_loopback.items()})
|
||
|
print(sum([len(x) for x in sent_loopback.values()]))
|
||
|
pprint(sm['pandaStates']) # may drop messages due to RX buffer overflow
|
||
2 years ago
|
for bus in sent_loopback.keys():
|
||
|
assert not len(sent_loopback[bus]), f"loop {i}: bus {bus} missing {len(sent_loopback[bus])} out of {sent_total[bus]} messages"
|