litex/migen/fhdl/structure.py

389 lines
8.6 KiB
Python
Raw Normal View History

import math
import inspect
import re
from collections import defaultdict
from migen.fhdl import tracer
2012-01-16 12:09:52 -05:00
2012-03-14 07:19:42 -04:00
def log2_int(n):
l = 1
r = 0
while l < n:
l *= 2
r += 1
if l == n:
return r
else:
raise ValueError("Not a power of 2")
2011-12-16 10:02:55 -05:00
def bits_for(n):
2011-12-09 07:11:34 -05:00
if isinstance(n, Constant):
2012-07-13 12:32:54 -04:00
return len(n)
else:
if n < 0:
return bits_for(-n) + 1
elif n == 0:
2011-12-09 07:11:34 -05:00
return 1
else:
return int(math.ceil(math.log(n+1, 2)))
class BV:
def __init__(self, width=1, signed=False):
self.width = width
self.signed = signed
2011-12-09 07:11:34 -05:00
def __repr__(self):
r = str(self.width) + "'"
if self.signed:
r += "s"
r += "d"
return r
2012-01-06 17:00:23 -05:00
def __eq__(self, other):
return self.width == other.width and self.signed == other.signed
class Value:
2011-12-08 15:15:44 -05:00
def __invert__(self):
2011-12-16 15:30:14 -05:00
return _Operator("~", [self])
2011-12-08 15:15:44 -05:00
def __add__(self, other):
2011-12-16 15:30:14 -05:00
return _Operator("+", [self, other])
def __radd__(self, other):
2011-12-16 15:30:14 -05:00
return _Operator("+", [other, self])
def __sub__(self, other):
2011-12-16 15:30:14 -05:00
return _Operator("-", [self, other])
def __rsub__(self, other):
2011-12-16 15:30:14 -05:00
return _Operator("-", [other, self])
def __mul__(self, other):
2011-12-16 15:30:14 -05:00
return _Operator("*", [self, other])
def __rmul__(self, other):
2011-12-16 15:30:14 -05:00
return _Operator("*", [other, self])
def __lshift__(self, other):
2011-12-16 15:30:14 -05:00
return _Operator("<<", [self, other])
def __rlshift__(self, other):
2011-12-16 15:30:14 -05:00
return _Operator("<<", [other, self])
def __rshift__(self, other):
2011-12-16 15:30:14 -05:00
return _Operator(">>", [self, other])
def __rrshift__(self, other):
2011-12-16 15:30:14 -05:00
return _Operator(">>", [other, self])
def __and__(self, other):
2011-12-16 15:30:14 -05:00
return _Operator("&", [self, other])
def __rand__(self, other):
2011-12-16 15:30:14 -05:00
return _Operator("&", [other, self])
def __xor__(self, other):
2011-12-16 15:30:14 -05:00
return _Operator("^", [self, other])
def __rxor__(self, other):
2011-12-16 15:30:14 -05:00
return _Operator("^", [other, self])
def __or__(self, other):
2011-12-16 15:30:14 -05:00
return _Operator("|", [self, other])
def __ror__(self, other):
2011-12-16 15:30:14 -05:00
return _Operator("|", [other, self])
def __lt__(self, other):
2011-12-16 15:30:14 -05:00
return _Operator("<", [self, other])
def __le__(self, other):
2011-12-16 15:30:14 -05:00
return _Operator("<=", [self, other])
def __eq__(self, other):
2011-12-16 15:30:14 -05:00
return _Operator("==", [self, other])
def __ne__(self, other):
2011-12-16 15:30:14 -05:00
return _Operator("!=", [self, other])
def __gt__(self, other):
2011-12-16 15:30:14 -05:00
return _Operator(">", [self, other])
def __ge__(self, other):
2011-12-16 15:30:14 -05:00
return _Operator(">=", [self, other])
def __getitem__(self, key):
if isinstance(key, int):
2011-12-21 16:57:07 -05:00
return _Slice(self, key, key+1)
elif isinstance(key, slice):
start = key.start or 0
2012-07-13 12:32:54 -04:00
stop = key.stop or len(self)
if stop > len(self):
stop = len(self)
if key.step != None:
raise KeyError
2011-12-21 16:57:07 -05:00
return _Slice(self, start, stop)
else:
raise KeyError
2011-12-16 15:30:14 -05:00
def eq(self, r):
2011-12-21 16:57:07 -05:00
return _Assign(self, r)
def __hash__(self):
return id(self)
2011-12-16 15:30:14 -05:00
class _Operator(Value):
def __init__(self, op, operands):
self.op = op
self.operands = list(map(_cst, operands))
2011-12-21 16:57:07 -05:00
class _Slice(Value):
def __init__(self, value, start, stop):
self.value = value
self.start = start
self.stop = stop
class Cat(Value):
def __init__(self, *args):
self.l = list(map(_cst, args))
2011-12-09 07:11:34 -05:00
class Replicate(Value):
def __init__(self, v, n):
2011-12-22 18:35:13 -05:00
self.v = _cst(v)
2011-12-09 07:11:34 -05:00
self.n = n
class Constant(Value):
def __init__(self, n, bv=None):
self.bv = bv or BV(bits_for(n), n < 0)
self.n = n
2011-12-09 07:11:34 -05:00
2012-07-13 12:16:50 -04:00
def __len__(self):
return self.bv.width
2011-12-09 07:11:34 -05:00
def __repr__(self):
return str(self.bv) + str(self.n)
2012-01-07 06:29:47 -05:00
def __eq__(self, other):
return self.bv == other.bv and self.n == other.n
2012-11-09 14:17:43 -05:00
def __hash__(self):
return id(self)
def binc(x, signed=False):
return Constant(int(x, 2), BV(len(x), signed))
def _cst(x):
if isinstance(x, int):
return Constant(x)
else:
return x
class Signal(Value):
2012-09-11 03:59:37 -04:00
_counter = 0
2012-01-16 12:09:52 -05:00
def __init__(self, bv=BV(), name=None, variable=False, reset=0, name_override=None):
2012-02-17 17:50:54 -05:00
assert(isinstance(bv, BV))
self.bv = bv
self.variable = variable
self.reset = Constant(reset, bv)
2012-01-16 12:09:52 -05:00
self.name_override = name_override
self.backtrace = tracer.trace_back(name)
2012-09-11 03:59:37 -04:00
self.order = Signal._counter
Signal._counter += 1
2012-04-08 12:06:22 -04:00
def __len__(self):
return self.bv.width
2012-01-06 05:20:33 -05:00
def __repr__(self):
2012-09-09 13:33:55 -04:00
return "<Signal " + (self.backtrace[-1][0] or "anonymous") + " at " + hex(id(self)) + ">"
# statements
2011-12-21 16:57:07 -05:00
class _Assign:
def __init__(self, l, r):
self.l = l
self.r = _cst(r)
2011-12-16 15:30:14 -05:00
class If:
def __init__(self, cond, *t):
self.cond = cond
2012-07-13 11:07:56 -04:00
self.t = list(t)
self.f = []
2011-12-16 15:30:14 -05:00
def Else(self, *f):
2012-07-13 11:07:56 -04:00
_insert_else(self, list(f))
2011-12-16 15:30:14 -05:00
return self
def Elif(self, cond, *t):
2012-07-13 11:07:56 -04:00
_insert_else(self, [If(cond, *t)])
2011-12-16 15:30:14 -05:00
return self
2011-12-17 14:31:42 -05:00
def _insert_else(obj, clause):
o = obj
2012-07-13 11:07:56 -04:00
while o.f:
assert(len(o.f) == 1)
assert(isinstance(o.f[0], If))
o = o.f[0]
2011-12-17 14:31:42 -05:00
o.f = clause
2011-12-16 15:30:14 -05:00
class Default:
pass
class Case:
2011-12-16 15:30:14 -05:00
def __init__(self, test, *cases):
self.test = test
2012-07-13 11:07:56 -04:00
self.cases = [(c[0], list(c[1:])) for c in cases if not isinstance(c[0], Default)]
2011-12-16 15:30:14 -05:00
self.default = None
for c in cases:
if isinstance(c[0], Default):
if self.default is not None:
raise ValueError
2012-07-13 11:07:56 -04:00
self.default = list(c[1:])
2011-12-16 15:30:14 -05:00
if self.default is None:
2012-07-13 11:07:56 -04:00
self.default = []
# arrays
class _ArrayProxy(Value):
def __init__(self, choices, key):
self.choices = choices
self.key = key
def __getattr__(self, attr):
return _ArrayProxy([getattr(choice, attr) for choice in self.choices],
self.key)
def __getitem__(self, key):
return _ArrayProxy([choice.__getitem__(key) for choice in self.choices],
self.key)
class Array(list):
def __getitem__(self, key):
if isinstance(key, Value):
return _ArrayProxy(self, key)
else:
return super().__getitem__(key)
# extras
2011-12-08 10:35:32 -05:00
class Instance:
def __init__(self, of, *items, name=""):
2011-12-08 10:35:32 -05:00
self.of = of
if name:
2012-01-20 16:36:17 -05:00
self.name_override = name
2011-12-08 10:35:32 -05:00
else:
2012-01-20 16:36:17 -05:00
self.name_override = of
self.items = items
class _IO:
def __init__(self, name, expr=BV(1)):
self.name = name
if isinstance(expr, BV):
self.expr = Signal(expr, name)
elif isinstance(expr, int):
self.expr = Constant(expr)
2011-12-08 12:56:14 -05:00
else:
self.expr = expr
class Input(_IO):
pass
class Output(_IO):
pass
class InOut(_IO):
pass
class Parameter:
def __init__(self, name, value):
self.name = name
self.value = value
class _CR:
def __init__(self, name_inst, domain="sys", invert=False):
self.name_inst = name_inst
self.domain = domain
self.invert = invert
class ClockPort(_CR):
pass
class ResetPort(_CR):
pass
def get_io(self, name):
for item in self.items:
if isinstance(item, Instance._IO) and item.name == name:
2012-09-28 12:02:03 -04:00
return item.expr
2011-12-08 10:35:32 -05:00
def __hash__(self):
return id(self)
2012-01-27 14:22:17 -05:00
(READ_FIRST, WRITE_FIRST, NO_CHANGE) = range(3)
2012-01-27 10:53:34 -05:00
class MemoryPort:
2012-01-27 14:22:17 -05:00
def __init__(self, adr, dat_r, we=None, dat_w=None,
async_read=False, re=None, we_granularity=0, mode=WRITE_FIRST,
clock_domain="sys"):
2012-01-27 10:53:34 -05:00
self.adr = adr
self.dat_r = dat_r
self.we = we
self.dat_w = dat_w
self.async_read = async_read
self.re = re
self.we_granularity = we_granularity
2012-01-27 14:22:17 -05:00
self.mode = mode
self.clock_domain = clock_domain
2012-01-27 10:53:34 -05:00
class Memory:
def __init__(self, width, depth, *ports, init=None):
self.width = width
self.depth = depth
self.ports = ports
self.init = init
#
class Fragment:
2012-04-02 13:21:43 -04:00
def __init__(self, comb=None, sync=None, instances=None, memories=None, sim=None):
if comb is None: comb = []
if sync is None: sync = dict()
if instances is None: instances = set()
if memories is None: memories = set()
if sim is None: sim = []
if isinstance(sync, list):
sync = {"sys": sync}
2012-07-13 11:07:56 -04:00
self.comb = comb
self.sync = sync
self.instances = set(instances)
self.memories = set(memories)
self.sim = sim
2011-12-05 11:43:56 -05:00
def __add__(self, other):
newsync = defaultdict(list)
for k, v in self.sync.items():
newsync[k] = v[:]
for k, v in other.sync.items():
newsync[k].extend(v)
return Fragment(self.comb + other.comb, newsync,
self.instances | other.instances,
self.memories | other.memories,
self.sim + other.sim)
def rename_clock_domain(self, old, new):
self.sync["new"] = self.sync["old"]
del self.sync["old"]
for inst in self.instances:
for cr in filter(lambda x: isinstance(x, Instance._CR), inst.items):
if cr.domain == old:
cr.domain = new
for mem in self.memories:
for port in mem.ports:
if port.clock_domain == old:
port.clock_domain = new
def get_clock_domains(self):
r = set(self.sync.keys())
r |= set(cr.domain
for inst in self.instances
for cr in filter(lambda x: isinstance(x, Instance._CR), inst.items))
r |= set(port.clock_domain
for mem in self.memories
for port in mem.ports)
return r
2012-03-06 13:29:39 -05:00
def call_sim(self, simulator):
for s in self.sim:
if simulator.cycle_counter >= 0 or (hasattr(s, "initialize") and s.initialize):
s(simulator)
class ClockDomain:
def __init__(self, n1, n2=None):
if n2 is None:
n_clk = n1 + "_clk"
n_rst = n1 + "_rst"
else:
n_clk = n1
n_rst = n2
self.clk = Signal(name_override=n_clk)
self.rst = Signal(name_override=n_rst)