2 # Copyright (c) 2013 The Chromium Authors. All rights reserved.
3 # Use of this source code is governed by a BSD-style license that can be
4 # found in the LICENSE file.
6 """ Lexer for PPAPI IDL
8 The lexer uses the PLY library to build a tokenizer which understands both
9 WebIDL and Pepper tokens.
11 WebIDL, and WebIDL regular expressions can be found at:
12 http://www.w3.org/TR/2012/CR-WebIDL-20120419/
14 http://www.dabeaz.com/ply/
21 # Try to load the ply module, if not, then assume it is in the third_party
25 # Disable lint check which fails to find the ply module.
26 # pylint: disable=F0401
29 module_path
, module_name
= os
.path
.split(__file__
)
30 third_party
= os
.path
.join(module_path
, '..', '..', 'third_party')
31 sys
.path
.append(third_party
)
32 # pylint: disable=F0401
38 class IDLLexer(object):
39 # 'literals' is a value expected by lex which specifies a list of valid
40 # literal tokens, meaning the token type and token value are identical.
41 literals
= r
'"*.(){}[],;:=+-/~|&^?<>'
43 # 't_ignore' contains ignored characters (spaces and tabs)
46 # 'tokens' is a value required by lex which specifies the complete list
47 # of valid token types.
54 # Symbol and keywords types
62 # 'keywords' is a map of string to token type. All tokens matching
63 # KEYWORD_OR_SYMBOL are matched against keywords dictionary, to determine
64 # if the token is actually a keyword.
67 'attribute' : 'ATTRIBUTE',
68 'boolean' : 'BOOLEAN',
70 'ByteString' : 'BYTESTRING',
71 'callback' : 'CALLBACK',
73 'creator' : 'CREATOR',
75 'deleter' : 'DELETER',
76 'dictionary' : 'DICTIONARY',
77 'DOMString' : 'DOMSTRING',
82 'exception' : 'EXCEPTION',
84 'implements' : 'IMPLEMENTS',
85 'Infinity' : 'INFINITY',
86 'inherit' : 'INHERIT',
87 'interface' : 'INTERFACE',
88 'legacycaller' : 'LEGACYCALLER',
94 'optional' : 'OPTIONAL',
96 'partial' : 'PARTIAL',
97 'Promise' : 'PROMISE',
98 'readonly' : 'READONLY',
100 'sequence' : 'SEQUENCE',
101 'serializer' : 'SERIALIZER',
105 'stringifier' : 'STRINGIFIER',
106 'typedef' : 'TYPEDEF',
108 'unsigned' : 'UNSIGNED',
109 'unrestricted' : 'UNRESTRICTED',
115 # Lex assumes any value or function in the form of 't_<TYPE>' represents a
116 # regular expression where a match will emit a token of type <TYPE>. In the
117 # case of a function, the function is called when a match is made. These
118 # definitions come from WebIDL.
120 # These need to be methods for lexer construction, despite not using self.
121 # pylint: disable=R0201
122 def t_ELLIPSIS(self
, t
):
126 # Regex needs to be in the docstring
127 # pylint: disable=C0301
128 def t_float(self
, t
):
129 r
'-?(([0-9]+\.[0-9]*|[0-9]*\.[0-9]+)([Ee][+-]?[0-9]+)?|[0-9]+[Ee][+-]?[0-9]+)'
132 def t_integer(self
, t
):
133 r
'-?([1-9][0-9]*|0[Xx][0-9A-Fa-f]+|0[0-7]*)'
137 # A line ending '\n', we use this to increment the line number
138 def t_LINE_END(self
, t
):
140 self
.AddLines(len(t
.value
))
142 # We do not process escapes in the IDL strings. Strings are exclusively
143 # used for attributes and enums, and not used as typical 'C' constants.
144 def t_string(self
, t
):
146 t
.value
= t
.value
[1:-1]
147 self
.AddLines(t
.value
.count('\n'))
150 # A C or C++ style comment: /* xxx */ or //
151 def t_COMMENT(self
, t
):
152 r
'(/\*(.|\n)*?\*/)|(//.*(\n[ \t]*//.*)*)'
153 self
.AddLines(t
.value
.count('\n'))
156 # A symbol or keyword.
157 def t_KEYWORD_OR_SYMBOL(self
, t
):
158 r
'_?[A-Za-z][A-Za-z_0-9]*'
160 # All non-keywords are assumed to be symbols
161 t
.type = self
.keywords
.get(t
.value
, 'identifier')
163 # We strip leading underscores so that you can specify symbols with the same
164 # value as a keywords (E.g. a dictionary named 'interface').
165 if t
.value
[0] == '_':
166 t
.value
= t
.value
[1:]
169 def t_ANY_error(self
, t
):
170 msg
= 'Unrecognized input'
171 line
= self
.Lexer().lineno
173 # If that line has not been accounted for, then we must have hit
174 # EoF, so compute the beginning of the line that caused the problem.
175 if line
>= len(self
.index
):
176 # Find the offset in the line of the first word causing the issue
177 word
= t
.value
.split()[0]
178 offs
= self
.lines
[line
- 1].find(word
)
179 # Add the computed line's starting position
180 self
.index
.append(self
.Lexer().lexpos
- offs
)
181 msg
= 'Unexpected EoF reached after'
183 pos
= self
.Lexer().lexpos
- self
.index
[line
]
184 out
= self
.ErrorMessage(line
, pos
, msg
)
185 sys
.stderr
.write(out
+ '\n')
186 self
._lex
_errors
+= 1
189 def AddLines(self
, count
):
190 # Set the lexer position for the beginning of the next line. In the case
191 # of multiple lines, tokens can not exist on any of the lines except the
192 # last one, so the recorded value for previous lines are unused. We still
193 # fill the array however, to make sure the line count is correct.
194 self
.Lexer().lineno
+= count
195 for _
in range(count
):
196 self
.index
.append(self
.Lexer().lexpos
)
198 def FileLineMsg(self
, line
, msg
):
199 # Generate a message containing the file and line number of a token.
200 filename
= self
.Lexer().filename
202 return "%s(%d) : %s" % (filename
, line
+ 1, msg
)
203 return "<BuiltIn> : %s" % msg
205 def SourceLine(self
, line
, pos
):
206 # Create a source line marker
207 caret
= ' ' * pos
+ '^'
208 # We decrement the line number since the array is 0 based while the
209 # line numbers are 1 based.
210 return "%s\n%s" % (self
.lines
[line
- 1], caret
)
212 def ErrorMessage(self
, line
, pos
, msg
):
213 return "\n%s\n%s" % (
214 self
.FileLineMsg(line
, msg
),
215 self
.SourceLine(line
, pos
))
220 # The token function returns the next token provided by IDLLexer for matching
221 # against the leaf paterns.
224 tok
= self
.Lexer().token()
233 t
= self
.Lexer().token()
239 def Tokenize(self
, data
, filename
='__no_file__'):
242 lexer
.filename
= filename
244 self
.lines
= data
.split('\n')
246 def KnownTokens(self
):
251 self
._lexobj
= lex
.lex(object=self
, lextab
=None, optimize
=0)
254 def _AddToken(self
, token
):
255 if token
in self
.tokens
:
256 raise RuntimeError('Same token: ' + token
)
257 self
.tokens
.append(token
)
259 def _AddTokens(self
, tokens
):
261 self
._AddToken
(token
)
263 def _AddKeywords(self
, keywords
):
266 self
._AddToken
(value
)
267 self
.keywords
[key
] = value
269 def _DelKeywords(self
, keywords
):
271 self
.tokens
.remove(key
.upper())
272 del self
.keywords
[key
]
281 self
._AddTokens
(IDLLexer
.tokens
)
282 self
._AddKeywords
(IDLLexer
.keywords
)
287 # If run by itself, attempt to build the lexer
288 if __name__
== '__main__':
289 lexer_object
= IDLLexer()