cores/icap/ICAP: Rewrite using constants and cleanup.

This commit is contained in:
Florent Kermarrec 2021-10-04 14:14:03 +02:00
parent 1f2b143c66
commit cb2f2d7021
2 changed files with 39 additions and 23 deletions

View File

@ -16,6 +16,14 @@ from litex.soc.interconnect import stream
# Constants ---------------------------------------------------------------------------------------- # Constants ----------------------------------------------------------------------------------------
# ICAP Words
ICAP_DUMMY = 0xffffffff
ICAP_SYNC = 0xaa995566
ICAP_NOOP = 0x20000000
ICAP_WRITE = 0x30000000
ICAP_READ = 0x28000000
# Configuration Registers (from UG470). # Configuration Registers (from UG470).
class ICAPRegisters(IntEnum): class ICAPRegisters(IntEnum):
@ -78,38 +86,45 @@ class ICAP(Module, AutoCSR):
# # # # # #
# Create slow icap clk (sys_clk/16) --------------------------------------------------------- # Create slow ICAP Clk (sys_clk/16).
self.clock_domains.cd_icap = ClockDomain() self.clock_domains.cd_icap = ClockDomain()
icap_clk_counter = Signal(4) icap_clk_counter = Signal(4)
self.sync += icap_clk_counter.eq(icap_clk_counter + 1) self.sync += icap_clk_counter.eq(icap_clk_counter + 1)
self.sync += self.cd_icap.clk.eq(icap_clk_counter[3]) self.sync += self.cd_icap.clk.eq(icap_clk_counter[3])
# Resynchronize send pulse to icap domain --------------------------------------------------- # Resynchronize send pulse to ICAP domain.
ps_send = PulseSynchronizer("sys", "icap") ps_send = PulseSynchronizer("sys", "icap")
self.submodules += ps_send self.submodules += ps_send
self.comb += ps_send.i.eq(self.send) self.comb += ps_send.i.eq(self.send)
# Generate icap bitstream write sequence # Generate ICAP bitstream write sequence.
self._csib = _csib = Signal(reset=1) self._csib = _csib = Signal(reset=1)
self._i = _i = Signal(32) self._i = _i = Signal(32)
_addr = self.addr << 13
_data = self.data
self.sync.icap += [ self.sync.icap += [
_i.eq(0xffffffff), # dummy _i.eq(ICAP_DUMMY), # Dummy (Default).
timeline(ps_send.o, [ timeline(ps_send.o, [
# Clear Done.
(1, [_csib.eq(1), self.done.eq(0)]), (1, [_csib.eq(1), self.done.eq(0)]),
(2, [_csib.eq(0), _i.eq(0x20000000)]), # noop
(3, [_csib.eq(0), _i.eq(0xaa995566)]), # sync word # Synchronize.
(4, [_csib.eq(0), _i.eq(0x20000000)]), # noop (2, [_csib.eq(0), _i.eq(ICAP_NOOP)]), # No Op.
(5, [_csib.eq(0), _i.eq(0x20000000)]), # noop (3, [_csib.eq(0), _i.eq(ICAP_SYNC)]), # Sync Word.
(6, [_csib.eq(0), _i.eq(0x30000001 | _addr)]), # write command (4, [_csib.eq(0), _i.eq(ICAP_NOOP)]), # No Op.
(7, [_csib.eq(0), _i.eq(_data)]), # write value (5, [_csib.eq(0), _i.eq(ICAP_NOOP)]), # No Op.
(8, [_csib.eq(0), _i.eq(0x20000000)]), # noop
(9, [_csib.eq(0), _i.eq(0x20000000)]), # noop # Write User's data to addr Register.
(10, [_csib.eq(0), _i.eq(0x30008001)]), # write to cmd register (6, [_csib.eq(0), _i.eq(ICAP_WRITE | (self.addr << 13) | 1)]), # Set Register.
(11, [_csib.eq(0), _i.eq(0x0000000d)]), # desync command (7, [_csib.eq(0), _i.eq(self.data)]), # Set Register Data.
(12, [_csib.eq(0), _i.eq(0x20000000)]), # noop (8, [_csib.eq(0), _i.eq(ICAP_NOOP)]), # No Op.
(13, [_csib.eq(0), _i.eq(0x20000000)]), # noop (9, [_csib.eq(0), _i.eq(ICAP_NOOP)]), # No Op.
# De-Synchronize.
(10, [_csib.eq(0), _i.eq(ICAP_WRITE | (ICAPRegisters.CMD << 13) | 1)]), # Write to CMD Register.
(11, [_csib.eq(0), _i.eq(ICAPCMDs.DESYNC)]), # DESYNC CMD.
(12, [_csib.eq(0), _i.eq(ICAP_NOOP)]), # No Op.
(13, [_csib.eq(0), _i.eq(ICAP_NOOP)]), # No Op.
# Set Done.
(14, [_csib.eq(1), self.done.eq(1)]), (14, [_csib.eq(1), self.done.eq(1)]),
]) ])
] ]

View File

@ -8,24 +8,25 @@ import unittest
from migen import * from migen import *
from litex.soc.cores.icap import ICAP, ICAPBitstream from litex.soc.cores.icap import *
class TestICAP(unittest.TestCase): class TestICAP(unittest.TestCase):
def test_icap_command_reload(self): def test_icap_command_reload(self):
def generator(dut): def generator(dut):
yield dut.addr.eq(0x4) yield dut.addr.eq(ICAPRegisters.CMD)
yield dut.data.eq(0xf) yield dut.data.eq(ICAPCMDs.IPROG)
for i in range(16): for i in range(16):
yield yield
yield dut.send.eq(1) yield dut.send.eq(1)
yield yield
yield dut.send.eq(0) yield dut.send.eq(0)
for i in range(256): for i in range(32):
print(f"{(yield dut._i):08x}")
yield yield
dut = ICAP(with_csr=False, simulation=True) dut = ICAP(with_csr=False, simulation=True)
clocks = {"sys": 10, "icap":20} clocks = {"sys": 10, "icap": 10}
run_simulation(dut, generator(dut), clocks, vcd_name="icap.vcd") run_simulation(dut, generator(dut), clocks, vcd_name="icap.vcd")
def test_icap_bitstream_syntax(self): def test_icap_bitstream_syntax(self):