@@ 4,6 4,7 @@ from dataclasses import dataclass
from cocotb.queue import Queue
from cocotb.clock import Clock
from cocotb.triggers import Trigger, First, Event, Timer, Edge, RisingEdge, FallingEdge
+from cocotb.utils import get_sim_time
import logging
@@ 100,7 101,19 @@ class SpiSlave:
async def wait_all(self):
await self._idle.wait()
+
+ async def _check_sck_period(self):
+ await Edge(self.sck)
+ while True:
+ sim_time = get_sim_time(self.config.sck_period_unit)
+ await Edge(self.sck)
+ next_sim_time = get_sim_time(self.config.sck_period_unit)
+
+ if next_sim_time - sim_time > 0 and (next_sim_time - sim_time) * 1.0005 < self.config.sck_period / 2:
+ raise Exception(f"The sck pulse is too narrow! (was: {next_sim_time - sim_time} {self.config.sck_period_unit}, expected: {self.config.sck_period / 2} {self.config.sck_period_unit})")
+
async def coroutine(self):
+ await cocotb.start(self._check_sck_period())
while True:
if self.transactions.empty():
self._idle.set()
@@ 176,17 189,20 @@ class SpiSlave:
break
# now wait for csn rising
- timeout = Timer(self.config.sck_period * 2, self.config.sck_period_unit)
- # sck_edge = Edge(self.sck)
- csn_rising = RisingEdge(self.csn)
- res = await First(csn_rising, timeout)
+ sim_time = get_sim_time(self.config.sck_period_unit)
+ while sim_time == get_sim_time(self.config.sck_period_unit):
+ timeout = Timer(self.config.sck_period * 2, self.config.sck_period_unit)
+ sck_edge = self.config.sampling(self.sck)
+ csn_rising = RisingEdge(self.csn)
+ res = await First(csn_rising, sck_edge, timeout)
if res == timeout:
self._log.error("Got no rising edge on csn")
raise Exception("Got no rising edge on csn")
continue
- # elif res == sck_edge:
- # self._log.error("Got sck edge when csn rising was expected")
+ elif res == sck_edge:
+ self._log.error("Got sck edge when csn rising was expected")
+ raise Exception("Got sck edge when csn rising was expected")
self._log.info("csn is rising")
@@ 455,7 455,7 @@ async def inverted_clock(dut):
async def shifted_inverted_clock(dut):
clk = Clock(dut.clk_i, 5, "ns")
interface = SpiInterface(dut.csn_o, dut.sck_o, dut.miso_i, dut.mosi_o)
- config = SpiConfig(8, FallingEdge, RisingEdge, 20, "ns")
+ config = SpiConfig(8, FallingEdge, RisingEdge, 10, "ns")
slave = SpiSlave(interface, config)
driver = DutDriver(dut)
@@ 474,6 474,7 @@ async def shifted_inverted_clock(dut):
count = 3
await perform_multiple_transmits(count, dut, slave, driver)
+ await Timer(100, "ns")
dut.clock_phase_i.value = 1
dut.clock_polarity_i.value = 0
await FallingEdge(dut.clk_i)
@@ 486,7 487,7 @@ async def shifted_inverted_clock(dut):
async def sixteen_bits(dut):
clk = Clock(dut.clk_i, 5, "ns")
interface = SpiInterface(dut.csn_o, dut.sck_o, dut.miso_i, dut.mosi_o)
- config = SpiConfig(16, FallingEdge, RisingEdge, 20, "ns")
+ config = SpiConfig(16, FallingEdge, RisingEdge, 10, "ns")
slave = SpiSlave(interface, config)
driver = DutDriver(dut)
@@ 538,7 539,7 @@ async def rx_blocking_tx(dut):
async def csn_pulse(dut):
clk = Clock(dut.clk_i, 5, "ns")
interface = SpiInterface(dut.csn_o, dut.sck_o, dut.miso_i, dut.mosi_o)
- config = SpiConfig(16, RisingEdge, FallingEdge, 20, "ns", csn_pulse = True)
+ config = SpiConfig(16, RisingEdge, FallingEdge, 10, "ns", csn_pulse = True)
slave = SpiSlave(interface, config)
driver = DutDriver(dut)