|
1
|
# -*- coding:utf-8 -*-
|
|
2
3
4
5
6
7
8
|
'''
Created on 23 sty 2014
@author: mlenart
'''
import re
from pyparsing import *
|
|
9
|
from morfeuszbuilder.utils import exceptions
|
|
10
|
from pyparseString import pyparseString
|
|
11
|
|
|
12
|
identifier = Word(alphas, bodyChars=alphanums+u'_>*+{},')
|
|
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
|
define = Keyword('#define').suppress() + identifier + Optional(Suppress('(') + identifier + Suppress(')')) + restOfLine + LineEnd() + StringEnd()
ifdef = Keyword('#ifdef').suppress() + identifier + LineEnd() + StringEnd()
endif = Keyword('#endif').suppress() + LineEnd() + StringEnd()
class NonArgDefine(object):
def __init__(self, name, val):
self.name = name
self.val = val
def hasArg(self):
return False
class ArgDefine(object):
def __init__(self, name, arg, val):
self.name = name
self.arg = arg
self.val = val
def hasArg(self):
return True
def __str__(self):
return '%s(%s) %s' % (self.name, self.arg, self.val)
def _tryToSubstituteArgDefine(s, t, defines):
defineName = t[0]
substituteValue = t[1]
if defineName in defines and defines[defineName].hasArg():
define = defines[defineName]
return re.sub(r'\b%s\b' % define.arg, substituteValue, define.val)
elif defineName in defines:
return '%s ( %s )' % (defines[defineName].val, substituteValue)
else:
return ' '.join(t)
def _tryToSubstituteNonArgDefine(s, t, defines):
defineName = t[0]
if defineName in defines and not defines[defineName].hasArg():
return defines[defineName].val
else:
return defineName
|
|
58
|
def _processLine(lineNum, line, defines, filename):
|
|
59
60
61
62
63
|
if line.strip():
rule = Forward()
defineInstance = Forward()
localId = identifier.copy()
|
|
64
|
weakLiteral = CaselessLiteral('!weak')
|
|
65
|
|
|
66
|
rule << OneOrMore(defineInstance ^ localId ^ Word('*|+?>') ^ (Literal('(') + rule + Literal(')')) ^ weakLiteral)
|
|
67
68
69
70
71
|
defineInstance << localId + Suppress('(') + rule + Suppress(')')
rule.setParseAction(lambda s, l, t: ' '.join(t))
defineInstance.setParseAction(lambda s, l, t: _tryToSubstituteArgDefine(s, t, defines))
localId.setParseAction(lambda s, l, t: _tryToSubstituteNonArgDefine(s, t, defines))
|
|
72
|
return pyparseString(rule, lineNum, line, filename)[0]
|
|
73
74
75
|
else:
return line
|
|
76
|
def preprocess(inputLines, defs, filename):
|
|
77
78
|
defines = {}
ifdefsStack = []
|
|
79
|
for lineNum, line in inputLines:
|
|
80
|
if line.startswith('#define'):
|
|
81
|
parsedDefine = list(pyparseString(define, lineNum, line, filename))
|
|
82
83
84
85
86
87
88
|
if len(parsedDefine) == 2:
name, val = parsedDefine
defines[name] = NonArgDefine(name, val)
else:
name, arg, val = parsedDefine
localDefines = defines.copy()
localDefines[arg] = NonArgDefine(arg, arg)
|
|
89
|
val = _processLine(lineNum, val, localDefines, filename)
|
|
90
|
defines[name] = ArgDefine(name, arg, val)
|
|
91
|
elif line.startswith('#ifdef'):
|
|
92
|
name = pyparseString(ifdef, lineNum, line, filename)[0]
|
|
93
|
ifdefsStack.append((name, True))
|
|
94
|
elif line.startswith('#else'):
|
|
95
96
|
name, isActive = ifdefsStack[-1]
assert isActive
|
|
97
|
ifdefsStack[-1] = (name, False)
|
|
98
|
# ifdefsStack.pop()
|
|
99
|
elif line.startswith('#endif'):
|
|
100
|
ifdefsStack.pop()
|
|
101
102
|
elif line.startswith('#'):
yield lineNum, line
|
|
103
|
elif len(ifdefsStack) == 0 or all(map(lambda (name, isActive): (name in defs and isActive) or (name not in defs and not isActive), ifdefsStack)):
|
|
104
|
yield lineNum, _processLine(lineNum, line, defines, filename)
|
|
105
|
|