Skip to content

Commit

Permalink
test_iostream: refactor SDR input sample test so it can be reused
Browse files Browse the repository at this point in the history
This change makes the SDR input sample test parametrizable.
The previous test relied on i_en having to toggle every clock
cycle, so to remove that requirement we are now toggling the
clk_out output every time i_en is on, and as such the
`save_expected_sample_values_tb` testbench must also look at
both edges of clk_out in order to determine when we're supposed
to sample the signal.

Docstrings have also been improved.
  • Loading branch information
purdeaandrei committed Sep 22, 2024
1 parent b3fed0d commit 3814e22
Showing 1 changed file with 127 additions and 22 deletions.
149 changes: 127 additions & 22 deletions software/tests/gateware/test_iostream.py
Original file line number Diff line number Diff line change
Expand Up @@ -7,31 +7,94 @@
from glasgow.gateware.iostream import IOStreamer

async def stream_get(ctx, stream):
"""
This helper coroutine takes one payload from the stream.
If a payload is not available it waits as many clocks cycles as
needed stream.valid goes high.
"""
ctx.set(stream.ready, 1)
payload, = await ctx.tick().sample(stream.payload).until(stream.valid)
ctx.set(stream.ready, 0)
return payload

async def stream_get_maybe(ctx, stream):
"""
This helper coroutine takes one payload from the stream,
if it is available at the next clock cycle. If a payload is not
available, then time is advanced by only a single clock cycle,
and this coroutine returns None.
"""
ctx.set(stream.ready, 1)
_, _, payload, stream_valid = await ctx.tick().sample(stream.payload, stream.valid)
ctx.set(stream.ready, 0)
if stream_valid:
return payload
else:
return None

async def stream_put(ctx, stream, payload):
"""
This helper coroutine presents a payload to the specified stream,
and waits as many clock cycles as it takes for the payload to be accepted.
"""
ctx.set(stream.payload, payload)
ctx.set(stream.valid, 1)
await ctx.tick().until(stream.ready)
ctx.set(stream.valid, 0)

class IOStreamTimeoutError(Exception):
"""
Custom exception class to make it easy to catch just this exception, for tests
that are expected to fail with a timeout.
"""
pass

class IOStreamTestCase(unittest.TestCase):
def test_sdr_input_sampled_correctly(self):
""" This is a latency-agnostic test, that verifies that the IOStreamer samples the inputs at the same time as the output signals change """
def _subtest_sdr_input_sampled_correctly(self, o_valid_bits, i_en_bits, i_ready_bits, timeout_clocks=None):
"""
This is a latency-agnostic test, that verifies that the IOStreamer samples the inputs at the same time as the output signals change.
o_valid_bits: is a string of "1"s and "0"s. Each character refers to one (or more) clock cycles. "1" means to send a payload,
and "0" means to leave o_stream idle for 1 clock cycle. When sending a payload o_stream is waited upon if it's not ready, so
each "1" might take more than one clock cycle. A "0" always takes a single clock cycle. During the test all the
payloads with "1" are guaranteed to be sent, unless the testcase raises an IOStreamTimeoutError.
i_en_bits: is a string of "1"s and "0"s, and it specifies for each payload, whether to send it with i_en high. The index of the
character in the i_en_bits string matches the index of the character in the o_valid_bits. That is o_valid_bits, and i_en_bits
never go out of sync. Setting i_en_bits high for the positions on which o_valid_bits is low has no effect. The value of i_en_bits
only matters when we're actually sending a payload. Note that having i_en_bits high for a payload also has an effect on the "clock_out"
pin, to allow the testcase to determine what data is expected to be sampled when it comes back o i_stream. In the case of an SDR test,
clk_out toggles every time inputs are sampled.
i_ready_bits: is a string of "1"s and "0"s, and it specifies on which clock cycle the i_stream is ready or not. Each character refers
to a single clock cycle, and if the stream does not have a payload available at the releavant clock cycle, then the coroutine that
consumes i_stream payloads moves on to the next character in this string. Make sure to have sufficient "1"s in this string so that all
sample requests (o_stream payloads with i_en high) that have been launched are collected, otherwise the testcase may fail, in one of two ways:
- back-pressure on i_stream may result in back-pressure on o_stream, never allowing the full o_valid_bits string to be completely played back,
and resulting in a timeout.
- the playback of the o_valid_bits string may complete, however it's possible a number of sample requests that are in flight remain stuck
in the IOStreamer, pending for them to be extracted from the i_stream, and that could result in the testcase declaring that the final
samples have been lost.
To make sure a testcase completes, you will see some testcases have a large number of "1"s in this string past the length of o_valid_bits.
timeout_clocks: The number of clock cycles we expect the testcase to complete. If None (default), then an appropriate amount of clock cycles is
calculated.
"""
ports = PortGroup()
ports.clk_out = io.SimulationPort("o", 1)
ports.data_in = io.SimulationPort("i", 8)

if timeout_clocks is None:
timeout_clocks = len(o_valid_bits) + len(i_ready_bits) + 20

dut = IOStreamer({
"clk_out": ("o", 1),
"data_in": ("i", 8),
}, ports, meta_layout=4)

expected_sample = []
actually_sampled = []
i_stream_consumer_finished = False

async def input_generator_tb(ctx):
"""
Expand All @@ -48,54 +111,96 @@ async def input_generator_tb(ctx):

async def save_expected_sample_values_tb(ctx):
"""
This testbench looks at the clk_out port and when it sees a positive edge it knows that
This testbench looks at the clk_out port and when it sees a positive or negative edge it knows that
IOStreamer is expected to sample the input signal, so the current state of the data_in port
becomes one of the expected sampled values. This is saved into expected_sample[] to be compared
later, when the sample actually arrives back on i_stream.
"""
while True:
await ctx.posedge(ports.clk_out.o[0])
value = ctx.get(ports.data_in.i)
_, value = await ctx.changed(ports.clk_out.o).sample(ports.data_in.i)
expected_sample.append(value)

async def i_stream_consumer_tb(ctx):
"""
This testbench saves all the samples coming in over i_stream
"""
while True:
payload = await stream_get(ctx,dut.i_stream)
actually_sampled.append(payload.port.data_in.i)
nonlocal i_stream_consumer_finished
for i_ready_bit in i_ready_bits:
if i_ready_bit == "1":
payload = await stream_get_maybe(ctx,dut.i_stream)
if payload is not None:
actually_sampled.append(payload.port.data_in.i)
else:
await ctx.tick()
i_stream_consumer_finished = True


async def check_timeout_tb(ctx):
"""
This testbench ends the testcase if the given number of timeout clock cycles has elapsed.
"""
await ctx.tick().repeat(timeout_clocks)
raise IOStreamTimeoutError("Testcase timeout")

async def main_testbench(ctx):
"""
This testbench is the producer for o_stream, and it also serves as the main orchestrator
for the entire testcase. After it produces the stimulus, it waits a number of clock cycles
to make sure any input latency has passed, and then it verifies that the expected number
of bytes has been received, and that the expected values have been sampled.
for the entire testcase. After it produces the stimulus, it waits for the i_stream_consumer_tb
to finish (which may take longer to potentially consume in-flight sampled payloads due to the
latency of the dut). and then it verifies that the expected number of bytes has been received,
and that the expected values have been sampled.
"""
await ctx.tick()

for i in range(0,8):
await stream_put(ctx, dut.o_stream, {"meta": i, "i_en": i % 2, "port": { "clk_out": {"o": i % 2}}})

await stream_put(ctx, dut.o_stream, {"meta": 0, "i_en": 0, "port": { "clk_out": {"o": 0}}})

for i in range(20):
expected_samples_count = 0
o_bit = 0

for i in range(len(o_valid_bits)):
o_valid_bit = 1 if o_valid_bits[i] == "1" else 0
i_en_bit = 1 if i_en_bits[i] == "1" else 0
if o_valid_bit:
if i_en_bit:
expected_samples_count += 1
o_bit ^= 1
await stream_put(ctx, dut.o_stream, {
"meta": i,
"i_en": i_en_bit,
"port": {
"clk_out": {
"o": o_bit,
}
}
})
else:
await ctx.tick()

while not i_stream_consumer_finished:
await ctx.tick()
assert len(actually_sampled) == 4 # This should be checked as well, because a possible failure mode is
# if IOStreamer never generates clock edges. We don't want to end up comparing two empty lists against
# eachother.
assert actually_sampled == expected_sample

assert len(actually_sampled) == expected_samples_count # This should be checked as well, because a
# possible failure mode is if IOStreamer never generates clock edges. We don't want to end up
# comparing two empty lists against eachother.
assert actually_sampled == expected_sample, (f"Expected [" +
", ".join(f"0x{s:02x}" for s in expected_sample) +
"] Got [" +
", ".join(f"0x{s:02x}" for s in actually_sampled) + "]")

sim = Simulator(dut)
sim.add_clock(1e-6)
sim.add_testbench(main_testbench)
sim.add_testbench(i_stream_consumer_tb, background = True)
sim.add_testbench(i_stream_consumer_tb, background=True)
sim.add_testbench(input_generator_tb, background=True)
sim.add_testbench(save_expected_sample_values_tb, background=True)
sim.add_testbench(check_timeout_tb, background=True)
with sim.write_vcd("test.vcd", fs_per_delta=1):
sim.run()

def test_sdr_input_sampled_correctly(self):
self._subtest_sdr_input_sampled_correctly(
o_valid_bits = "111111111",
i_en_bits = "010101010",
i_ready_bits = "111111111" + ("1"*20))

def test_basic(self):
ports = PortGroup()
ports.data = port = io.SimulationPort("io", 1)
Expand Down

0 comments on commit 3814e22

Please sign in to comment.