mirror of
https://github.com/enjoy-digital/litex.git
synced 2025-01-04 09:52:26 -05:00
197 lines
5.4 KiB
Python
197 lines
5.4 KiB
Python
from litex.gen import *
|
|
from litex.gen.genlib.misc import timeline
|
|
|
|
from litex.soc.interconnect import wishbone
|
|
from litex.soc.interconnect.csr import AutoCSR, CSRStorage, CSRStatus
|
|
|
|
|
|
_FAST_READ = 0x0b
|
|
_DIOFR = 0xbb
|
|
_QIOFR = 0xeb
|
|
|
|
|
|
def _format_cmd(cmd, spi_width):
|
|
"""
|
|
`cmd` is the read instruction. Since everything is transmitted on all
|
|
dq lines (cmd, adr and data), extend/interleave cmd to full pads.dq
|
|
width even if dq1-dq3 are don't care during the command phase:
|
|
For example, for N25Q128, 0xeb is the quad i/o fast read, and
|
|
extended to 4 bits (dq1,dq2,dq3 high) is: 0xfffefeff
|
|
"""
|
|
c = 2**(8*spi_width)-1
|
|
for b in range(8):
|
|
if not (cmd>>b)%2:
|
|
c &= ~(1<<(b*spi_width))
|
|
return c
|
|
|
|
|
|
class SpiFlashDualQuad(Module, AutoCSR):
|
|
def __init__(self, pads, dummy=15, div=2):
|
|
"""
|
|
Simple SPI flash.
|
|
Supports multi-bit pseudo-parallel reads (aka Dual or Quad I/O Fast
|
|
Read). Only supports mode0 (cpol=0, cpha=0).
|
|
"""
|
|
self.bus = bus = wishbone.Interface()
|
|
spi_width = len(pads.dq)
|
|
assert spi_width >= 2
|
|
|
|
# # #
|
|
|
|
cs_n = Signal(reset=1)
|
|
clk = Signal()
|
|
dq_oe = Signal()
|
|
wbone_width = len(bus.dat_r)
|
|
|
|
|
|
read_cmd_params = {
|
|
4: (_format_cmd(_QIOFR, 4), 4*8),
|
|
2: (_format_cmd(_DIOFR, 2), 2*8),
|
|
1: (_format_cmd(_FAST_READ, 1), 1*8)
|
|
}
|
|
read_cmd, cmd_width = read_cmd_params[spi_width]
|
|
addr_width = 24
|
|
|
|
dq = TSTriple(spi_width)
|
|
self.specials.dq = dq.get_tristate(pads.dq)
|
|
|
|
sr = Signal(max(cmd_width, addr_width, wbone_width))
|
|
self.comb += bus.dat_r.eq(sr)
|
|
|
|
self.comb += [
|
|
pads.clk.eq(clk),
|
|
pads.cs_n.eq(cs_n),
|
|
dq.o.eq(sr[-spi_width:]),
|
|
dq.oe.eq(dq_oe)
|
|
]
|
|
|
|
if div < 2:
|
|
raise ValueError("Unsupported value \'{}\' for div parameter for SpiFlash core".format(div))
|
|
else:
|
|
i = Signal(max=div)
|
|
dqi = Signal(spi_width)
|
|
self.sync += [
|
|
If(i == div//2 - 1,
|
|
clk.eq(1),
|
|
dqi.eq(dq.i),
|
|
),
|
|
If(i == div - 1,
|
|
i.eq(0),
|
|
clk.eq(0),
|
|
sr.eq(Cat(dqi, sr[:-spi_width]))
|
|
).Else(
|
|
i.eq(i + 1),
|
|
),
|
|
]
|
|
|
|
# spi is byte-addressed, prefix by zeros
|
|
z = Replicate(0, log2_int(wbone_width//8))
|
|
|
|
seq = [
|
|
(cmd_width//spi_width*div,
|
|
[dq_oe.eq(1), cs_n.eq(0), sr[-cmd_width:].eq(read_cmd)]),
|
|
(addr_width//spi_width*div,
|
|
[sr[-addr_width:].eq(Cat(z, bus.adr))]),
|
|
((dummy + wbone_width//spi_width)*div,
|
|
[dq_oe.eq(0)]),
|
|
(1,
|
|
[bus.ack.eq(1), cs_n.eq(1)]),
|
|
(div, # tSHSL!
|
|
[bus.ack.eq(0)]),
|
|
(0,
|
|
[]),
|
|
]
|
|
|
|
# accumulate timeline deltas
|
|
t, tseq = 0, []
|
|
for dt, a in seq:
|
|
tseq.append((t, a))
|
|
t += dt
|
|
|
|
self.sync += timeline(bus.cyc & bus.stb & (i == div - 1), tseq)
|
|
|
|
|
|
class SpiFlashSingle(Module, AutoCSR):
|
|
def __init__(self, pads, dummy=15, div=2):
|
|
"""
|
|
Simple SPI flash.
|
|
Supports 1-bit reads. Only supports mode0 (cpol=0, cpha=0).
|
|
"""
|
|
self.bus = bus = wishbone.Interface()
|
|
|
|
# # #
|
|
|
|
if hasattr(pads, "wp"):
|
|
self.comb += pads.wp.eq(1)
|
|
|
|
if hasattr(pads, "hold"):
|
|
self.comb += pads.hold.eq(1)
|
|
|
|
cs_n = Signal(reset=1)
|
|
clk = Signal()
|
|
wbone_width = len(bus.dat_r)
|
|
|
|
read_cmd = _FAST_READ
|
|
cmd_width = 8
|
|
addr_width = 24
|
|
|
|
sr = Signal(max(cmd_width, addr_width, wbone_width))
|
|
self.comb += bus.dat_r.eq(sr)
|
|
|
|
self.comb += [
|
|
pads.clk.eq(clk),
|
|
pads.cs_n.eq(cs_n),
|
|
pads.mosi.eq(sr[-1:])
|
|
]
|
|
|
|
if div < 2:
|
|
raise ValueError("Unsupported value \'{}\' for div parameter for SpiFlash core".format(div))
|
|
else:
|
|
i = Signal(max=div)
|
|
miso = Signal()
|
|
self.sync += [
|
|
If(i == div//2 - 1,
|
|
clk.eq(1),
|
|
miso.eq(pads.miso),
|
|
),
|
|
If(i == div - 1,
|
|
i.eq(0),
|
|
clk.eq(0),
|
|
sr.eq(Cat(miso, sr[:-1]))
|
|
).Else(
|
|
i.eq(i + 1),
|
|
),
|
|
]
|
|
|
|
# spi is byte-addressed, prefix by zeros
|
|
z = Replicate(0, log2_int(wbone_width//8))
|
|
|
|
seq = [
|
|
(cmd_width*div,
|
|
[cs_n.eq(0), sr[-cmd_width:].eq(read_cmd)]),
|
|
(addr_width*div,
|
|
[sr[-addr_width:].eq(Cat(z, bus.adr))]),
|
|
((dummy + wbone_width)*div,
|
|
[]),
|
|
(1,
|
|
[bus.ack.eq(1), cs_n.eq(1)]),
|
|
(div, # tSHSL!
|
|
[bus.ack.eq(0)]),
|
|
(0,
|
|
[]),
|
|
]
|
|
|
|
# accumulate timeline deltas
|
|
t, tseq = 0, []
|
|
for dt, a in seq:
|
|
tseq.append((t, a))
|
|
t += dt
|
|
|
|
self.sync += timeline(bus.cyc & bus.stb & (i == div - 1), tseq)
|
|
|
|
|
|
def SpiFlash(pads, *args, **kw):
|
|
if hasattr(pads, "mosi"):
|
|
return SpiFlashSingle(pads, *args, **kw)
|
|
else:
|
|
return SpiFlashDualQuad(pads, *args, **kw)
|