forked from M-Labs/artiq
409 lines
14 KiB
Python
409 lines
14 KiB
Python
"""Real-time packet layer for masters"""
|
|
|
|
from migen import *
|
|
from migen.genlib.fsm import *
|
|
from migen.genlib.fifo import AsyncFIFO
|
|
from migen.genlib.cdc import PulseSynchronizer
|
|
|
|
from artiq.gateware.rtio.cdc import GrayCodeTransfer, BlindTransfer
|
|
from artiq.gateware.drtio.rt_serializer import *
|
|
|
|
|
|
class _CrossDomainRequest(Module):
|
|
def __init__(self, domain,
|
|
req_stb, req_ack, req_data,
|
|
srv_stb, srv_ack, srv_data):
|
|
dsync = getattr(self.sync, domain)
|
|
|
|
request = PulseSynchronizer("sys", domain)
|
|
reply = PulseSynchronizer(domain, "sys")
|
|
self.submodules += request, reply
|
|
|
|
ongoing = Signal()
|
|
self.comb += request.i.eq(~ongoing & req_stb)
|
|
self.sync += [
|
|
req_ack.eq(reply.o),
|
|
If(req_stb, ongoing.eq(1)),
|
|
If(req_ack, ongoing.eq(0))
|
|
]
|
|
if req_data is not None:
|
|
req_data_r = Signal.like(req_data)
|
|
req_data_r.attr.add("no_retiming")
|
|
self.sync += If(req_stb, req_data_r.eq(req_data))
|
|
dsync += [
|
|
If(request.o, srv_stb.eq(1)),
|
|
If(srv_ack, srv_stb.eq(0))
|
|
]
|
|
if req_data is not None:
|
|
dsync += If(request.o, srv_data.eq(req_data_r))
|
|
self.comb += reply.i.eq(srv_stb & srv_ack)
|
|
|
|
|
|
class _CrossDomainNotification(Module):
|
|
def __init__(self, domain,
|
|
emi_stb, emi_data,
|
|
rec_stb, rec_ack, rec_data):
|
|
emi_data_r = Signal(len(emi_data))
|
|
emi_data_r.attr.add("no_retiming")
|
|
dsync = getattr(self.sync, domain)
|
|
dsync += If(emi_stb, emi_data_r.eq(emi_data))
|
|
|
|
ps = PulseSynchronizer(domain, "sys")
|
|
self.submodules += ps
|
|
self.comb += ps.i.eq(emi_stb)
|
|
self.sync += [
|
|
If(rec_ack, rec_stb.eq(0)),
|
|
If(ps.o,
|
|
rec_data.eq(emi_data_r),
|
|
rec_stb.eq(1)
|
|
)
|
|
]
|
|
|
|
|
|
class RTPacketMaster(Module):
|
|
def __init__(self, link_layer, sr_fifo_depth=4):
|
|
# all interface signals in sys domain unless otherwise specified
|
|
|
|
# standard request interface
|
|
#
|
|
# notwrite=1 address=0 buffer space request
|
|
# notwrite=1 address=1 read request <channel, timestamp>
|
|
#
|
|
# optimized for write throughput
|
|
# requests are performed on the DRTIO link preserving their order of issue
|
|
# this is important for buffer space requests, which have to be ordered
|
|
# wrt writes.
|
|
self.sr_stb = Signal()
|
|
self.sr_ack = Signal()
|
|
self.sr_notwrite = Signal()
|
|
self.sr_timestamp = Signal(64)
|
|
self.sr_channel = Signal(16)
|
|
self.sr_address = Signal(16)
|
|
self.sr_data = Signal(512)
|
|
|
|
# buffer space reply interface
|
|
self.buffer_space_not = Signal()
|
|
self.buffer_space_not_ack = Signal()
|
|
self.buffer_space = Signal(16)
|
|
|
|
# read reply interface
|
|
self.read_not = Signal()
|
|
self.read_not_ack = Signal()
|
|
# no_event is_overflow
|
|
# 0 X event
|
|
# 1 0 timeout
|
|
# 1 1 overflow
|
|
self.read_no_event = Signal()
|
|
self.read_is_overflow = Signal()
|
|
self.read_data = Signal(32)
|
|
self.read_timestamp = Signal(64)
|
|
|
|
# echo interface
|
|
self.echo_stb = Signal()
|
|
self.echo_ack = Signal()
|
|
self.echo_sent_now = Signal() # in rtio domain
|
|
self.echo_received_now = Signal() # in rtio_rx domain
|
|
|
|
# set_time interface
|
|
self.set_time_stb = Signal()
|
|
self.set_time_ack = Signal()
|
|
# in rtio domain, must be valid all time while there is
|
|
# a set_time request pending
|
|
self.tsc_value = Signal(64)
|
|
|
|
# rx errors
|
|
self.err_unknown_packet_type = Signal()
|
|
self.err_packet_truncated = Signal()
|
|
|
|
# packet counters
|
|
self.packet_cnt_tx = Signal(32)
|
|
self.packet_cnt_rx = Signal(32)
|
|
|
|
# # #
|
|
|
|
# RX/TX datapath
|
|
assert len(link_layer.tx_rt_data) == len(link_layer.rx_rt_data)
|
|
assert len(link_layer.tx_rt_data) % 8 == 0
|
|
ws = len(link_layer.tx_rt_data)
|
|
tx_plm = get_m2s_layouts(ws)
|
|
tx_dp = ClockDomainsRenamer("rtio")(TransmitDatapath(
|
|
link_layer.tx_rt_frame, link_layer.tx_rt_data, tx_plm))
|
|
self.submodules += tx_dp
|
|
rx_plm = get_s2m_layouts(ws)
|
|
rx_dp = ClockDomainsRenamer("rtio_rx")(ReceiveDatapath(
|
|
link_layer.rx_rt_frame, link_layer.rx_rt_data, rx_plm))
|
|
self.submodules += rx_dp
|
|
|
|
# Write FIFO and extra data count
|
|
sr_fifo = ClockDomainsRenamer({"write": "sys_with_rst", "read": "rtio_with_rst"})(
|
|
AsyncFIFO(1+64+16+16+512, sr_fifo_depth))
|
|
self.submodules += sr_fifo
|
|
sr_notwrite_d = Signal()
|
|
sr_timestamp_d = Signal(64)
|
|
sr_channel_d = Signal(16)
|
|
sr_address_d = Signal(16)
|
|
sr_data_d = Signal(512)
|
|
self.comb += [
|
|
sr_fifo.we.eq(self.sr_stb),
|
|
self.sr_ack.eq(sr_fifo.writable),
|
|
sr_fifo.din.eq(Cat(self.sr_notwrite, self.sr_timestamp, self.sr_channel,
|
|
self.sr_address, self.sr_data)),
|
|
Cat(sr_notwrite_d, sr_timestamp_d, sr_channel_d,
|
|
sr_address_d, sr_data_d).eq(sr_fifo.dout)
|
|
]
|
|
|
|
sr_buf_readable = Signal()
|
|
sr_buf_re = Signal()
|
|
|
|
self.comb += sr_fifo.re.eq(sr_fifo.readable & (~sr_buf_readable | sr_buf_re))
|
|
self.sync.rtio += \
|
|
If(sr_fifo.re,
|
|
sr_buf_readable.eq(1),
|
|
).Elif(sr_buf_re,
|
|
sr_buf_readable.eq(0),
|
|
)
|
|
|
|
sr_notwrite = Signal()
|
|
sr_timestamp = Signal(64)
|
|
sr_channel = Signal(16)
|
|
sr_address = Signal(16)
|
|
sr_extra_data_cnt = Signal(8)
|
|
sr_data = Signal(512)
|
|
|
|
self.sync.rtio += If(sr_fifo.re,
|
|
sr_notwrite.eq(sr_notwrite_d),
|
|
sr_timestamp.eq(sr_timestamp_d),
|
|
sr_channel.eq(sr_channel_d),
|
|
sr_address.eq(sr_address_d),
|
|
sr_data.eq(sr_data_d))
|
|
|
|
short_data_len = tx_plm.field_length("write", "short_data")
|
|
sr_extra_data_d = Signal(512)
|
|
self.comb += sr_extra_data_d.eq(sr_data_d[short_data_len:])
|
|
for i in range(512//ws):
|
|
self.sync.rtio += If(sr_fifo.re,
|
|
If(sr_extra_data_d[ws*i:ws*(i+1)] != 0, sr_extra_data_cnt.eq(i+1)))
|
|
|
|
sr_extra_data = Signal(512)
|
|
self.sync.rtio += If(sr_fifo.re, sr_extra_data.eq(sr_extra_data_d))
|
|
|
|
extra_data_ce = Signal()
|
|
extra_data_last = Signal()
|
|
extra_data_counter = Signal(max=512//ws+1)
|
|
self.comb += [
|
|
Case(extra_data_counter,
|
|
{i+1: tx_dp.raw_data.eq(sr_extra_data[i*ws:(i+1)*ws])
|
|
for i in range(512//ws)}),
|
|
extra_data_last.eq(extra_data_counter == sr_extra_data_cnt)
|
|
]
|
|
self.sync.rtio += \
|
|
If(extra_data_ce,
|
|
extra_data_counter.eq(extra_data_counter + 1),
|
|
).Else(
|
|
extra_data_counter.eq(1)
|
|
)
|
|
|
|
# CDC
|
|
buffer_space_not = Signal()
|
|
buffer_space = Signal(16)
|
|
self.submodules += _CrossDomainNotification("rtio_rx",
|
|
buffer_space_not, buffer_space,
|
|
self.buffer_space_not, self.buffer_space_not_ack, self.buffer_space)
|
|
|
|
set_time_stb = Signal()
|
|
set_time_ack = Signal()
|
|
self.submodules += _CrossDomainRequest("rtio",
|
|
self.set_time_stb, self.set_time_ack, None,
|
|
set_time_stb, set_time_ack, None)
|
|
|
|
echo_stb = Signal()
|
|
echo_ack = Signal()
|
|
self.submodules += _CrossDomainRequest("rtio",
|
|
self.echo_stb, self.echo_ack, None,
|
|
echo_stb, echo_ack, None)
|
|
|
|
read_not = Signal()
|
|
read_no_event = Signal()
|
|
read_is_overflow = Signal()
|
|
read_data = Signal(32)
|
|
read_timestamp = Signal(64)
|
|
self.submodules += _CrossDomainNotification("rtio_rx",
|
|
read_not,
|
|
Cat(read_no_event, read_is_overflow, read_data, read_timestamp),
|
|
|
|
self.read_not, self.read_not_ack,
|
|
Cat(self.read_no_event, self.read_is_overflow,
|
|
self.read_data, self.read_timestamp))
|
|
self.comb += [
|
|
read_is_overflow.eq(rx_dp.packet_as["read_reply_noevent"].overflow),
|
|
read_data.eq(rx_dp.packet_as["read_reply"].data),
|
|
read_timestamp.eq(rx_dp.packet_as["read_reply"].timestamp)
|
|
]
|
|
|
|
err_unknown_packet_type = BlindTransfer("rtio_rx", "sys")
|
|
err_packet_truncated = BlindTransfer("rtio_rx", "sys")
|
|
self.submodules += err_unknown_packet_type, err_packet_truncated
|
|
self.comb += [
|
|
self.err_unknown_packet_type.eq(err_unknown_packet_type.o),
|
|
self.err_packet_truncated.eq(err_packet_truncated.o)
|
|
]
|
|
|
|
# TX FSM
|
|
tx_fsm = ClockDomainsRenamer("rtio")(FSM(reset_state="IDLE"))
|
|
self.submodules += tx_fsm
|
|
|
|
echo_sent_now = Signal()
|
|
self.sync.rtio += self.echo_sent_now.eq(echo_sent_now)
|
|
tsc_value = Signal(64)
|
|
tsc_value_load = Signal()
|
|
self.sync.rtio += If(tsc_value_load, tsc_value.eq(self.tsc_value))
|
|
|
|
tx_fsm.act("IDLE",
|
|
If(sr_buf_readable,
|
|
If(sr_notwrite,
|
|
Case(sr_address[0], {
|
|
0: NextState("BUFFER_SPACE"),
|
|
1: NextState("READ")
|
|
}),
|
|
).Else(
|
|
NextState("WRITE")
|
|
)
|
|
).Else(
|
|
If(echo_stb,
|
|
echo_sent_now.eq(1),
|
|
NextState("ECHO")
|
|
).Elif(set_time_stb,
|
|
tsc_value_load.eq(1),
|
|
NextState("SET_TIME")
|
|
)
|
|
)
|
|
)
|
|
tx_fsm.act("WRITE",
|
|
tx_dp.send("write",
|
|
timestamp=sr_timestamp,
|
|
channel=sr_channel,
|
|
address=sr_address,
|
|
extra_data_cnt=sr_extra_data_cnt,
|
|
short_data=sr_data[:short_data_len]),
|
|
If(tx_dp.packet_last,
|
|
If(sr_extra_data_cnt == 0,
|
|
sr_buf_re.eq(1),
|
|
NextState("IDLE")
|
|
).Else(
|
|
NextState("WRITE_EXTRA")
|
|
)
|
|
)
|
|
)
|
|
tx_fsm.act("WRITE_EXTRA",
|
|
tx_dp.raw_stb.eq(1),
|
|
extra_data_ce.eq(1),
|
|
If(extra_data_last,
|
|
sr_buf_re.eq(1),
|
|
NextState("IDLE")
|
|
)
|
|
)
|
|
tx_fsm.act("BUFFER_SPACE",
|
|
tx_dp.send("buffer_space_request"),
|
|
If(tx_dp.packet_last,
|
|
sr_buf_re.eq(1),
|
|
NextState("IDLE")
|
|
)
|
|
)
|
|
tx_fsm.act("READ",
|
|
tx_dp.send("read_request", channel=sr_channel, timeout=sr_timestamp),
|
|
If(tx_dp.packet_last,
|
|
sr_buf_re.eq(1),
|
|
NextState("IDLE")
|
|
)
|
|
)
|
|
tx_fsm.act("ECHO",
|
|
tx_dp.send("echo_request"),
|
|
If(tx_dp.packet_last,
|
|
echo_ack.eq(1),
|
|
NextState("IDLE")
|
|
)
|
|
)
|
|
tx_fsm.act("SET_TIME",
|
|
tx_dp.send("set_time", timestamp=tsc_value),
|
|
If(tx_dp.packet_last,
|
|
set_time_ack.eq(1),
|
|
NextState("IDLE")
|
|
)
|
|
)
|
|
|
|
# RX FSM
|
|
rx_fsm = ClockDomainsRenamer("rtio_rx")(FSM(reset_state="INPUT"))
|
|
self.submodules += rx_fsm
|
|
|
|
ongoing_packet_next = Signal()
|
|
ongoing_packet = Signal()
|
|
self.sync.rtio_rx += ongoing_packet.eq(ongoing_packet_next)
|
|
|
|
echo_received_now = Signal()
|
|
self.sync.rtio_rx += self.echo_received_now.eq(echo_received_now)
|
|
|
|
rx_fsm.act("INPUT",
|
|
If(rx_dp.frame_r,
|
|
rx_dp.packet_buffer_load.eq(1),
|
|
If(rx_dp.packet_last,
|
|
Case(rx_dp.packet_type, {
|
|
rx_plm.types["echo_reply"]: echo_received_now.eq(1),
|
|
rx_plm.types["buffer_space_reply"]: NextState("BUFFER_SPACE"),
|
|
rx_plm.types["read_reply"]: NextState("READ_REPLY"),
|
|
rx_plm.types["read_reply_noevent"]: NextState("READ_REPLY_NOEVENT"),
|
|
"default": err_unknown_packet_type.i.eq(1)
|
|
})
|
|
).Else(
|
|
ongoing_packet_next.eq(1)
|
|
)
|
|
),
|
|
If(~rx_dp.frame_r & ongoing_packet,
|
|
err_packet_truncated.i.eq(1)
|
|
)
|
|
)
|
|
rx_fsm.act("BUFFER_SPACE",
|
|
buffer_space_not.eq(1),
|
|
buffer_space.eq(rx_dp.packet_as["buffer_space_reply"].space),
|
|
NextState("INPUT")
|
|
)
|
|
rx_fsm.act("READ_REPLY",
|
|
read_not.eq(1),
|
|
read_no_event.eq(0),
|
|
NextState("INPUT")
|
|
)
|
|
rx_fsm.act("READ_REPLY_NOEVENT",
|
|
read_not.eq(1),
|
|
read_no_event.eq(1),
|
|
NextState("INPUT")
|
|
)
|
|
|
|
# packet counters
|
|
tx_frame_r = Signal()
|
|
packet_cnt_tx = Signal(32)
|
|
self.sync.rtio += [
|
|
tx_frame_r.eq(link_layer.tx_rt_frame),
|
|
If(link_layer.tx_rt_frame & ~tx_frame_r,
|
|
packet_cnt_tx.eq(packet_cnt_tx + 1))
|
|
]
|
|
cdc_packet_cnt_tx = GrayCodeTransfer(32)
|
|
self.submodules += cdc_packet_cnt_tx
|
|
self.comb += [
|
|
cdc_packet_cnt_tx.i.eq(packet_cnt_tx),
|
|
self.packet_cnt_tx.eq(cdc_packet_cnt_tx.o)
|
|
]
|
|
|
|
rx_frame_r = Signal()
|
|
packet_cnt_rx = Signal(32)
|
|
self.sync.rtio_rx += [
|
|
rx_frame_r.eq(link_layer.rx_rt_frame),
|
|
If(link_layer.rx_rt_frame & ~rx_frame_r,
|
|
packet_cnt_rx.eq(packet_cnt_rx + 1))
|
|
]
|
|
cdc_packet_cnt_rx = ClockDomainsRenamer({"rtio": "rtio_rx"})(
|
|
GrayCodeTransfer(32))
|
|
self.submodules += cdc_packet_cnt_rx
|
|
self.comb += [
|
|
cdc_packet_cnt_rx.i.eq(packet_cnt_rx),
|
|
self.packet_cnt_rx.eq(cdc_packet_cnt_rx.o)
|
|
]
|