19f22d7c2SAndrew Rist# *************************************************************
29f22d7c2SAndrew Rist#
39f22d7c2SAndrew Rist#  Licensed to the Apache Software Foundation (ASF) under one
49f22d7c2SAndrew Rist#  or more contributor license agreements.  See the NOTICE file
59f22d7c2SAndrew Rist#  distributed with this work for additional information
69f22d7c2SAndrew Rist#  regarding copyright ownership.  The ASF licenses this file
79f22d7c2SAndrew Rist#  to you under the Apache License, Version 2.0 (the
89f22d7c2SAndrew Rist#  "License"); you may not use this file except in compliance
99f22d7c2SAndrew Rist#  with the License.  You may obtain a copy of the License at
109f22d7c2SAndrew Rist#
119f22d7c2SAndrew Rist#    http://www.apache.org/licenses/LICENSE-2.0
129f22d7c2SAndrew Rist#
139f22d7c2SAndrew Rist#  Unless required by applicable law or agreed to in writing,
149f22d7c2SAndrew Rist#  software distributed under the License is distributed on an
159f22d7c2SAndrew Rist#  "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
169f22d7c2SAndrew Rist#  KIND, either express or implied.  See the License for the
179f22d7c2SAndrew Rist#  specific language governing permissions and limitations
189f22d7c2SAndrew Rist#  under the License.
199f22d7c2SAndrew Rist#
209f22d7c2SAndrew Rist# *************************************************************
219f22d7c2SAndrew Rist
22cdf0e10cSrcweir
23cdf0e10cSrcweirimport sys
24cdf0e10cSrcweirfrom globals import *
25cdf0e10cSrcweirimport srclexer
26cdf0e10cSrcweir
27cdf0e10cSrcweirclass MacroParser(object):
28cdf0e10cSrcweir
29cdf0e10cSrcweir    def __init__ (self, buf):
30cdf0e10cSrcweir        self.buffer = buf
31cdf0e10cSrcweir        self.macro = None
32cdf0e10cSrcweir        self.debug = False
33cdf0e10cSrcweir
34cdf0e10cSrcweir    def parse (self):
35cdf0e10cSrcweir        """
36ae54856bSPedro GiffuniA macro with arguments must have its open paren immediately following
37cdf0e10cSrcweirits name without any whitespace.
38cdf0e10cSrcweir"""
39cdf0e10cSrcweir        if self.debug:
40*7d9fa7c3SPedro Giffuni            print("-"*68)
41*7d9fa7c3SPedro Giffuni            print("parsing '%s'"%self.buffer)
42cdf0e10cSrcweir
43cdf0e10cSrcweir        i = 0
44cdf0e10cSrcweir        bufSize = len(self.buffer)
45cdf0e10cSrcweir        name, buf = '', ''
46cdf0e10cSrcweir        while i < bufSize:
47cdf0e10cSrcweir            c = self.buffer[i]
48cdf0e10cSrcweir            if c in [' ', "\t"] and len(name) == 0:
49cdf0e10cSrcweir                # This is a simple macro with no arguments.
50cdf0e10cSrcweir                name = buf
51cdf0e10cSrcweir                vars = []
52cdf0e10cSrcweir                content = self.buffer[i:]
53cdf0e10cSrcweir                self.setMacro(name, vars, content)
54cdf0e10cSrcweir                return
55cdf0e10cSrcweir            elif c == '(' and len(name) == 0:
56cdf0e10cSrcweir                # This one has arguments.
57cdf0e10cSrcweir                name = buf
58cdf0e10cSrcweir                buf = self.buffer[i:]
59cdf0e10cSrcweir                vars, content = self.parseArgs(buf)
60cdf0e10cSrcweir                self.setMacro(name, vars, content)
61cdf0e10cSrcweir                return
62cdf0e10cSrcweir            else:
63cdf0e10cSrcweir                buf += c
64cdf0e10cSrcweir                i += 1
65cdf0e10cSrcweir
66cdf0e10cSrcweir    def parseArgs (self, buffer):
67cdf0e10cSrcweir        """Parse arguments.
68cdf0e10cSrcweir
69ae54856bSPedro GiffuniThe buffer is expected to be formatted like '(a, b, c)' where the first
70cdf0e10cSrcweircharacter is the open paren.
71cdf0e10cSrcweir"""
72cdf0e10cSrcweir        scope = 0
73cdf0e10cSrcweir        buf = ''
74cdf0e10cSrcweir        vars = []
75cdf0e10cSrcweir        content = ''
76cdf0e10cSrcweir        bufSize = len(buffer)
77cdf0e10cSrcweir        i = 0
78cdf0e10cSrcweir        while i < bufSize:
79cdf0e10cSrcweir            c = buffer[i]
80cdf0e10cSrcweir            if c == '(':
81cdf0e10cSrcweir                scope += 1
82cdf0e10cSrcweir            elif c == ')':
83cdf0e10cSrcweir                scope -= 1
84cdf0e10cSrcweir                if len(buf) > 0:
85cdf0e10cSrcweir                    vars.append(buf)
86cdf0e10cSrcweir                if scope == 0:
87cdf0e10cSrcweir                    break
88cdf0e10cSrcweir            elif c == ',':
89cdf0e10cSrcweir                if len(buf) == 0:
90cdf0e10cSrcweir                    raise globals.ParseError ('')
91cdf0e10cSrcweir                vars.append(buf)
92cdf0e10cSrcweir                buf = ''
93cdf0e10cSrcweir            elif c in " \t" and scope > 0:
94cdf0e10cSrcweir                pass
95cdf0e10cSrcweir            else:
96cdf0e10cSrcweir                buf += c
97cdf0e10cSrcweir
98cdf0e10cSrcweir            i += 1
99cdf0e10cSrcweir
100cdf0e10cSrcweir        if scope > 0:
101cdf0e10cSrcweir            raise globals.ParseError ('')
102cdf0e10cSrcweir
103cdf0e10cSrcweir        return vars, buffer[i+1:]
104cdf0e10cSrcweir
105cdf0e10cSrcweir
106cdf0e10cSrcweir    def setMacro (self, name, vars, content):
107cdf0e10cSrcweir        if self.debug:
108*7d9fa7c3SPedro Giffuni            print("-"*68)
109*7d9fa7c3SPedro Giffuni            print("name: %s"%name)
110cdf0e10cSrcweir            for var in vars:
111*7d9fa7c3SPedro Giffuni                print("var: %s"%var)
112cdf0e10cSrcweir            if len(vars) == 0:
113*7d9fa7c3SPedro Giffuni                print("no vars")
114*7d9fa7c3SPedro Giffuni            print("content: '%s'"%content)
115cdf0e10cSrcweir
116cdf0e10cSrcweir        if len(content) > 0:
117cdf0e10cSrcweir            self.macro = Macro(name)
118*7d9fa7c3SPedro Giffuni            for i in range(0, len(vars)):
119cdf0e10cSrcweir                self.macro.vars[vars[i]] = i
120cdf0e10cSrcweir
121cdf0e10cSrcweir            # tokinize it using lexer.
122cdf0e10cSrcweir            mclexer = srclexer.SrcLexer(content)
123cdf0e10cSrcweir            mclexer.expandHeaders = False
124cdf0e10cSrcweir            mclexer.inMacroDefine = True
125cdf0e10cSrcweir            mclexer.tokenize()
126cdf0e10cSrcweir            self.macro.tokens = mclexer.getTokens()
127cdf0e10cSrcweir            if self.debug:
128*7d9fa7c3SPedro Giffuni                print(self.macro.tokens)
129ae54856bSPedro Giffuni
130cdf0e10cSrcweir            if not self.isValidMacro(self.macro):
131cdf0e10cSrcweir                self.macro = None
132cdf0e10cSrcweir
133cdf0e10cSrcweir        if self.debug:
134cdf0e10cSrcweir            if self.macro != None:
135*7d9fa7c3SPedro Giffuni                print("macro registered!")
136cdf0e10cSrcweir            else:
137*7d9fa7c3SPedro Giffuni                print("macro not registered")
138cdf0e10cSrcweir
139cdf0e10cSrcweir    def isValidMacro (self, macro):
140cdf0e10cSrcweir
141cdf0e10cSrcweir        n = len(macro.tokens)
142cdf0e10cSrcweir        if n == 0:
143cdf0e10cSrcweir            return False
144cdf0e10cSrcweir        elif len(macro.name) > 4 and macro.name[1:4] == 'ID_':
145cdf0e10cSrcweir            # We don't want to expand macros like HID_, SID_, WID_, etc.
146cdf0e10cSrcweir            return False
147cdf0e10cSrcweir        return True
148cdf0e10cSrcweir
149cdf0e10cSrcweir
150cdf0e10cSrcweir    def getMacro (self):
151cdf0e10cSrcweir        return self.macro
152