forked from PAWPAW-Mirror/lib_xua
Add tmpdirs and initial build so test works with xdist
This commit is contained in:
1
.gitignore
vendored
1
.gitignore
vendored
@@ -50,3 +50,4 @@ host_usb_mixer_control/xmos_mixer
|
|||||||
*tests/logs/*
|
*tests/logs/*
|
||||||
midi_tx_cmds.txt
|
midi_tx_cmds.txt
|
||||||
midi_rx_cmds.txt
|
midi_rx_cmds.txt
|
||||||
|
trace.txt
|
||||||
|
|||||||
@@ -2,7 +2,9 @@
|
|||||||
# This Software is subject to the terms of the XMOS Public Licence: Version 1.
|
# This Software is subject to the terms of the XMOS Public Licence: Version 1.
|
||||||
import pytest
|
import pytest
|
||||||
import time
|
import time
|
||||||
|
import Pyxsim
|
||||||
|
from pathlib import Path
|
||||||
|
from midi_test_helpers import MIDI_TEST_CONFIGS
|
||||||
|
|
||||||
@pytest.fixture()
|
@pytest.fixture()
|
||||||
def test_file(request):
|
def test_file(request):
|
||||||
@@ -39,3 +41,18 @@ def pytest_addoption(parser):
|
|||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
def options(request):
|
def options(request):
|
||||||
yield request.config.option
|
yield request.config.option
|
||||||
|
|
||||||
|
# We use the same binary multiple times so just build once
|
||||||
|
@pytest.fixture(scope="session")
|
||||||
|
def build_midi():
|
||||||
|
all_build_success = True
|
||||||
|
for config in MIDI_TEST_CONFIGS:
|
||||||
|
xe = str(Path(__file__).parent / f"test_midi/bin/{config}/test_midi_{config}.xe")
|
||||||
|
|
||||||
|
success, output = Pyxsim._build(xe, build_options=["-j"])
|
||||||
|
all_build_success |= success
|
||||||
|
|
||||||
|
if not all_build_success:
|
||||||
|
print(f"ERROR MIDI Build failed: {output}")
|
||||||
|
|
||||||
|
return str(Path(__file__).parent / f"test_midi/bin/") if all_build_success else False
|
||||||
|
|||||||
@@ -1,6 +1,32 @@
|
|||||||
# Copyright 2024 XMOS LIMITED.
|
# Copyright 2024 XMOS LIMITED.
|
||||||
# This Software is subject to the terms of the XMOS Public Licence: Version 1.
|
# This Software is subject to the terms of the XMOS Public Licence: Version 1.
|
||||||
|
|
||||||
|
import contextlib
|
||||||
|
import os
|
||||||
|
import shutil
|
||||||
|
import tempfile
|
||||||
|
|
||||||
|
MIDI_TEST_CONFIGS = ["xs2", "xs3"]
|
||||||
|
MIDI_RATE = 31250
|
||||||
|
|
||||||
|
@contextlib.contextmanager
|
||||||
|
def cd(newdir, cleanup=lambda: True):
|
||||||
|
prevdir = os.getcwd()
|
||||||
|
os.chdir(os.path.expanduser(newdir))
|
||||||
|
try:
|
||||||
|
yield
|
||||||
|
finally:
|
||||||
|
os.chdir(prevdir)
|
||||||
|
cleanup()
|
||||||
|
|
||||||
|
@contextlib.contextmanager
|
||||||
|
def tempdir():
|
||||||
|
dirpath = tempfile.mkdtemp()
|
||||||
|
def cleanup():
|
||||||
|
shutil.rmtree(dirpath)
|
||||||
|
with cd(dirpath, cleanup):
|
||||||
|
yield dirpath
|
||||||
|
|
||||||
class midi_expect_tx:
|
class midi_expect_tx:
|
||||||
def __init(self):
|
def __init(self):
|
||||||
pass
|
pass
|
||||||
@@ -12,7 +38,7 @@ class midi_expect_tx:
|
|||||||
command.append(0)
|
command.append(0)
|
||||||
expected += "uart_tx_checker: " + " ".join([f"0x{byte:02x}" for byte in command]) + "\n"
|
expected += "uart_tx_checker: " + " ".join([f"0x{byte:02x}" for byte in command]) + "\n"
|
||||||
|
|
||||||
return expected
|
return expected + "\n"
|
||||||
|
|
||||||
class midi_expect_rx:
|
class midi_expect_rx:
|
||||||
def __init(self):
|
def __init(self):
|
||||||
@@ -25,7 +51,7 @@ class midi_expect_rx:
|
|||||||
command.append(0)
|
command.append(0)
|
||||||
expected += "dut_midi_rx: " + " ".join([f"{byte}" for byte in command]) + "\n"
|
expected += "dut_midi_rx: " + " ".join([f"{byte}" for byte in command]) + "\n"
|
||||||
|
|
||||||
return expected
|
return expected + "\n"
|
||||||
|
|
||||||
midi_tx_file = "midi_tx_cmds.txt"
|
midi_tx_file = "midi_tx_cmds.txt"
|
||||||
midi_rx_file = "midi_rx_cmds.txt"
|
midi_rx_file = "midi_rx_cmds.txt"
|
||||||
@@ -44,3 +70,12 @@ def create_midi_rx_file(num_commands=0):
|
|||||||
with open(midi_rx_file, "wt") as mr:
|
with open(midi_rx_file, "wt") as mr:
|
||||||
text = f"{num_commands}\n"
|
text = f"{num_commands}\n"
|
||||||
mr.write(text)
|
mr.write(text)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
# Test/dev only
|
||||||
|
if __name__ == "__main__":
|
||||||
|
with tempdir() as td:
|
||||||
|
print(td)
|
||||||
|
create_midi_tx_file()
|
||||||
|
input("PRESS ENTER TO CONTINUE")
|
||||||
|
|||||||
@@ -6,66 +6,63 @@ import Pyxsim
|
|||||||
from Pyxsim import testers
|
from Pyxsim import testers
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from uart_rx_checker import UARTRxChecker
|
from uart_rx_checker import UARTRxChecker
|
||||||
from midi_test_helpers import midi_expect_rx, create_midi_rx_file, create_midi_tx_file
|
from midi_test_helpers import midi_expect_rx, create_midi_rx_file, create_midi_tx_file, tempdir, MIDI_TEST_CONFIGS, MIDI_RATE
|
||||||
|
from distutils.dir_util import copy_tree # we're using python 3.7 and dirs_exist_ok=True isn't available until 3.8 :(
|
||||||
|
|
||||||
MAX_CYCLES = 15000000
|
MAX_CYCLES = 15000000
|
||||||
MIDI_RATE = 31250
|
|
||||||
CONFIGS = ["xs2", "xs3"]
|
|
||||||
CONFIGS = ["xs3"]
|
|
||||||
|
|
||||||
|
|
||||||
#####
|
#####
|
||||||
# This test builds the spdif transmitter app with a verity of presets and tests that the output matches those presets
|
# This test takes the built binary, copies it to a tmp dir and runs the midi Rx test which sends some commands
|
||||||
|
# to using the UARTRX checker and the firmware receives them
|
||||||
#####
|
#####
|
||||||
@pytest.mark.parametrize("config", CONFIGS)
|
@pytest.mark.parametrize("config", MIDI_TEST_CONFIGS)
|
||||||
def test_rx(capfd, config):
|
def test_rx(capfd, config, build_midi):
|
||||||
xe = str(Path(__file__).parent / f"test_midi/bin/{config}/test_midi_{config}.xe")
|
# Need tempdir as we use the same config files and this causes issues when using xdist
|
||||||
|
with tempdir() as tmpdirname:
|
||||||
|
copy_tree(build_midi, tmpdirname)
|
||||||
|
xe = str(Path(tmpdirname) / f"{config}/test_midi_{config}.xe")
|
||||||
|
|
||||||
midi_commands = [[0x90, 60, 81]]
|
midi_commands = [[0x90, 60, 81]]
|
||||||
create_midi_rx_file(1)
|
create_midi_rx_file(1)
|
||||||
create_midi_tx_file()
|
create_midi_tx_file()
|
||||||
|
|
||||||
|
expected = midi_expect_rx().expect(midi_commands)
|
||||||
|
tester = testers.ComparisonTester(midi_expect_rx().expect(midi_commands),
|
||||||
|
ordered = True)
|
||||||
|
|
||||||
|
rx_port = "tile[1]:XS1_PORT_1F"
|
||||||
|
tx_port = "tile[1]:XS1_PORT_4C" # Needed so that UARTRxChecker (a transmitter) knows when to start
|
||||||
|
baud = MIDI_RATE
|
||||||
|
bpb = 8
|
||||||
|
parity = 0
|
||||||
|
stop = 1
|
||||||
|
|
||||||
tester = testers.ComparisonTester(midi_expect_rx().expect(midi_commands),
|
midi_commands_flattened = [item for row in midi_commands for item in row]
|
||||||
regexp = "uart_tx_checker:.+",
|
# midi_commands_flattened.append(0x00) # send a null afterwards to give RXChecker to complete
|
||||||
ordered = True)
|
|
||||||
|
|
||||||
rx_port = "tile[1]:XS1_PORT_1F"
|
|
||||||
tx_port = "tile[1]:XS1_PORT_4C" # Needed so that UARTRxChecker (a transmitter) knows when to start
|
|
||||||
baud = MIDI_RATE
|
|
||||||
bpb = 8
|
|
||||||
parity = 0
|
|
||||||
stop = 1
|
|
||||||
|
|
||||||
midi_commands_flattened = [item for row in midi_commands for item in row]
|
simthreads = [
|
||||||
# midi_commands_flattened.append(0x00) # send a null afterwards to give RXChecker to complete
|
UARTRxChecker(tx_port, rx_port, parity, baud, stop, bpb, midi_commands_flattened, debug=False)
|
||||||
|
]
|
||||||
|
|
||||||
simthreads = [
|
simargs = ["--max-cycles", str(MAX_CYCLES)]
|
||||||
UARTRxChecker(tx_port, rx_port, parity, baud, stop, bpb, midi_commands_flattened, debug=False)
|
#This is just for local debug so we can capture the traces if needed. It slows xsim down so not needed
|
||||||
]
|
# simargs.extend(["--trace-to", "trace.txt", "--vcd-tracing", "-tile tile[1] -ports -o trace.vcd"])
|
||||||
|
|
||||||
simargs = ["--max-cycles", str(MAX_CYCLES)]
|
# Print to console
|
||||||
simargs.extend(["--trace-to", "trace.txt", "--vcd-tracing", "-tile tile[1] -ports -o trace.vcd"]) #This is just for local debug so we can capture the run, pass as kwarg to run_with_pyxsim
|
# with capfd.disabled():
|
||||||
|
# print("++++", expected, "++++")
|
||||||
|
# print("****", capture, "****")
|
||||||
|
# print("****", capture.split("\n"), "****")
|
||||||
|
# print(xe)
|
||||||
|
|
||||||
# result = Pyxsim.run_on_simulator(
|
Pyxsim.run_with_pyxsim(
|
||||||
result = Pyxsim.run_on_simulator(
|
xe,
|
||||||
xe,
|
simthreads=simthreads,
|
||||||
simthreads=simthreads,
|
timeout=1200,
|
||||||
instTracing=True,
|
simargs=simargs,
|
||||||
# clean_before_build=True,
|
)
|
||||||
clean_before_build=False,
|
capture = capfd.readouterr().out
|
||||||
tester=tester,
|
result = tester.run(capture.split("\n"))
|
||||||
capfd=capfd,
|
|
||||||
# capfd=None,
|
|
||||||
timeout=120,
|
|
||||||
simargs=simargs,
|
|
||||||
build_options=[
|
|
||||||
"-j",
|
|
||||||
f"CONFIG={config}",
|
|
||||||
"EXTRA_BUILD_FLAGS="
|
|
||||||
+ f" -DMIDI_RATE_HZ={MIDI_RATE}"
|
|
||||||
,
|
|
||||||
],
|
|
||||||
)
|
|
||||||
|
|
||||||
assert result
|
assert result
|
||||||
@@ -6,62 +6,61 @@ import Pyxsim
|
|||||||
from Pyxsim import testers
|
from Pyxsim import testers
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from uart_tx_checker import UARTTxChecker
|
from uart_tx_checker import UARTTxChecker
|
||||||
from midi_test_helpers import midi_expect_tx, create_midi_tx_file
|
from midi_test_helpers import midi_expect_tx, create_midi_tx_file, create_midi_rx_file, tempdir, MIDI_TEST_CONFIGS, MIDI_RATE
|
||||||
|
from distutils.dir_util import copy_tree # we're using python 3.7 and dirs_exist_ok=True isn't available until 3.8 :(
|
||||||
|
|
||||||
MAX_CYCLES = 15000000
|
MAX_CYCLES = 15000000
|
||||||
MIDI_RATE = 31250
|
|
||||||
CONFIGS = ["xs2", "xs3"]
|
|
||||||
CONFIGS = ["xs3"]
|
|
||||||
|
|
||||||
|
|
||||||
#####
|
#####
|
||||||
# This test builds the spdif transmitter app with a verity of presets and tests that the output matches those presets
|
# This test takes the built binary, copies it to a tmp dir and runs the midi Tx test which sends some commands
|
||||||
|
# to the firmware and then receives them using the UARTTX checker
|
||||||
#####
|
#####
|
||||||
@pytest.mark.parametrize("config", CONFIGS)
|
@pytest.mark.parametrize("config", MIDI_TEST_CONFIGS)
|
||||||
def test_tx(capfd, config):
|
def test_tx(capfd, config, build_midi):
|
||||||
xe = str(Path(__file__).parent / f"test_midi/bin/{config}/test_midi_{config}.xe")
|
|
||||||
|
|
||||||
midi_commands = [[0x90, 60, 81]]
|
# Need tempdir as we use the same config files and this causes issues when using xdist
|
||||||
create_midi_tx_file(midi_commands)
|
with tempdir() as tmpdirname:
|
||||||
|
copy_tree(build_midi, tmpdirname)
|
||||||
|
xe = str(Path(tmpdirname) / f"{config}/test_midi_{config}.xe")
|
||||||
|
|
||||||
tester = testers.ComparisonTester(midi_expect_tx().expect(midi_commands),
|
midi_commands = [[0x90, 60, 81]]
|
||||||
regexp = "uart_tx_checker:.+",
|
create_midi_tx_file(midi_commands)
|
||||||
ordered = True)
|
create_midi_rx_file()
|
||||||
|
|
||||||
|
expected = midi_expect_tx().expect(midi_commands)
|
||||||
tx_port = "tile[1]:XS1_PORT_4C"
|
tester = testers.ComparisonTester( expected,
|
||||||
baud = MIDI_RATE
|
ordered = True)
|
||||||
bpb = 8
|
|
||||||
parity = 0
|
|
||||||
stop = 1
|
|
||||||
length_of_test = sum(len(cmd) for cmd in midi_commands)
|
|
||||||
|
|
||||||
simthreads = [
|
tx_port = "tile[1]:XS1_PORT_4C"
|
||||||
UARTTxChecker(tx_port, parity, baud, length_of_test, stop, bpb, debug=False)
|
baud = MIDI_RATE
|
||||||
]
|
bpb = 8
|
||||||
|
parity = 0
|
||||||
|
stop = 1
|
||||||
|
length_of_test = sum(len(cmd) for cmd in midi_commands)
|
||||||
|
|
||||||
simargs = ["--max-cycles", str(MAX_CYCLES)]
|
simthreads = [
|
||||||
# simargs.extend(["--trace-to", "trace.txt", "--vcd-tracing", "-tile tile[1] -ports -o trace.vcd"]) #This is just for local debug so we can capture the run, pass as kwarg to run_with_pyxsim
|
UARTTxChecker(tx_port, parity, baud, length_of_test, stop, bpb, debug=False)
|
||||||
|
]
|
||||||
|
|
||||||
# result = Pyxsim.run_on_simulator(
|
|
||||||
result = Pyxsim.run_on_simulator(
|
|
||||||
xe,
|
|
||||||
simthreads=simthreads,
|
|
||||||
instTracing=True,
|
|
||||||
# clean_before_build=True,
|
|
||||||
clean_before_build=False,
|
|
||||||
tester=tester,
|
|
||||||
capfd=capfd,
|
|
||||||
# capfd=None,
|
|
||||||
timeout=120,
|
|
||||||
simargs=simargs,
|
|
||||||
build_options=[
|
|
||||||
"-j",
|
|
||||||
f"CONFIG={config}",
|
|
||||||
"EXTRA_BUILD_FLAGS="
|
|
||||||
+ f" -DMIDI_RATE_HZ={MIDI_RATE}"
|
|
||||||
,
|
|
||||||
],
|
|
||||||
)
|
|
||||||
|
|
||||||
assert result
|
simargs = ["--max-cycles", str(MAX_CYCLES)]
|
||||||
|
#This is just for local debug so we can capture the traces if needed. It slows xsim down so not needed
|
||||||
|
# simargs.extend(["--trace-to", "trace.txt", "--vcd-tracing", "-tile tile[1] -ports -o trace.vcd"])
|
||||||
|
|
||||||
|
Pyxsim.run_with_pyxsim(
|
||||||
|
xe,
|
||||||
|
simthreads=simthreads,
|
||||||
|
timeout=1200,
|
||||||
|
simargs=simargs,
|
||||||
|
)
|
||||||
|
capture = capfd.readouterr().out
|
||||||
|
result = tester.run(capture.split("\n"))
|
||||||
|
|
||||||
|
# Print to console
|
||||||
|
# with capfd.disabled():
|
||||||
|
# print("++++", expected, "++++")
|
||||||
|
# print("****", capture, "****")
|
||||||
|
# print("****", capture.split("\n"), "****")
|
||||||
|
# print(xe)
|
||||||
|
|
||||||
|
assert result
|
||||||
Reference in New Issue
Block a user