blob: 6b0742e9759ef7ec118be349e2b100c4a276dd58 [file] [log] [blame]
"""
Interface between Python fuzzer scripts and Lattice Diamond ispTcl
"""
import database
import subprocess
import tempfile
from os import path
import re
def run(commands):
"""Run a list of Tcl commands, returning the output as a string"""
dtcl_path = path.join(database.get_trellis_root(), "diamond_tcl.sh")
workdir = tempfile.mkdtemp()
scriptfile = path.join(workdir, "script.tcl")
with open(scriptfile, 'w') as f:
f.write('source $::env(FOUNDRY)/data/tcltool/IspTclDev.tcl\n')
f.write('source $::env(FOUNDRY)/data/tcltool/IspTclCmd.tcl\n')
for c in commands:
f.write(c + '\n')
result = subprocess.run(["bash", dtcl_path, scriptfile], cwd=workdir).returncode
assert result == 0, "ispTcl returned non-zero status code {}".format(result)
outfile = path.join(workdir, 'ispTcl.log')
with open(outfile, 'r') as f:
output = f.read()
# Strip Lattice header
delimiter = "-" * 80
output = output[output.rindex(delimiter)+81:].strip()
# Strip Lattice pleasantry
pleasantry = "Thank you for using ispTcl."
output = output.replace(pleasantry, "").strip()
return output
# All these following commands require a tuple (ncdfile, prffile) containing
# a specimen design for the target device
def run_ncd_prf(desfiles, commands):
"""
Run a list of Tcl commands after loading given .ncd and .prf files
desfiles: a tuple (ncdfile, prffile)
commands: list of Tcl commands to run
Returns the output from IspTcl, excluding header and pleasantry
"""
run_cmds = [
"des_read_ncd {}".format(path.abspath(desfiles[0])),
"des_read_prf {}".format(path.abspath(desfiles[1]))
] + commands
result = run(run_cmds)
# Remove output of des_read_x
is_header = True
output = ""
for line in result.split('\n'):
if line.startswith("Reading preference file"):
is_header = False
elif not is_header:
output += line
output += "\n"
output = output.replace("ERROR: Placement is not performanced.", "")
return output
def get_wires_at_position(desfiles, position):
"""
Use ispTcl to get a list of wires at a given grid position
desfiles: a tuple (ncdfile, prffile)
position: a tuple (row, col)
Returns a list of tuples
(name, type, typeid)
"""
command = ["dev_list_node_at_location -row {} -col {}".format(position[0], position[1])]
result = run_ncd_prf(desfiles, command)
wires = []
for line in result.split('\n'):
sline = line.strip()
if sline == "":
continue
splitline = re.split('\s+', sline)
assert len(splitline) >= 3
wires.append((splitline[2].strip(), splitline[0].strip(), int(splitline[1])))
return wires
def get_arcs_on_wires(desfiles, wires, drivers_only=False):
"""
Use ispTcl to get a list of arcs sinking or sourcing a list of wires
desfiles: a tuple (ncdfile, prffile)
wires: list of canonical names of the wire
drivers_only: only include arcs driving the wire in the output
Returns a map between wire name and a list of arc tuples (source, sink)
"""
arcmap = {}
wire_idx = 0
# We can only process a limited number of nodes at a time, due to a memory leak in the Tcl API :facepalm:
for i in range(0, len(wires), 10):
subwires = wires[i:i+10]
command = []
for wire in subwires:
command += ["dev_list_arc_by_node_name -to {} -num 100000".format(wire), 'prj_list']
result = run_ncd_prf(desfiles, command)
arcs = []
for line in result.split('\n'):
sline = line.strip()
if sline == "":
pass
elif sline.startswith("MyIspProject"):
arcmap[wires[wire_idx]] = list(arcs)
wire_idx += 1
arcs = []
else:
splitline = re.split('\s+', sline)
assert len(splitline) >= 3
if splitline[1].strip() == "-->":
arcs.append((splitline[0].strip(), splitline[2].strip()))
elif splitline[1].strip() == "<--":
if not drivers_only:
arcs.append((splitline[2].strip(), splitline[0].strip()))
elif splitline[1].strip() == "---":
# Edge wires, currently ignored
pass
else:
print (splitline)
assert False, "invalid output from Tcl command `dev_list_arcs`"
return arcmap
def main():
print(run([]))
if __name__ == "__main__":
main()