litex/migen/fhdl/verilog.py

104 lines
2.8 KiB
Python
Raw Normal View History

2011-12-04 16:26:32 -05:00
from .structure import *
2011-12-04 16:41:50 -05:00
from .convtools import *
2011-12-04 16:26:32 -05:00
from functools import partial
2011-12-05 11:43:56 -05:00
def Convert(f, outs=set(), ins=set(), name="top", clkname="sys_clk", rstname="sys_rst"):
2011-12-04 16:26:32 -05:00
ns = Namespace()
2011-12-04 16:41:50 -05:00
clks = Signal(name=clkname)
rsts = Signal(name=rstname)
2011-12-04 16:26:32 -05:00
clk = ns.GetName(clks)
rst = ns.GetName(rsts)
def printsig(s):
if s.bv.signed:
n = "signed "
else:
n = ""
if s.bv.width > 1:
n += "[" + str(s.bv.width-1) + ":0] "
n += ns.GetName(s)
return n
2011-12-05 13:25:32 -05:00
def printexpr(node):
2011-12-04 16:26:32 -05:00
if isinstance(node, Constant):
2011-12-05 13:25:32 -05:00
if node.n >= 0:
return str(node.bv) + str(node.n)
else:
return "-" + str(node.bv) + str(-self.n)
2011-12-04 16:26:32 -05:00
elif isinstance(node, Signal):
return ns.GetName(node)
elif isinstance(node, Operator):
arity = len(node.operands)
if arity == 1:
2011-12-05 13:25:32 -05:00
r = self.op + printexpr(node.operands[0])
2011-12-04 16:26:32 -05:00
elif arity == 2:
2011-12-05 13:25:32 -05:00
r = printexpr(node.operands[0]) + " " + node.op + " " + printexpr(node.operands[1])
2011-12-04 16:26:32 -05:00
else:
raise TypeError
return "(" + r + ")"
elif isinstance(node, Slice):
if node.start + 1 == node.stop:
sr = "[" + str(node.start) + "]"
else:
sr = "[" + str(node.stop-1) + ":" + str(node.start) + "]"
2011-12-05 13:25:32 -05:00
return printexpr(node.value) + sr
2011-12-04 16:26:32 -05:00
elif isinstance(node, Cat):
2011-12-05 13:25:32 -05:00
l = list(map(printexpr, node.l))
2011-12-04 16:26:32 -05:00
l.reverse()
return "{" + ", ".join(l) + "}"
2011-12-05 13:25:32 -05:00
else:
raise TypeError
def printnode(level, node):
if isinstance(node, Assign):
2011-12-04 16:26:32 -05:00
# TODO: variables
2011-12-05 13:25:32 -05:00
return "\t"*level + printexpr(node.l) + " <= " + printexpr(node.r) + ";\n"
2011-12-04 16:26:32 -05:00
elif isinstance(node, StatementList):
return "".join(list(map(partial(printnode, level), node.l)))
elif isinstance(node, If):
2011-12-05 13:25:32 -05:00
r = "\t"*level + "if (" + printexpr(node.cond) + ") begin\n"
2011-12-04 16:26:32 -05:00
r += printnode(level + 1, node.t)
if node.f.l:
r += "\t"*level + "end else begin\n"
r += printnode(level + 1, node.f)
r += "\t"*level + "end\n"
return r
2011-12-05 11:43:56 -05:00
elif isinstance(node, Case):
2011-12-05 13:25:32 -05:00
r = "\t"*level + "case (" + printexpr(node.test) + ")\n"
2011-12-05 11:43:56 -05:00
for case in node.cases:
2011-12-05 13:25:32 -05:00
r += "\t"*(level + 1) + printexpr(case[0]) + ": begin\n"
2011-12-05 11:43:56 -05:00
r += printnode(level + 2, case[1])
r += "\t"*(level + 1) + "end\n"
r += "\t"*level + "endcase\n"
return r
2011-12-04 16:26:32 -05:00
else:
raise TypeError
r = "/* Autogenerated by Migen */\n"
r += "module " + name + "(\n"
r += "\tinput " + clk + ",\n"
r += "\tinput " + rst
if ins:
r += ",\n\tinput " + ",\n\tinput ".join(map(printsig, ins))
if outs:
r += ",\n\toutput reg " + ",\n\toutput reg ".join(map(printsig, outs))
r += "\n);\n\n"
sigs = ListSignals(f).difference(ins, outs)
for sig in sigs:
r += "reg " + printsig(sig) + ";\n"
r += "\n"
if f.comb.l:
r += "always @(*) begin\n"
r += printnode(1, f.comb)
r += "end\n\n"
if f.sync.l:
r += "always @(posedge " + clk + ") begin\n"
2011-12-04 16:41:50 -05:00
r += printnode(1, InsertReset(rsts, f.sync))
2011-12-04 16:26:32 -05:00
r += "end\n\n"
r += "endmodule\n"
return r