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:
Jason Power 2012-09-12 14:52:04 -05:00
parent c6927ed138
commit aa8bcd15ec
9 changed files with 72 additions and 37 deletions

View file

@ -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',
]
################################################### ###################################################
# #

View file

@ -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})

View file

@ -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')

View file

@ -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):

View file

@ -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):

View file

@ -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;

View file

@ -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)

View file

@ -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

View file

@ -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" ]