mirror of
https://github.com/enjoy-digital/litex.git
synced 2025-01-04 09:52:26 -05:00
Initial Efinix Trion support
This commit is contained in:
parent
7f8e2e39f3
commit
000aabf85b
5 changed files with 463 additions and 0 deletions
1
litex/build/efinix/__init__.py
Normal file
1
litex/build/efinix/__init__.py
Normal file
|
@ -0,0 +1 @@
|
|||
from litex.build.efinix.programmer import EfinixProgrammer
|
8
litex/build/efinix/common.py
Normal file
8
litex/build/efinix/common.py
Normal file
|
@ -0,0 +1,8 @@
|
|||
#
|
||||
# This file is part of LiteX.
|
||||
#
|
||||
# Copyright (c) 2021 Franck Jullien <franck.jullien@collshade.fr>
|
||||
# Copyright (c) 2015-2018 Florent Kermarrec <florent@enjoy-digital.fr>
|
||||
# SPDX-License-Identifier: BSD-2-Clause
|
||||
|
||||
efinix_special_overrides = {}
|
375
litex/build/efinix/efinity.py
Normal file
375
litex/build/efinix/efinity.py
Normal file
|
@ -0,0 +1,375 @@
|
|||
#
|
||||
# This file is part of LiteX.
|
||||
#
|
||||
# Copyright (c) 2021 Franck Jullien <franck.jullien@collshade.fr>
|
||||
# Copyright (c) 2015-2018 Florent Kermarrec <florent@enjoy-digital.fr>
|
||||
# SPDX-License-Identifier: BSD-2-Clause
|
||||
|
||||
import os
|
||||
import subprocess
|
||||
import pathlib
|
||||
import math
|
||||
import sys
|
||||
import site
|
||||
import subprocess
|
||||
import inspect
|
||||
|
||||
from lxml import etree
|
||||
|
||||
from litex.build.generic_platform import *
|
||||
|
||||
from migen.fhdl.structure import _Fragment
|
||||
from migen.fhdl.tools import *
|
||||
from migen.fhdl.namer import build_namespace
|
||||
|
||||
from litex.build.generic_platform import Pins, IOStandard, Misc
|
||||
from litex.build import tools
|
||||
|
||||
_reserved_keywords = {
|
||||
"always", "and", "assign", "automatic", "begin", "buf", "bufif0", "bufif1",
|
||||
"case", "casex", "casez", "cell", "cmos", "config", "deassign", "default",
|
||||
"defparam", "design", "disable", "edge", "else", "end", "endcase",
|
||||
"endconfig", "endfunction", "endgenerate", "endmodule", "endprimitive",
|
||||
"endspecify", "endtable", "endtask", "event", "for", "force", "forever",
|
||||
"fork", "function", "generate", "genvar", "highz0", "highz1", "if",
|
||||
"ifnone", "incdir", "include", "initial", "inout", "input",
|
||||
"instance", "integer", "join", "large", "liblist", "library", "localparam",
|
||||
"macromodule", "medium", "module", "nand", "negedge", "nmos", "nor",
|
||||
"noshowcancelled", "not", "notif0", "notif1", "or", "output", "parameter",
|
||||
"pmos", "posedge", "primitive", "pull0", "pull1" "pulldown",
|
||||
"pullup", "pulsestyle_onevent", "pulsestyle_ondetect", "remos", "real",
|
||||
"realtime", "reg", "release", "repeat", "rnmos", "rpmos", "rtran",
|
||||
"rtranif0", "rtranif1", "scalared", "showcancelled", "signed", "small",
|
||||
"specify", "specparam", "strong0", "strong1", "supply0", "supply1",
|
||||
"table", "task", "time", "tran", "tranif0", "tranif1", "tri", "tri0",
|
||||
"tri1", "triand", "trior", "trireg", "unsigned", "use", "vectored", "wait",
|
||||
"wand", "weak0", "weak1", "while", "wire", "wor","xnor", "xor", "do"
|
||||
}
|
||||
|
||||
def get_pin_direction(fragment, platform, pinname):
|
||||
ios = platform.constraint_manager.get_io_signals()
|
||||
sigs = list_signals(fragment) | list_special_ios(fragment, True, True, True)
|
||||
special_outs = list_special_ios(fragment, False, True, True)
|
||||
inouts = list_special_ios(fragment, False, False, True)
|
||||
targets = list_targets(fragment) | special_outs
|
||||
|
||||
ns = build_namespace(list_signals(fragment) \
|
||||
| list_special_ios(fragment, True, True, True) \
|
||||
| ios, _reserved_keywords)
|
||||
ns.clock_domains = fragment.clock_domains
|
||||
|
||||
dir = "Unknown"
|
||||
|
||||
for sig in sorted(ios, key=lambda x: x.duid):
|
||||
# Better idea ???
|
||||
if (pinname.split('[')[0] == ns.get_name(sig)):
|
||||
if sig in inouts:
|
||||
dir = "inout"
|
||||
elif sig in targets:
|
||||
dir = "output"
|
||||
else:
|
||||
dir = "input"
|
||||
|
||||
return dir
|
||||
|
||||
# Timing Constraints (.sdc) ------------------------------------------------------------------------
|
||||
|
||||
def _build_sdc(clocks, false_paths, vns, named_sc, build_name, additional_sdc_commands):
|
||||
sdc = []
|
||||
|
||||
# Clock constraints
|
||||
for clk, period in sorted(clocks.items(), key=lambda x: x[0].duid):
|
||||
is_port = False
|
||||
for sig, pins, others, resname in named_sc:
|
||||
if sig == vns.get_name(clk):
|
||||
is_port = True
|
||||
if is_port:
|
||||
tpl = "create_clock -name {clk} -period {period} [get_ports {{{clk}}}]"
|
||||
sdc.append(tpl.format(clk=vns.get_name(clk), period=str(period)))
|
||||
else:
|
||||
tpl = "create_clock -name {clk} -period {period} [get_nets {{{clk}}}]"
|
||||
sdc.append(tpl.format(clk=vns.get_name(clk), period=str(period)))
|
||||
|
||||
# False path constraints
|
||||
for from_, to in sorted(false_paths, key=lambda x: (x[0].duid, x[1].duid)):
|
||||
tpl = "set_false_path -from [get_clocks {{{from_}}}] -to [get_clocks {{{to}}}]"
|
||||
sdc.append(tpl.format(from_=vns.get_name(from_), to=vns.get_name(to)))
|
||||
|
||||
# Add additional commands
|
||||
sdc += additional_sdc_commands
|
||||
|
||||
# Generate .sdc
|
||||
tools.write_to_file("{}.sdc".format(build_name), "\n".join(sdc))
|
||||
|
||||
# Peripheral configuration ------------------------------------------------------------------------
|
||||
|
||||
def _create_gpio_instance(fragment, platform, sig, pins):
|
||||
l = ""
|
||||
if len(pins) > 1:
|
||||
l = ',{},0'.format(len(pins) - 1)
|
||||
d = get_pin_direction(fragment, platform, sig)
|
||||
return 'design.create_{d}_gpio("{name}"{len})'.format(d =d, name=sig, len=l)
|
||||
|
||||
def _format_constraint(c, signame, fmt_r, fragment, platform):
|
||||
# IO location constraints
|
||||
if isinstance(c, Pins):
|
||||
tpl = 'design.assign_pkg_pin("{signame}","{pin}")\n'
|
||||
return tpl.format(signame=signame, name=fmt_r, pin=c.identifiers[0])
|
||||
|
||||
# IO standard property
|
||||
elif isinstance(c, IOStandard):
|
||||
prop = ""
|
||||
valid = ['3.3_V_LVTTL_/_LVCMOS', '2.5_V_LVCMOS', '1.8_V_LVCMOS']
|
||||
if c.name in valid:
|
||||
prop = 'IO_STANDARD'
|
||||
|
||||
if prop == "":
|
||||
print("{} has a wrong IOStandard format [{}]".format(signame, c.name))
|
||||
print("Sould be selected from {}\n".format(valid))
|
||||
# Print error, warning ??
|
||||
return ""
|
||||
|
||||
tpl = 'design.set_property( "{signame}","{prop}","{val}")\n'
|
||||
return tpl.format(signame=signame, prop=prop, val=c.name)
|
||||
|
||||
# Others constraints
|
||||
elif isinstance(c, Misc):
|
||||
prop = ""
|
||||
if c.misc in ['WEAK_PULLUP', 'WEAK_PULLDOWN']:
|
||||
prop = 'PULL_OPTION'
|
||||
val = c.misc
|
||||
|
||||
if 'DRIVE_STRENGTH' in c.misc:
|
||||
prop = 'DRIVE_STRENGTH'
|
||||
val = c.misc.split('=')[1]
|
||||
|
||||
if prop == "":
|
||||
# Print error, warning ??
|
||||
return ""
|
||||
|
||||
tpl = 'design.set_property( "{signame}","{prop}","{val}")\n'
|
||||
return tpl.format(signame=signame, prop=prop, val=val)
|
||||
|
||||
def _format_conf_constraint(signame, pin, others, resname, fragment, platform):
|
||||
fmt_r = "{}:{}".format(*resname[:2])
|
||||
if resname[2] is not None:
|
||||
fmt_r += "." + resname[2]
|
||||
fmt_c = [_format_constraint(c, signame, fmt_r, fragment, platform) for c in ([Pins(pin)] + others)]
|
||||
return ''.join(fmt_c)
|
||||
|
||||
def _build_iface_conf(named_sc, named_pc, fragment, platform):
|
||||
conf = []
|
||||
inst = []
|
||||
|
||||
# GPIO
|
||||
for sig, pins, others, resname in named_sc:
|
||||
inst.append(_create_gpio_instance(fragment, platform, sig, pins))
|
||||
if len(pins) > 1:
|
||||
for i, p in enumerate(pins):
|
||||
conf.append(_format_conf_constraint("{}[{}]".format(sig, i), p, others, resname, fragment, platform))
|
||||
else:
|
||||
conf.append(_format_conf_constraint(sig, pins[0], others, resname, fragment, platform))
|
||||
if named_pc:
|
||||
conf.append("\n\n".join(named_pc))
|
||||
|
||||
# PLL
|
||||
#inst.append()
|
||||
|
||||
conf = inst + conf
|
||||
|
||||
return "\n".join(conf)
|
||||
|
||||
def _build_peri(efinity_path, build_name, partnumber, named_sc, named_pc, fragment, platform):
|
||||
pythonpath = ""
|
||||
|
||||
header = "# Autogenerated by LiteX / git: " + tools.get_litex_git_revision()
|
||||
|
||||
header += """
|
||||
import os
|
||||
import sys
|
||||
|
||||
home = '{0}'
|
||||
|
||||
os.environ['EFXPT_HOME'] = home + '/pt'
|
||||
os.environ['EFXPGM_HOME'] = home + '/pgm'
|
||||
os.environ['EFXDBG_HOME'] = home + '/debugger'
|
||||
os.environ['EFXIPM_HOME'] = home + '/ipm'
|
||||
|
||||
sys.path.append(home + '/pt/bin')
|
||||
sys.path.append(home + '/lib/python3.8/site-packages')
|
||||
|
||||
from api_service.design import DesignAPI
|
||||
from api_service.device import DeviceAPI
|
||||
|
||||
is_verbose = {1}
|
||||
|
||||
design = DesignAPI(is_verbose)
|
||||
device = DeviceAPI(is_verbose)
|
||||
|
||||
design.create('{2}', '{3}', './../build', overwrite=True)
|
||||
|
||||
"""
|
||||
|
||||
header = header.format(efinity_path, 'True', build_name, partnumber)
|
||||
|
||||
conf = _build_iface_conf(named_sc, named_pc, fragment, platform)
|
||||
|
||||
footer = """
|
||||
# Check design, generate constraints and reports
|
||||
#design.generate(enable_bitstream=True)
|
||||
# Save the configured periphery design
|
||||
design.save()
|
||||
"""
|
||||
|
||||
tools.write_to_file("iface.py", header + conf + footer)
|
||||
|
||||
subprocess.call([efinity_path + '/bin/python3', 'iface.py'])
|
||||
|
||||
# Project configuration ------------------------------------------------------------------------
|
||||
|
||||
def _build_xml(name, partnumber, build_name, sources, additional_xml_commands):
|
||||
|
||||
test = '<project xmlns:efx="http://www.efinixinc.com/enf_proj"'
|
||||
test += ' xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"'
|
||||
test += ' name="{}"'.format(name)
|
||||
test += ' description=""'
|
||||
test += ' last_change_date="mer. sept. 15 2021 12:04:56"'
|
||||
test += ' location="{}"'.format(pathlib.Path().resolve())
|
||||
test += ' sw_version="2021.1.165.2.19"'
|
||||
test += ' last_run_state="" last_run_tool="" last_run_flow=""'
|
||||
test += ' config_result_in_sync="sync" design_ood="sync" place_ood="sync" route_ood="sync"'
|
||||
test += ' xsi:schemaLocation="http://www.efinixinc.com/enf_proj enf_proj.xsd"'
|
||||
test += '/>'
|
||||
|
||||
root = etree.XML(test)
|
||||
|
||||
device_info = etree.SubElement(root, "device_info")
|
||||
etree.SubElement(device_info, "family", name = "Trion")
|
||||
etree.SubElement(device_info, "device", name = partnumber)
|
||||
etree.SubElement(device_info, "timing_model", name = "C4")
|
||||
|
||||
design_info = etree.SubElement(root, "design_info")
|
||||
etree.SubElement(design_info, "top_module", name = build_name)
|
||||
for filename, language, library in sources:
|
||||
val = {'name':filename, 'version':'default', 'library':'default'}
|
||||
etree.SubElement(design_info, "design_file", val)
|
||||
etree.SubElement(design_info, "top_vhdl_arch", name = "")
|
||||
|
||||
constraint_info = etree.SubElement(root, "constraint_info")
|
||||
etree.SubElement(constraint_info, "sdc_file", name = "{}.sdc".format(build_name))
|
||||
|
||||
misc_info = etree.SubElement(root, "misc_info")
|
||||
ip_info = etree.SubElement(root, "ip_info")
|
||||
|
||||
synthesis = etree.SubElement(root, "synthesis", tool_name="efx_map")
|
||||
for l in additional_xml_commands:
|
||||
if l[0] == 'efx_map':
|
||||
val = {'name':l[1], 'value':l[2], 'value_type':l[3]}
|
||||
etree.SubElement(synthesis, "param", val)
|
||||
|
||||
place_and_route = etree.SubElement(root, "place_and_route", tool_name="efx_pnr")
|
||||
for l in additional_xml_commands:
|
||||
if l[0] == 'efx_pnr':
|
||||
val = {'name':l[1], 'value':l[2], 'value_type':l[3]}
|
||||
etree.SubElement(place_and_route, "param", val)
|
||||
|
||||
bitstream_generation = etree.SubElement(root, "bitstream_generation", tool_name="efx_pgm")
|
||||
for l in additional_xml_commands:
|
||||
if l[0] == 'efx_pgm':
|
||||
val = {'name':l[1], 'value':l[2], 'value_type':l[3]}
|
||||
etree.SubElement(bitstream_generation, "param", val)
|
||||
|
||||
# Hack
|
||||
output = etree.tostring(root, pretty_print=True).decode().replace('<', '<efx:')
|
||||
output = output.replace('<efx:/', '</efx:')
|
||||
|
||||
# Generate .xml
|
||||
tools.write_to_file("{}.xml".format(build_name), output)
|
||||
|
||||
class EfinityToolchain():
|
||||
attr_translate = {}
|
||||
|
||||
def __init__(self, efinity_path):
|
||||
self.clocks = dict()
|
||||
self.false_paths = set()
|
||||
self.efinity_path = efinity_path
|
||||
self.additional_sdc_commands = []
|
||||
self.additional_xml_commands = []
|
||||
|
||||
def build(self, platform, fragment,
|
||||
build_dir = "build",
|
||||
build_name = "top",
|
||||
run = True,
|
||||
**kwargs):
|
||||
|
||||
# Create build directory
|
||||
cwd = os.getcwd()
|
||||
os.makedirs(build_dir, exist_ok=True)
|
||||
os.chdir(build_dir)
|
||||
|
||||
# Finalize design
|
||||
if not isinstance(fragment, _Fragment):
|
||||
fragment = fragment.get_fragment()
|
||||
platform.finalize(fragment)
|
||||
|
||||
# Generate verilog
|
||||
v_output = platform.get_verilog(fragment, name=build_name, **kwargs)
|
||||
named_sc, named_pc = platform.resolve_signals(v_output.ns)
|
||||
v_file = build_name + ".v"
|
||||
v_output.write(v_file)
|
||||
platform.add_source(v_file)
|
||||
|
||||
if platform.verilog_include_paths:
|
||||
self.options['includ_path'] = '{' + ';'.join(platform.verilog_include_paths) + '}'
|
||||
|
||||
os.environ['EFXPT_HOME'] = self.efinity_path + '/pt'
|
||||
|
||||
# Generate design timing constraints file (.sdc)
|
||||
_build_sdc(
|
||||
clocks = self.clocks,
|
||||
false_paths = self.false_paths,
|
||||
vns = v_output.ns,
|
||||
named_sc = named_sc,
|
||||
build_name = build_name,
|
||||
additional_sdc_commands = self.additional_sdc_commands)
|
||||
|
||||
# Generate project file (.xml)
|
||||
_build_xml(
|
||||
name = build_name,
|
||||
partnumber = platform.device,
|
||||
build_name = build_name,
|
||||
sources = platform.sources,
|
||||
additional_xml_commands = self.additional_xml_commands)
|
||||
|
||||
# Generate constraints file (.peri.xml)
|
||||
_build_peri(
|
||||
efinity_path = self.efinity_path,
|
||||
build_name = build_name,
|
||||
partnumber = platform.device,
|
||||
named_sc = named_sc,
|
||||
named_pc = named_pc,
|
||||
fragment = fragment,
|
||||
platform = platform)
|
||||
|
||||
# Run
|
||||
if run:
|
||||
subprocess.call([self.efinity_path + '/scripts/efx_run.py', build_name + '.xml', '-f', 'compile'])
|
||||
|
||||
os.chdir(cwd)
|
||||
|
||||
return v_output.ns
|
||||
|
||||
def add_period_constraint(self, platform, clk, period):
|
||||
clk.attr.add("keep")
|
||||
period = math.floor(period*1e3)/1e3 # round to lowest picosecond
|
||||
if clk in self.clocks:
|
||||
if period != self.clocks[clk]:
|
||||
raise ValueError("Clock already constrained to {:.2f}ns, new constraint to {:.2f}ns"
|
||||
.format(self.clocks[clk], period))
|
||||
self.clocks[clk] = period
|
||||
|
||||
def add_false_path_constraint(self, platform, from_, to):
|
||||
from_.attr.add("keep")
|
||||
to.attr.add("keep")
|
||||
if (to, from_) not in self.false_paths:
|
||||
self.false_paths.add((from_, to))
|
52
litex/build/efinix/platform.py
Normal file
52
litex/build/efinix/platform.py
Normal file
|
@ -0,0 +1,52 @@
|
|||
#
|
||||
# This file is part of LiteX.
|
||||
#
|
||||
# Copyright (c) 2021 Franck Jullien <franck.jullien@collshade.fr>
|
||||
# Copyright (c) 2015-2018 Florent Kermarrec <florent@enjoy-digital.fr>
|
||||
# SPDX-License-Identifier: BSD-2-Clause
|
||||
|
||||
import os
|
||||
|
||||
from litex.build.generic_platform import GenericPlatform
|
||||
from litex.build.efinix import common, efinity
|
||||
|
||||
# EfinixPlatform -----------------------------------------------------------------------------------
|
||||
|
||||
class EfinixPlatform(GenericPlatform):
|
||||
bitstream_ext = ".bit"
|
||||
|
||||
def __init__(self, *args, toolchain="efinity", **kwargs):
|
||||
GenericPlatform.__init__(self, *args, **kwargs)
|
||||
|
||||
if 'LITEX_ENV_EFINITY' in os.environ:
|
||||
self.efinity_path = os.environ['LITEX_ENV_EFINITY'].rstrip('/')
|
||||
os.environ['EFINITY_HOME'] = self.efinity_path
|
||||
else:
|
||||
raise OSError('Unable to find Efinity toolchain, please set LITEX_ENV_EFINITY to ${install_dir}')
|
||||
|
||||
if toolchain == "efinity":
|
||||
self.toolchain = efinity.EfinityToolchain(self.efinity_path)
|
||||
else:
|
||||
raise ValueError("Unknown toolchain")
|
||||
|
||||
def get_verilog(self, *args, special_overrides=dict(), **kwargs):
|
||||
so = dict(common.efinix_special_overrides)
|
||||
so.update(special_overrides)
|
||||
return GenericPlatform.get_verilog(self, *args, special_overrides=so,
|
||||
attr_translate=self.toolchain.attr_translate, **kwargs)
|
||||
|
||||
def build(self, *args, **kwargs):
|
||||
return self.toolchain.build(self, *args, **kwargs)
|
||||
|
||||
def add_period_constraint(self, clk, period):
|
||||
if clk is None: return
|
||||
if hasattr(clk, "p"):
|
||||
clk = clk.p
|
||||
self.toolchain.add_period_constraint(self, clk, period)
|
||||
|
||||
def add_false_path_constraint(self, from_, to):
|
||||
if hasattr(from_, "p"):
|
||||
from_ = from_.p
|
||||
if hasattr(to, "p"):
|
||||
to = to.p
|
||||
self.toolchain.add_false_path_constraint(self, from_, to)
|
27
litex/build/efinix/programmer.py
Normal file
27
litex/build/efinix/programmer.py
Normal file
|
@ -0,0 +1,27 @@
|
|||
#
|
||||
# This file is part of LiteX.
|
||||
#
|
||||
# Copyright (c) 2021 Franck Jullien <franck.jullien@collshade.fr>
|
||||
# SPDX-License-Identifier: BSD-2-Clause
|
||||
|
||||
import os
|
||||
import sys
|
||||
|
||||
from litex.build.generic_programmer import GenericProgrammer
|
||||
|
||||
class EfinixProgrammer(GenericProgrammer):
|
||||
|
||||
def __init__(self, cable_name=""):
|
||||
self.cable_name = cable_name
|
||||
if 'LITEX_ENV_EFINITY' in os.environ:
|
||||
self.efinity_path = os.environ['LITEX_ENV_EFINITY'].rstrip('/')
|
||||
os.environ['EFINITY_HOME'] = self.efinity_path
|
||||
else:
|
||||
raise OSError('Unable to find Efinity toolchain, please set LITEX_ENV_EFINITY to ${install_dir}')
|
||||
|
||||
def load_bitstream(self, bitstream_file, cable_suffix=""):
|
||||
os.environ['EFXPGM_HOME'] = self.efinity_path + '/pgm'
|
||||
self.call([self.efinity_path + '/bin/python3', self.efinity_path +
|
||||
'pgm/bin/efx_pgm/ftdi_program.py', bitstream_file,
|
||||
"-m", "jtag"
|
||||
])
|
Loading…
Reference in a new issue