# EMACS settings: -*- tab-width: 2; indent-tabs-mode: t; python-indent-offset: 2 -*-
# vim: tabstop=2:shiftwidth=2:noexpandtab
# kate: tab-width 2; replace-tabs off; indent-width 2;
#
# ==============================================================================
# Authors: Patrick Lehmann
# Martin Zabel
#
# Python Class: Xilinx ISE specific classes
#
# License:
# ==============================================================================
# Copyright 2007-2016 Technische Universitaet Dresden - Germany
# Chair of VLSI-Design, Diagnostics and Architecture
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
# ==============================================================================
#
# load dependencies
from subprocess import check_output
from lib.Functions import CallByRefParam, Init
from Base.Exceptions import PlatformNotSupportedException
from Base.Executable import ExecutableArgument, ShortFlagArgument, ShortTupleArgument, StringArgument, CommandLineArgumentList, DryRunException
from Base.Logging import LogEntry, Severity
from Base.Project import Project as BaseProject, ProjectFile, ConstraintFile, FileTypes
from DataBase.Entity import SimulationResult
from ToolChain import ToolMixIn, ConfigurationException, ToolConfiguration, OutputFilteredExecutable
from ToolChain.GNU import Bash
from ToolChain.Windows import Cmd
from ToolChain.Xilinx import XilinxException
from Simulator import PoCSimulationResultFilter
__api__ = [
'ISEException',
'Configuration',
'ISE',
'Fuse',
'ISESimulator',
'Xst',
'CoreGenerator',
'VhCompFilter',
'FuseFilter',
'SimulatorFilter',
'XstFilter',
'CoreGeneratorFilter',
'ISEProject',
'ISEProjectFile',
'UserConstraintFile'
]
__all__ = __api__
[docs]class ISEException(XilinxException):
pass
[docs]class Configuration(ToolConfiguration):
_vendor = "Xilinx" #: The name of the tools vendor.
_toolName = "Xilinx ISE" #: The name of the tool.
_section = "INSTALL.Xilinx.ISE" #: The name of the configuration section. Pattern: ``INSTALL.Vendor.ToolName``.
_multiVersionSupport = True #: Xilinx ISE supports multiple versions installed on the same system.
_template = {
"Windows": {
_section: {
"Version": "14.7",
"SectionName": ("%{PathWithRoot}#${Version}", None),
"InstallationDirectory": ("${${SectionName}:InstallationDirectory}", "${INSTALL.Xilinx:InstallationDirectory}/${Version}/ISE_DS"),
"BinaryDirectory": ("${${SectionName}:BinaryDirectory}", "${InstallationDirectory}/ISE/bin/nt64")
}
},
"Linux": {
_section: {
"Version": "14.7",
"SectionName": ("%{PathWithRoot}#${Version}", None),
"InstallationDirectory": ("${${SectionName}:InstallationDirectory}", "${INSTALL.Xilinx:InstallationDirectory}/${Version}/ISE_DS"),
"BinaryDirectory": ("${${SectionName}:BinaryDirectory}", "${InstallationDirectory}/ISE/bin/lin64")
}
}
} #: The template for the configuration sections represented as nested dictionaries.
[docs] def CheckDependency(self):
"""Check if general Xilinx support is configured in PoC."""
return (len(self._host.PoCConfig['INSTALL.Xilinx']) != 0)
def __CheckISEVersion(self, binPath):
# check for ISE 14.7
if (self._host.Platform == "Windows"):
fusePath = binPath / "fuse.exe"
else:
fusePath = binPath / "fuse"
if not fusePath.exists():
raise ConfigurationException("Executable '{0!s}' not found.".format(fusePath)) from FileNotFoundError(str(fusePath))
output = check_output([str(fusePath), "--version"], universal_newlines=True)
if "P.20131013" not in output:
raise ConfigurationException("ISE version mismatch. Expected version 14.7 (P.20131013).")
[docs]class ISE(ToolMixIn):
[docs] def PreparseEnvironment(self, installationDirectory):
if (self._platform == "Linux"):
cmd = Bash(self._platform, self._dryrun, logger=self._logger)
settingsFile = installationDirectory / "settings64.sh"
elif (self._platform == "Windows"):
cmd = Cmd(self._platform, self._dryrun, logger=self._logger)
settingsFile = installationDirectory / "settings64.bat"
self._environment = cmd.GetEnvironment(settingsFile)
[docs] def GetVHDLCompiler(self):
raise NotImplementedError("ISE.GetVHDLCompiler")
# return ISEVHDLCompiler(self._platform, self._dryrun, self._binaryDirectoryPath, self._version, logger=self._logger)
[docs] def GetFuse(self):
return Fuse(self)
[docs] def GetXst(self):
return Xst(self)
[docs] def GetCoreGenerator(self):
return CoreGenerator(self)
[docs]class Fuse(OutputFilteredExecutable, ToolMixIn):
def __init__(self, toolchain : ToolMixIn):
ToolMixIn.__init__(
self, toolchain._platform, toolchain._dryrun, toolchain._binaryDirectoryPath, toolchain._version,
toolchain._logger)
if (self._platform == "Windows"): executablePath = self._binaryDirectoryPath / "fuse.exe"
elif (self._platform == "Linux"): executablePath = self._binaryDirectoryPath / "fuse"
else: raise PlatformNotSupportedException(self._platform)
super().__init__(self._platform, self._dryrun, executablePath, environment=toolchain._environment, logger=self._logger)
self.Parameters[self.Executable] = executablePath
[docs] class Executable(metaclass=ExecutableArgument): pass
[docs] class FlagIncremental(metaclass=ShortFlagArgument):
_name = "incremental"
# FlagIncremental = ShortFlagArgument(_name="incremntal")
[docs] class FlagRangeCheck(metaclass=ShortFlagArgument):
_name = "rangecheck"
[docs] class SwitchMultiThreading(metaclass=ShortTupleArgument):
_name = "mt"
[docs] class SwitchTimeResolution(metaclass=ShortTupleArgument):
_name = "timeprecision_vhdl"
[docs] class SwitchProjectFile(metaclass=ShortTupleArgument):
_name = "prj"
[docs] class SwitchOutputFile(metaclass=ShortTupleArgument):
_name = "o"
[docs] class ArgTopLevel(metaclass=StringArgument): pass
Parameters = CommandLineArgumentList(
Executable,
FlagIncremental,
FlagRangeCheck,
SwitchMultiThreading,
SwitchTimeResolution,
SwitchProjectFile,
SwitchOutputFile,
ArgTopLevel
)
[docs] def Link(self):
parameterList = self.Parameters.ToArgumentList()
self.LogVerbose("command: {0}".format(" ".join(parameterList)))
try:
self.StartProcess(parameterList)
except Exception as ex:
raise ISEException("Failed to launch fuse.") from ex
self._hasOutput = False
self._hasWarnings = False
self._hasErrors = False
try:
iterator = iter(FuseFilter(self.GetReader()))
line = next(iterator)
self._hasOutput = True
self.LogNormal(" fuse messages for '{0}'".format(self.Parameters[self.SwitchProjectFile]))
self.LogNormal(" " + ("-" * (78 - self.Logger.BaseIndent*2)))
while True:
self._hasWarnings |= (line.Severity is Severity.Warning)
self._hasErrors |= (line.Severity is Severity.Error)
line.IndentBy(self.Logger.BaseIndent + 1)
self.Log(line)
line = next(iterator)
except DryRunException:
pass
except StopIteration:
pass
finally:
if self._hasOutput:
self.LogNormal(" " + ("-" * (78 - self.Logger.BaseIndent*2)))
[docs]class ISESimulator(OutputFilteredExecutable):
def __init__(self, platform, dryrun, executablePath, environment, logger=None):
super().__init__(platform, dryrun, executablePath, environment=environment, logger=logger)
self.Parameters[self.Executable] = executablePath
[docs] class Executable(metaclass=ExecutableArgument): pass
[docs] class SwitchLogFile(metaclass=ShortTupleArgument):
_name = "log"
[docs] class FlagGuiMode(metaclass=ShortFlagArgument):
_name = "gui"
[docs] class SwitchTclBatchFile(metaclass=ShortTupleArgument):
_name = "tclbatch"
Parameters = CommandLineArgumentList(
Executable,
SwitchLogFile,
FlagGuiMode,
SwitchTclBatchFile,
SwitchWaveformFile
)
[docs] def Simulate(self):
parameterList = self.Parameters.ToArgumentList()
self.LogVerbose("command: {0}".format(" ".join(parameterList)))
try:
self.StartProcess(parameterList)
except Exception as ex:
raise ISEException("Failed to launch isim.") from ex
self._hasOutput = False
self._hasWarnings = False
self._hasErrors = False
simulationResult = CallByRefParam(SimulationResult.Error)
try:
iterator = iter(PoCSimulationResultFilter(SimulatorFilter(self.GetReader()), simulationResult))
line = next(iterator)
self._hasOutput = True
self.LogNormal(" isim messages for '{0}'".format(self.Parameters[self.Executable]))
self.LogNormal(" " + ("-" * (78 - self.Logger.BaseIndent*2)))
while True:
self._hasWarnings |= (line.Severity is Severity.Warning)
self._hasErrors |= (line.Severity is Severity.Error)
line.IndentBy(self.Logger.BaseIndent + 1)
self.Log(line)
line = next(iterator)
except DryRunException:
simulationResult <<= SimulationResult.DryRun
except StopIteration:
pass
finally:
if self._hasOutput:
self.LogNormal(" " + ("-" * (78 - self.Logger.BaseIndent*2)))
return simulationResult.value
[docs]class Xst(OutputFilteredExecutable, ToolMixIn):
def __init__(self, toolchain : ToolMixIn):
ToolMixIn.__init__(
self, toolchain._platform, toolchain._dryrun, toolchain._binaryDirectoryPath, toolchain._version,
toolchain._logger)
if (self._platform == "Windows"): executablePath = self._binaryDirectoryPath / "xst.exe"
elif (self._platform == "Linux"): executablePath = self._binaryDirectoryPath / "xst"
else: raise PlatformNotSupportedException(self._platform)
super().__init__(self._platform, self._dryrun, executablePath, environment=toolchain._environment, logger=self._logger)
self.Parameters[self.Executable] = executablePath
[docs] class Executable(metaclass=ExecutableArgument):
pass
[docs] class SwitchIntStyle(metaclass=ShortTupleArgument):
_name = "intstyle"
[docs] class SwitchXstFile(metaclass=ShortTupleArgument):
_name = "ifn"
[docs] class SwitchReportFile(metaclass=ShortTupleArgument):
_name = "ofn"
Parameters = CommandLineArgumentList(
Executable,
SwitchIntStyle,
SwitchXstFile,
SwitchReportFile
)
[docs] def Compile(self):
parameterList = self.Parameters.ToArgumentList()
self.LogVerbose("command: {0}".format(" ".join(parameterList)))
try:
self.StartProcess(parameterList)
except Exception as ex:
raise ISEException("Failed to launch xst.") from ex
self._hasOutput = False
self._hasWarnings = False
self._hasErrors = False
try:
iterator = iter(XstFilter(self.GetReader()))
line = next(iterator)
self._hasOutput = True
self.LogNormal(" xst messages for '{0}'".format(self.Parameters[self.SwitchXstFile]))
self.LogNormal(" " + ("-" * (78 - self.Logger.BaseIndent*2)))
while True:
self._hasWarnings |= (line.Severity is Severity.Warning)
self._hasErrors |= (line.Severity is Severity.Error)
line.IndentBy(self.Logger.BaseIndent + 1)
self.Log(line)
line = next(iterator)
except DryRunException:
pass
except StopIteration:
pass
finally:
if self._hasOutput:
self.LogNormal(" " + ("-" * (78 - self.Logger.BaseIndent*2)))
[docs]class CoreGenerator(OutputFilteredExecutable, ToolMixIn):
def __init__(self, toolchain : ToolMixIn):
ToolMixIn.__init__(
self, toolchain._platform, toolchain._dryrun, toolchain._binaryDirectoryPath, toolchain._version,
toolchain._logger)
if (self._platform == "Windows"): executablePath = self._binaryDirectoryPath / "coregen.exe"
elif (self._platform == "Linux"): executablePath = self._binaryDirectoryPath / "coregen"
else: raise PlatformNotSupportedException(self._platform)
super().__init__(self._platform, self._dryrun, executablePath, environment=toolchain._environment, logger=self._logger)
self.Parameters[self.Executable] = executablePath
[docs] class Executable(metaclass=ExecutableArgument): pass
[docs] class FlagRegenerate(metaclass=ShortFlagArgument):
_name = "r"
[docs] class SwitchProjectFile(metaclass=ShortTupleArgument):
_name = "p"
[docs] class SwitchBatchFile(metaclass=ShortTupleArgument):
_name = "b"
Parameters = CommandLineArgumentList(
Executable,
FlagRegenerate,
SwitchProjectFile,
SwitchBatchFile
)
[docs] def Generate(self):
parameterList = self.Parameters.ToArgumentList()
self.LogVerbose("command: {0}".format(" ".join(parameterList)))
try:
self.StartProcess(parameterList)
except Exception as ex:
raise ISEException("Failed to launch corgen.") from ex
self._hasOutput = False
self._hasWarnings = False
self._hasErrors = False
try:
iterator = iter(CoreGeneratorFilter(self.GetReader()))
line = next(iterator)
self._hasOutput = True
self.LogNormal(" coregen messages for '{0}'".format(self.Parameters[self.SwitchProjectFile]))
self.LogNormal(" " + ("-" * (78 - self.Logger.BaseIndent*2)))
while True:
self._hasWarnings |= (line.Severity is Severity.Warning)
self._hasErrors |= (line.Severity is Severity.Error)
line.IndentBy(self.Logger.BaseIndent + 1)
self.Log(line)
line = next(iterator)
except DryRunException:
pass
except StopIteration:
pass
finally:
if self._hasOutput:
self.LogNormal(" " + ("-" * (78 - self.Logger.BaseIndent*2)))
[docs]def VhCompFilter(gen):
for line in gen:
yield LogEntry(line, Severity.Normal)
[docs]def FuseFilter(gen):
for line in gen:
if line.startswith("ISim "):
yield LogEntry(line, Severity.Debug)
elif line.startswith("Fuse "):
yield LogEntry(line, Severity.Debug)
elif line.startswith("Determining compilation order of HDL files"):
yield LogEntry(line, Severity.Verbose)
elif line.startswith("Parsing VHDL file "):
yield LogEntry(line, Severity.Verbose)
elif line.startswith("WARNING:HDLCompiler:"):
yield LogEntry(line, Severity.Warning)
elif line.startswith("Starting static elaboration"):
yield LogEntry(line, Severity.Verbose)
elif line.startswith("Completed static elaboration"):
yield LogEntry(line, Severity.Verbose)
elif line.startswith("Compiling package "):
yield LogEntry(line, Severity.Verbose)
elif line.startswith("Compiling architecture "):
yield LogEntry(line, Severity.Verbose)
elif line.startswith("Time Resolution for simulation is"):
yield LogEntry(line, Severity.Verbose)
elif (line.startswith("Waiting for ") and line.endswith(" to finish...")):
yield LogEntry(line, Severity.Verbose)
elif (line.startswith("Compiled ") and line.endswith(" VHDL Units")):
yield LogEntry(line, Severity.Verbose)
else:
yield LogEntry(line, Severity.Normal)
[docs]def SimulatorFilter(gen):
for line in gen:
if line.startswith("ISim "):
yield LogEntry(line, Severity.Debug)
elif line.startswith("This is a Full version of ISim."):
yield LogEntry(line, Severity.Verbose)
elif line.startswith("Time resolution is "):
yield LogEntry(line, Severity.Verbose)
elif line.startswith("Simulator is doing circuit initialization process."):
yield LogEntry(line, Severity.Debug)
elif line.startswith("Finished circuit initialization process."):
yield LogEntry(line, Severity.Verbose)
elif line.startswith("ERROR:"):
yield LogEntry(line, Severity.Error)
elif line.startswith("WARNING:"):
yield LogEntry(line, Severity.Warning)
elif line.startswith("INFO:"):
yield LogEntry(line, Severity.Verbose)
else:
yield LogEntry(line, Severity.Normal)
[docs]def XstFilter(gen):
flagNormal = False
for line in gen:
if line.startswith("ERROR:"):
yield LogEntry(line, Severity.Error)
elif line.startswith("WARNING:"):
yield LogEntry(line, Severity.Warning)
elif line.startswith("Note:"):
yield LogEntry(line, Severity.Info)
elif line.startswith("* "): # progress
yield LogEntry(line, Severity.Normal)
flagNormal = True
else:
yield LogEntry(line, Severity.Normal if flagNormal else Severity.Verbose)
flagNormal = False
[docs]def CoreGeneratorFilter(gen):
for line in gen:
if line.startswith("ERROR:"):
yield LogEntry(line, Severity.Error)
elif line.startswith("WARNING:"):
yield LogEntry(line, Severity.Warning)
elif line.startswith("Note:"):
yield LogEntry(line, Severity.Info)
else:
yield LogEntry(line, Severity.Normal)
[docs]class ISEProject(BaseProject):
def __init__(self, name):
super().__init__(name)
[docs]class ISEProjectFile(ProjectFile):
def __init__(self, file):
super().__init__(file)
[docs]class UserConstraintFile(ConstraintFile):
_FileType = FileTypes.UcfConstraintFile
def __str__(self):
return "UCF file: '{0!s}".format(self._file)