Use cocotb async queues

This commit is contained in:
Alex Forencich
2021-03-17 18:13:04 -07:00
parent a81d43216e
commit 5855644670
4 changed files with 164 additions and 108 deletions

View File

@@ -23,9 +23,9 @@ THE SOFTWARE.
"""
import logging
from collections import deque
import cocotb
from cocotb.queue import Queue
from cocotb.triggers import RisingEdge, FallingEdge, Timer, First, Event
from cocotb.utils import get_sim_time, get_sim_steps
@@ -56,7 +56,9 @@ class RgmiiSource(Reset):
super().__init__(*args, **kwargs)
self.active = False
self.queue = deque()
self.queue = Queue()
self.idle_event = Event()
self.idle_event.set()
self.ifg = 12
self.mii_mode = False
@@ -77,31 +79,37 @@ class RgmiiSource(Reset):
self._init_reset(reset, reset_active_level)
async def send(self, frame):
self.send_nowait(frame)
frame = GmiiFrame(frame)
await self.queue.put(frame)
self.idle_event.clear()
self.queue_occupancy_bytes += len(frame)
self.queue_occupancy_frames += 1
def send_nowait(self, frame):
frame = GmiiFrame(frame)
self.queue.put_nowait(frame)
self.idle_event.clear()
self.queue_occupancy_bytes += len(frame)
self.queue_occupancy_frames += 1
self.queue.append(frame)
def count(self):
return len(self.queue)
return self.queue.qsize()
def empty(self):
return not self.queue
return self.queue.empty()
def idle(self):
return self.empty() and not self.active
def clear(self):
self.queue.clear()
while not self.queue.empty():
self.queue.get_nowait()
self.idle_event.set()
self.queue_occupancy_bytes = 0
self.queue_occupancy_frames = 0
async def wait(self):
while not self.idle():
await RisingEdge(self.clock)
await self.idle_event.wait()
def _handle_reset(self, state):
if state:
@@ -138,9 +146,9 @@ class RgmiiSource(Reset):
# in IFG
ifg_cnt -= 1
elif frame is None and self.queue:
elif frame is None and not self.queue.empty():
# send frame
frame = self.queue.popleft()
frame = self.queue.get_nowait()
self.queue_occupancy_bytes -= len(frame)
self.queue_occupancy_frames -= 1
frame.sim_time_start = get_sim_time()
@@ -183,6 +191,7 @@ class RgmiiSource(Reset):
er = 0
en = 0
self.active = False
self.idle_event.set()
await FallingEdge(self.clock)
@@ -212,8 +221,8 @@ class RgmiiSink(Reset):
super().__init__(*args, **kwargs)
self.active = False
self.queue = deque()
self.sync = Event()
self.queue = Queue()
self.active_event = Event()
self.mii_mode = False
@@ -231,41 +240,46 @@ class RgmiiSink(Reset):
self._init_reset(reset, reset_active_level)
async def recv(self, compact=True):
while self.empty():
self.sync.clear()
await self.sync.wait()
return self.recv_nowait(compact)
frame = await self.queue.get()
if self.queue.empty():
self.active_event.clear()
self.queue_occupancy_bytes -= len(frame)
self.queue_occupancy_frames -= 1
return frame
def recv_nowait(self, compact=True):
if self.queue:
frame = self.queue.popleft()
if not self.queue.empty():
frame = self.queue.get_nowait()
if self.queue.empty():
self.active_event.clear()
self.queue_occupancy_bytes -= len(frame)
self.queue_occupancy_frames -= 1
return frame
return None
def count(self):
return len(self.queue)
return self.queue.qsize()
def empty(self):
return not self.queue
return self.queue.empty()
def idle(self):
return not self.active
def clear(self):
self.queue.clear()
while not self.queue.empty():
self.queue.get_nowait()
self.active_event.clear()
self.queue_occupancy_bytes = 0
self.queue_occupancy_frames = 0
async def wait(self, timeout=0, timeout_unit=None):
if not self.empty():
return
self.sync.clear()
if timeout:
await First(self.sync.wait(), Timer(timeout, timeout_unit))
await First(self.active_event.wait(), Timer(timeout, timeout_unit))
else:
await self.sync.wait()
await self.active_event.wait()
def _handle_reset(self, state):
if state:
@@ -342,8 +356,8 @@ class RgmiiSink(Reset):
self.queue_occupancy_bytes += len(frame)
self.queue_occupancy_frames += 1
self.queue.append(frame)
self.sync.set()
self.queue.put_nowait(frame)
self.active_event.set()
frame = None