SymbolTable.py revision 6999:f226c098c393
1# Copyright (c) 1999-2008 Mark D. Hill and David A. Wood
2# Copyright (c) 2009 The Hewlett-Packard Development Company
3# All rights reserved.
4#
5# Redistribution and use in source and binary forms, with or without
6# modification, are permitted provided that the following conditions are
7# met: redistributions of source code must retain the above copyright
8# notice, this list of conditions and the following disclaimer;
9# redistributions in binary form must reproduce the above copyright
10# notice, this list of conditions and the following disclaimer in the
11# documentation and/or other materials provided with the distribution;
12# neither the name of the copyright holders nor the names of its
13# contributors may be used to endorse or promote products derived from
14# this software without specific prior written permission.
15#
16# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
17# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
18# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
19# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
20# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
21# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27
28from m5.util import code_formatter
29
30from slicc.generate import html
31from slicc.symbols.StateMachine import StateMachine
32from slicc.symbols.Type import Type
33from slicc.util import Location
34
35class SymbolTable(object):
36    def __init__(self, slicc):
37        self.slicc = slicc
38
39        self.sym_vec = []
40        self.sym_map_vec = [ {} ]
41        self.machine_components = {}
42
43        pairs = {}
44        pairs["enumeration"] = "yes"
45        MachineType = Type(self, "MachineType", Location("init", 0), pairs)
46        self.newSymbol(MachineType)
47
48        pairs = {}
49        pairs["primitive"] = "yes"
50        pairs["external"] = "yes"
51        void = Type(self, "void", Location("init", 0), pairs)
52        self.newSymbol(void)
53
54    def __repr__(self):
55        return "[SymbolTable]" # FIXME
56
57    def codeFormatter(self, *args, **kwargs):
58        return self.slicc.codeFormatter(*args, **kwargs)
59
60    def newSymbol(self, sym):
61        self.registerSym(str(sym), sym)
62        self.sym_vec.append(sym)
63
64    def registerSym(self, id, sym):
65        # Check for redeclaration (in the current frame only)
66        if id in self.sym_map_vec[-1]:
67            sym.error("Symbol '%s' redeclared in same scope.", id)
68
69        # FIXME - warn on masking of a declaration in a previous frame
70        self.sym_map_vec[-1][id] = sym
71
72    def find(self, ident, types=None):
73        for sym_map in reversed(self.sym_map_vec):
74            try:
75                symbol = sym_map[ident]
76            except KeyError:
77                continue
78
79            if types is not None:
80                if not isinstance(symbol, types):
81                    symbol.error("Symbol '%s' is not of types '%s'.",
82                                 symbol,
83                                 types)
84
85            return symbol
86
87        return None
88
89    def newMachComponentSym(self, symbol):
90        # used to cheat-- that is, access components in other machines
91        machine = self.find("current_machine", StateMachine)
92        if machine:
93            self.machine_components[str(machine)][str(symbol)] = symbol
94
95    def newCurrentMachine(self, sym):
96        self.registerGlobalSym(str(sym), sym)
97        self.registerSym("current_machine", sym)
98        self.sym_vec.append(sym)
99
100        self.machine_components[str(sym)] = {}
101
102    @property
103    def state_machine(self):
104        return self.find("current_machine", StateMachine)
105
106    def pushFrame(self):
107        self.sym_map_vec.append({})
108
109    def popFrame(self):
110        assert len(self.sym_map_vec) > 0
111        self.sym_map_vec.pop()
112
113    def registerGlobalSym(self, ident, symbol):
114        # Check for redeclaration (global frame only)
115        if ident in self.sym_map_vec[0]:
116            symbol.error("Symbol '%s' redeclared in global scope." % ident)
117
118        self.sym_map_vec[0][ident] = symbol
119
120    def getAllType(self, type):
121        for symbol in self.sym_vec:
122            if isinstance(symbol, type):
123                yield symbol
124
125    def writeCodeFiles(self, path):
126        code = self.codeFormatter()
127        code('''
128/** Auto generated C++ code started by $__file__:$__line__ */
129
130#include "mem/ruby/slicc_interface/RubySlicc_includes.hh"
131''')
132        for symbol in self.sym_vec:
133            if isinstance(symbol, Type) and not symbol.isPrimitive:
134                code('#include "mem/protocol/${{symbol.c_ident}}.hh"')
135
136        code.write(path, "Types.hh")
137
138        for symbol in self.sym_vec:
139            symbol.writeCodeFiles(path)
140
141    def writeHTMLFiles(self, path):
142        machines = list(self.getAllType(StateMachine))
143        if len(machines) > 1:
144            name = "%s_table.html" % machines[0].ident
145        else:
146            name = "empty.html"
147
148        code = self.codeFormatter()
149        code('''
150<html>
151<head>
152<title>$path</title>
153</head>
154<frameset rows="*,30">
155    <frame name="Table" src="$name">
156    <frame name="Status" src="empty.html">
157</frameset>
158</html>
159''')
160        code.write(path, "index.html")
161
162        code = self.codeFormatter()
163        code("<HTML></HTML>")
164        code.write(path, "empty.html")
165
166        for symbol in self.sym_vec:
167            symbol.writeHTMLFiles(path)
168
169__all__ = [ "SymbolTable" ]
170