Files
gem5/src/mem/ruby/protocol/SConscript

182 lines
5.8 KiB
Python

# -*- mode:python -*-
# Copyright (c) 2009 The Hewlett-Packard Development Company
# All rights reserved.
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are
# met: redistributions of source code must retain the above copyright
# notice, this list of conditions and the following disclaimer;
# redistributions in binary form must reproduce the above copyright
# notice, this list of conditions and the following disclaimer in the
# documentation and/or other materials provided with the distribution;
# neither the name of the copyright holders nor the names of its
# contributors may be used to endorse or promote products derived from
# this software without specific prior written permission.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
import os.path
import re
import sys
from SCons.Scanner import Classic
from gem5_scons import Transform
Import("*")
if not env["CONF"]["RUBY"]:
Return()
output_dir = Dir(".")
html_dir = Dir("html")
slicc_dir = Dir("../slicc")
sys.path[1:1] = [Dir("..").Dir("..").srcnode().abspath]
from slicc.parser import SLICC
slicc_depends = []
for root, dirs, files in os.walk(slicc_dir.srcnode().abspath):
for f in files:
if f.endswith(".py"):
slicc_depends.append(File(os.path.join(root, f)))
#
# Use SLICC
#
env["SLICC_PATH"] = env["PROTOCOL_DIRS"]
slicc_scanner = Classic(
"SliccScanner",
[".sm", ".slicc"],
"SLICC_PATH",
r"""include[ \t]["'](.*)["'];""",
)
env.Append(SCANNERS=slicc_scanner)
slicc_includes = ["mem/ruby/slicc_interface/RubySlicc_includes.hh"] + env[
"SLICC_INCLUDES"
]
def slicc_emitter(target, source, env):
files = set(target)
for s in source:
filepath = s.srcnode().abspath
slicc = SLICC(
filepath,
[
os.path.join(
protocol_base.abspath, "RubySlicc_interfaces.slicc"
)
],
protocol_base.abspath,
verbose=False,
)
slicc.process()
slicc.writeCodeFiles(output_dir.abspath, slicc_includes)
if env["CONF"]["SLICC_HTML"]:
slicc.writeHTMLFiles(html_dir.abspath)
files.update([output_dir.File(f) for f in sorted(slicc.files())])
# Dynamically determine protocol and add ProtocolInfo.hh to the list of
# files to be built
protocol_file = output_dir.File(
f"{slicc.protocol}/{slicc.protocol}ProtocolInfo.hh"
)
files.update([protocol_file])
return list(files), source
def slicc_action(target, source, env):
for s in source:
filepath = s.srcnode().abspath
slicc = SLICC(
filepath,
[
os.path.join(
protocol_base.abspath, "RubySlicc_interfaces.slicc"
)
],
protocol_base.abspath,
verbose=True,
)
slicc.process()
slicc.writeCodeFiles(output_dir.abspath, slicc_includes)
if env["CONF"]["SLICC_HTML"]:
slicc.writeHTMLFiles(html_dir.abspath)
slicc_builder = Builder(
action=MakeAction(slicc_action, Transform("SLICC")), emitter=slicc_emitter
)
if env["CONF"]["PROTOCOL"] != "MULTIPLE":
# Using backward compatibility
if (
f"RUBY_PROTOCOL_{env['CONF']['PROTOCOL']}" not in env["CONF"].keys()
or not env["CONF"][f"RUBY_PROTOCOL_{env['CONF']['PROTOCOL']}"]
):
raise ValueError(
"Your build config must be updated for the new multiple Ruby "
"build system. Please select the correct protocol under Ruby. "
"Deselect \"Use multiple protocols\" and select the desired "
f"protocol ({env['CONF']['PROTOCOL']})\n"
"Run the following code:\n"
f"scons menuconfig {env['BUILDDIR']}\n"
)
# Gather protocol names
protocols = []
for variable in env["CONF"]:
if variable.startswith("RUBY_PROTOCOL_") and env["CONF"][variable]:
protocols.append(variable[len("RUBY_PROTOCOL_") :])
def find_protocol_sources(protocol):
protocol_dir = None
for path in env["PROTOCOL_DIRS"]:
if os.path.exists(path.File("%s.slicc" % protocol).abspath):
protocol_dir = Dir(path)
break
if not protocol_dir:
raise ValueError(
"Could not find {}.slicc in PROTOCOL_DIRS".format(protocol)
)
return protocol_dir.File("%s.slicc" % protocol)
sources = [find_protocol_sources(protocol) for protocol in protocols]
env.Append(BUILDERS={"SLICC": slicc_builder})
nodes = env.SLICC([], sources)
env.Depends(nodes, slicc_depends)
append = {}
if env["CLANG"]:
append["CCFLAGS"] = "-Wno-parentheses"
for f in nodes:
s = str(f)
if s.endswith(".cc"):
Source(f, append=append)
elif s.endswith(".py"):
filename = os.path.basename(s)
# We currently only expect ${ident}_Controller.py to be generated, and
# for it to contain a single SimObject with the same name.
assert filename.endswith("_Controller.py")
SimObject(f, sim_objects=[os.path.splitext(filename)[0]])