Ruby: Modify Scons so that we can put .sm files in extras
Also allows for header files which are required in slicc generated code to be in a directory other than src/mem/ruby/slicc_interface.
This commit is contained in:
parent
c6927ed138
commit
aa8bcd15ec
9 changed files with 72 additions and 37 deletions
13
SConstruct
13
SConstruct
|
@ -835,6 +835,14 @@ Export('sticky_vars')
|
||||||
export_vars = []
|
export_vars = []
|
||||||
Export('export_vars')
|
Export('export_vars')
|
||||||
|
|
||||||
|
# For Ruby
|
||||||
|
all_protocols = []
|
||||||
|
Export('all_protocols')
|
||||||
|
protocol_dirs = []
|
||||||
|
Export('protocol_dirs')
|
||||||
|
slicc_includes = []
|
||||||
|
Export('slicc_includes')
|
||||||
|
|
||||||
# Walk the tree and execute all SConsopts scripts that wil add to the
|
# Walk the tree and execute all SConsopts scripts that wil add to the
|
||||||
# above variables
|
# above variables
|
||||||
if not GetOption('verbose'):
|
if not GetOption('verbose'):
|
||||||
|
@ -867,11 +875,14 @@ sticky_vars.AddVariables(
|
||||||
BoolVariable('USE_POSIX_CLOCK', 'Use POSIX Clocks', have_posix_clock),
|
BoolVariable('USE_POSIX_CLOCK', 'Use POSIX Clocks', have_posix_clock),
|
||||||
BoolVariable('USE_FENV', 'Use <fenv.h> IEEE mode control', have_fenv),
|
BoolVariable('USE_FENV', 'Use <fenv.h> IEEE mode control', have_fenv),
|
||||||
BoolVariable('CP_ANNOTATE', 'Enable critical path annotation capability', False),
|
BoolVariable('CP_ANNOTATE', 'Enable critical path annotation capability', False),
|
||||||
|
EnumVariable('PROTOCOL', 'Coherence protocol for Ruby', 'None',
|
||||||
|
all_protocols),
|
||||||
)
|
)
|
||||||
|
|
||||||
# These variables get exported to #defines in config/*.hh (see src/SConscript).
|
# These variables get exported to #defines in config/*.hh (see src/SConscript).
|
||||||
export_vars += ['USE_FENV', 'SS_COMPATIBLE_FP',
|
export_vars += ['USE_FENV', 'SS_COMPATIBLE_FP',
|
||||||
'TARGET_ISA', 'CP_ANNOTATE', 'USE_POSIX_CLOCK' ]
|
'TARGET_ISA', 'CP_ANNOTATE', 'USE_POSIX_CLOCK', 'PROTOCOL',
|
||||||
|
]
|
||||||
|
|
||||||
###################################################
|
###################################################
|
||||||
#
|
#
|
||||||
|
|
|
@ -41,7 +41,7 @@ Import('*')
|
||||||
if env['PROTOCOL'] == 'None':
|
if env['PROTOCOL'] == 'None':
|
||||||
Return()
|
Return()
|
||||||
|
|
||||||
protocol_dir = Dir('.')
|
output_dir = Dir('.')
|
||||||
html_dir = Dir('html')
|
html_dir = Dir('html')
|
||||||
slicc_dir = Dir('../slicc')
|
slicc_dir = Dir('../slicc')
|
||||||
|
|
||||||
|
@ -57,7 +57,7 @@ for root,dirs,files in os.walk(slicc_dir.srcnode().abspath):
|
||||||
#
|
#
|
||||||
# Use SLICC
|
# Use SLICC
|
||||||
#
|
#
|
||||||
env['SLICC_PATH'] = str(protocol_dir)
|
env["SLICC_PATH"] = protocol_dirs
|
||||||
slicc_scanner = Classic("SliccScanner", ['.sm', '.slicc'], "SLICC_PATH",
|
slicc_scanner = Classic("SliccScanner", ['.sm', '.slicc'], "SLICC_PATH",
|
||||||
r'''include[ \t]["'](.*)["'];''')
|
r'''include[ \t]["'](.*)["'];''')
|
||||||
env.Append(SCANNERS=slicc_scanner)
|
env.Append(SCANNERS=slicc_scanner)
|
||||||
|
@ -66,22 +66,22 @@ def slicc_emitter(target, source, env):
|
||||||
assert len(source) == 1
|
assert len(source) == 1
|
||||||
filepath = source[0].srcnode().abspath
|
filepath = source[0].srcnode().abspath
|
||||||
|
|
||||||
slicc = SLICC(filepath, verbose=False)
|
slicc = SLICC(filepath, protocol_base.abspath, verbose=False)
|
||||||
slicc.process()
|
slicc.process()
|
||||||
slicc.writeCodeFiles(protocol_dir.abspath)
|
slicc.writeCodeFiles(output_dir.abspath, slicc_includes)
|
||||||
if env['SLICC_HTML']:
|
if env['SLICC_HTML']:
|
||||||
slicc.writeHTMLFiles(html_dir.abspath)
|
slicc.writeHTMLFiles(html_dir.abspath)
|
||||||
|
|
||||||
target.extend([protocol_dir.File(f) for f in sorted(slicc.files())])
|
target.extend([output_dir.File(f) for f in sorted(slicc.files())])
|
||||||
return target, source
|
return target, source
|
||||||
|
|
||||||
def slicc_action(target, source, env):
|
def slicc_action(target, source, env):
|
||||||
assert len(source) == 1
|
assert len(source) == 1
|
||||||
filepath = source[0].srcnode().abspath
|
filepath = source[0].srcnode().abspath
|
||||||
|
|
||||||
slicc = SLICC(filepath, verbose=True)
|
slicc = SLICC(filepath, protocol_base.abspath, verbose=True)
|
||||||
slicc.process()
|
slicc.process()
|
||||||
slicc.writeCodeFiles(protocol_dir.abspath)
|
slicc.writeCodeFiles(output_dir.abspath, slicc_includes)
|
||||||
if env['SLICC_HTML']:
|
if env['SLICC_HTML']:
|
||||||
slicc.writeHTMLFiles(html_dir.abspath)
|
slicc.writeHTMLFiles(html_dir.abspath)
|
||||||
|
|
||||||
|
@ -89,6 +89,15 @@ slicc_builder = Builder(action=MakeAction(slicc_action, Transform("SLICC")),
|
||||||
emitter=slicc_emitter)
|
emitter=slicc_emitter)
|
||||||
|
|
||||||
protocol = env['PROTOCOL']
|
protocol = env['PROTOCOL']
|
||||||
|
protocol_dir = None
|
||||||
|
for path in protocol_dirs:
|
||||||
|
if os.path.exists(os.path.join(path, "%s.slicc" % protocol)):
|
||||||
|
protocol_dir = Dir(path)
|
||||||
|
break
|
||||||
|
|
||||||
|
if not protocol_dir:
|
||||||
|
raise ValueError, "Could not find %s.slicc in protocol_dirs" % protocol
|
||||||
|
|
||||||
sources = [ protocol_dir.File("%s.slicc" % protocol) ]
|
sources = [ protocol_dir.File("%s.slicc" % protocol) ]
|
||||||
|
|
||||||
env.Append(BUILDERS={'SLICC' : slicc_builder})
|
env.Append(BUILDERS={'SLICC' : slicc_builder})
|
||||||
|
|
|
@ -32,7 +32,7 @@ import os
|
||||||
|
|
||||||
Import('*')
|
Import('*')
|
||||||
|
|
||||||
all_protocols = [
|
all_protocols.extend([
|
||||||
'MESI_CMP_directory',
|
'MESI_CMP_directory',
|
||||||
'MI_example',
|
'MI_example',
|
||||||
'MOESI_CMP_directory',
|
'MOESI_CMP_directory',
|
||||||
|
@ -40,13 +40,14 @@ all_protocols = [
|
||||||
'MOESI_hammer',
|
'MOESI_hammer',
|
||||||
'Network_test',
|
'Network_test',
|
||||||
'None'
|
'None'
|
||||||
]
|
])
|
||||||
|
|
||||||
opt = EnumVariable('PROTOCOL', 'Coherence protocol for Ruby', 'None',
|
|
||||||
all_protocols)
|
|
||||||
|
|
||||||
sticky_vars.AddVariables(opt)
|
|
||||||
export_vars += ['PROTOCOL']
|
|
||||||
|
|
||||||
opt = BoolVariable('SLICC_HTML', 'Create HTML files', False)
|
opt = BoolVariable('SLICC_HTML', 'Create HTML files', False)
|
||||||
sticky_vars.AddVariables(opt)
|
sticky_vars.AddVariables(opt)
|
||||||
|
|
||||||
|
protocol_dirs.append(Dir('.').abspath)
|
||||||
|
|
||||||
|
protocol_base = Dir('.')
|
||||||
|
Export('protocol_base')
|
||||||
|
|
||||||
|
slicc_includes.append('mem/ruby/slicc_interface/RubySlicc_includes.hh')
|
||||||
|
|
|
@ -38,11 +38,12 @@ import slicc.util as util
|
||||||
from slicc.symbols import SymbolTable
|
from slicc.symbols import SymbolTable
|
||||||
|
|
||||||
class SLICC(Grammar):
|
class SLICC(Grammar):
|
||||||
def __init__(self, filename, verbose=False, traceback=False, **kwargs):
|
def __init__(self, filename, base_dir, verbose=False, traceback=False, **kwargs):
|
||||||
self.protocol = None
|
self.protocol = None
|
||||||
self.traceback = traceback
|
self.traceback = traceback
|
||||||
self.verbose = verbose
|
self.verbose = verbose
|
||||||
self.symtab = SymbolTable(self)
|
self.symtab = SymbolTable(self)
|
||||||
|
self.base_dir = base_dir
|
||||||
|
|
||||||
try:
|
try:
|
||||||
self.decl_list = self.parse_file(filename, **kwargs)
|
self.decl_list = self.parse_file(filename, **kwargs)
|
||||||
|
@ -64,8 +65,8 @@ class SLICC(Grammar):
|
||||||
self.decl_list.findMachines()
|
self.decl_list.findMachines()
|
||||||
self.decl_list.generate()
|
self.decl_list.generate()
|
||||||
|
|
||||||
def writeCodeFiles(self, code_path):
|
def writeCodeFiles(self, code_path, includes):
|
||||||
self.symtab.writeCodeFiles(code_path)
|
self.symtab.writeCodeFiles(code_path, includes)
|
||||||
|
|
||||||
def writeHTMLFiles(self, html_path):
|
def writeHTMLFiles(self, html_path):
|
||||||
self.symtab.writeHTMLFiles(html_path)
|
self.symtab.writeHTMLFiles(html_path)
|
||||||
|
@ -249,7 +250,10 @@ class SLICC(Grammar):
|
||||||
def p_decl__include(self, p):
|
def p_decl__include(self, p):
|
||||||
"decl : INCLUDE STRING SEMI"
|
"decl : INCLUDE STRING SEMI"
|
||||||
dirname = os.path.dirname(self.current_source)
|
dirname = os.path.dirname(self.current_source)
|
||||||
filename = os.path.join(dirname, p[2])
|
if os.path.exists(os.path.join(dirname, p[2])):
|
||||||
|
filename = os.path.join(dirname, p[2])
|
||||||
|
else:
|
||||||
|
filename = os.path.join(self.base_dir, p[2])
|
||||||
p[0] = self.parse_file(filename)
|
p[0] = self.parse_file(filename)
|
||||||
|
|
||||||
def p_decl__machine(self, p):
|
def p_decl__machine(self, p):
|
||||||
|
|
|
@ -63,7 +63,7 @@ class Func(Symbol):
|
||||||
return "%s %s(%s);" % (return_type, self.c_ident,
|
return "%s %s(%s);" % (return_type, self.c_ident,
|
||||||
", ".join(self.param_strings))
|
", ".join(self.param_strings))
|
||||||
|
|
||||||
def writeCodeFiles(self, path):
|
def writeCodeFiles(self, path, includes):
|
||||||
return
|
return
|
||||||
|
|
||||||
def generateCode(self):
|
def generateCode(self):
|
||||||
|
|
|
@ -162,12 +162,12 @@ class StateMachine(Symbol):
|
||||||
action.warning(error_msg)
|
action.warning(error_msg)
|
||||||
self.table = table
|
self.table = table
|
||||||
|
|
||||||
def writeCodeFiles(self, path):
|
def writeCodeFiles(self, path, includes):
|
||||||
self.printControllerPython(path)
|
self.printControllerPython(path)
|
||||||
self.printControllerHH(path)
|
self.printControllerHH(path)
|
||||||
self.printControllerCC(path)
|
self.printControllerCC(path, includes)
|
||||||
self.printCSwitch(path)
|
self.printCSwitch(path)
|
||||||
self.printCWakeup(path)
|
self.printCWakeup(path, includes)
|
||||||
self.printProfilerCC(path)
|
self.printProfilerCC(path)
|
||||||
self.printProfilerHH(path)
|
self.printProfilerHH(path)
|
||||||
self.printProfileDumperCC(path)
|
self.printProfileDumperCC(path)
|
||||||
|
@ -399,7 +399,7 @@ void unset_tbe(${{self.TBEType.c_ident}}*& m_tbe_ptr);
|
||||||
code('#endif // __${ident}_CONTROLLER_H__')
|
code('#endif // __${ident}_CONTROLLER_H__')
|
||||||
code.write(path, '%s.hh' % c_ident)
|
code.write(path, '%s.hh' % c_ident)
|
||||||
|
|
||||||
def printControllerCC(self, path):
|
def printControllerCC(self, path, includes):
|
||||||
'''Output the actions for performing the actions'''
|
'''Output the actions for performing the actions'''
|
||||||
|
|
||||||
code = self.symtab.codeFormatter()
|
code = self.symtab.codeFormatter()
|
||||||
|
@ -429,8 +429,12 @@ void unset_tbe(${{self.TBEType.c_ident}}*& m_tbe_ptr);
|
||||||
#include "mem/protocol/${ident}_State.hh"
|
#include "mem/protocol/${ident}_State.hh"
|
||||||
#include "mem/protocol/Types.hh"
|
#include "mem/protocol/Types.hh"
|
||||||
#include "mem/ruby/common/Global.hh"
|
#include "mem/ruby/common/Global.hh"
|
||||||
#include "mem/ruby/slicc_interface/RubySlicc_includes.hh"
|
|
||||||
#include "mem/ruby/system/System.hh"
|
#include "mem/ruby/system/System.hh"
|
||||||
|
''')
|
||||||
|
for include_path in includes:
|
||||||
|
code('#include "${{include_path}}"')
|
||||||
|
|
||||||
|
code('''
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
''')
|
''')
|
||||||
|
@ -988,7 +992,7 @@ $c_ident::${{action.ident}}(const Address& addr)
|
||||||
|
|
||||||
code.write(path, "%s.cc" % c_ident)
|
code.write(path, "%s.cc" % c_ident)
|
||||||
|
|
||||||
def printCWakeup(self, path):
|
def printCWakeup(self, path, includes):
|
||||||
'''Output the wakeup loop for the events'''
|
'''Output the wakeup loop for the events'''
|
||||||
|
|
||||||
code = self.symtab.codeFormatter()
|
code = self.symtab.codeFormatter()
|
||||||
|
@ -1020,8 +1024,14 @@ $c_ident::${{action.ident}}(const Address& addr)
|
||||||
code('''
|
code('''
|
||||||
#include "mem/protocol/Types.hh"
|
#include "mem/protocol/Types.hh"
|
||||||
#include "mem/ruby/common/Global.hh"
|
#include "mem/ruby/common/Global.hh"
|
||||||
#include "mem/ruby/slicc_interface/RubySlicc_includes.hh"
|
|
||||||
#include "mem/ruby/system/System.hh"
|
#include "mem/ruby/system/System.hh"
|
||||||
|
''')
|
||||||
|
|
||||||
|
|
||||||
|
for include_path in includes:
|
||||||
|
code('#include "${{include_path}}"')
|
||||||
|
|
||||||
|
code('''
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
|
|
||||||
|
|
|
@ -124,15 +124,15 @@ class SymbolTable(object):
|
||||||
if isinstance(symbol, type):
|
if isinstance(symbol, type):
|
||||||
yield symbol
|
yield symbol
|
||||||
|
|
||||||
def writeCodeFiles(self, path):
|
def writeCodeFiles(self, path, includes):
|
||||||
makeDir(path)
|
makeDir(path)
|
||||||
|
|
||||||
code = self.codeFormatter()
|
code = self.codeFormatter()
|
||||||
code('''
|
code('/** Auto generated C++ code started by $__file__:$__line__ */')
|
||||||
/** Auto generated C++ code started by $__file__:$__line__ */
|
|
||||||
|
for include_path in includes:
|
||||||
|
code('#include "${{include_path}}"')
|
||||||
|
|
||||||
#include "mem/ruby/slicc_interface/RubySlicc_includes.hh"
|
|
||||||
''')
|
|
||||||
for symbol in self.sym_vec:
|
for symbol in self.sym_vec:
|
||||||
if isinstance(symbol, Type) and not symbol.isPrimitive:
|
if isinstance(symbol, Type) and not symbol.isPrimitive:
|
||||||
code('#include "mem/protocol/${{symbol.c_ident}}.hh"')
|
code('#include "mem/protocol/${{symbol.c_ident}}.hh"')
|
||||||
|
@ -140,7 +140,7 @@ class SymbolTable(object):
|
||||||
code.write(path, "Types.hh")
|
code.write(path, "Types.hh")
|
||||||
|
|
||||||
for symbol in self.sym_vec:
|
for symbol in self.sym_vec:
|
||||||
symbol.writeCodeFiles(path)
|
symbol.writeCodeFiles(path, includes)
|
||||||
|
|
||||||
def writeHTMLFiles(self, path):
|
def writeHTMLFiles(self, path):
|
||||||
makeDir(path)
|
makeDir(path)
|
||||||
|
|
|
@ -184,7 +184,7 @@ class Type(Symbol):
|
||||||
|
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def writeCodeFiles(self, path):
|
def writeCodeFiles(self, path, includes):
|
||||||
if self.isExternal:
|
if self.isExternal:
|
||||||
# Do nothing
|
# Do nothing
|
||||||
pass
|
pass
|
||||||
|
|
|
@ -44,7 +44,7 @@ class Var(Symbol):
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
return "[Var id: %s]" % (self.c_ident)
|
return "[Var id: %s]" % (self.c_ident)
|
||||||
|
|
||||||
def writeCodeFiles(self, path):
|
def writeCodeFiles(self, path, includes):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
__all__ = [ "Var" ]
|
__all__ = [ "Var" ]
|
||||||
|
|
Loading…
Reference in a new issue