Index: tools/idl_parser/idl_lexer.py |
diff --git a/tools/idl_parser/idl_lexer.py b/tools/idl_parser/idl_lexer.py |
new file mode 100755 |
index 0000000000000000000000000000000000000000..c30de04ff4fd06d3bb4f64027c8161a9e79b3a7a |
--- /dev/null |
+++ b/tools/idl_parser/idl_lexer.py |
@@ -0,0 +1,360 @@ |
+#!/usr/bin/env python |
+# Copyright (c) 2013 The Chromium Authors. All rights reserved. |
+# Use of this source code is governed by a BSD-style license that can be |
+# found in the LICENSE file. |
+ |
+""" Lexer for PPAPI IDL |
+ |
+The lexer is uses the PLY to build a tokenizer which understands WebIDL |
sehr
2013/04/08 16:41:20
s/is uses/uses/
noelallen1
2013/04/11 21:52:09
Done.
|
+tokens. The type of the tokens that match WebIDL will be: |
+ 1- The grammar name exactly: float, integer, string, identifier |
+ 2- The grammer for single characters operators: '?', ':', '{', ... |
sehr
2013/04/08 16:41:20
grammar
noelallen1
2013/04/11 21:52:09
Done.
|
+ 3- The uppercase version of the multicharacter string for keywords |
+ ANY, ATTRIBUTE, BOOLEAN, ... |
+ |
+In addition, there are Pepper specific tokens for comments and inline blocks, |
+and multicharacter operators such as >>, <<, and ... |
+ |
+WebIDL, and WebIDL regular expressions can be found at: |
+ http://www.w3.org/TR/2012/CR-WebIDL-20120419/ |
+PLY can be found at: |
+ http://www.dabeaz.com/ply/ |
+""" |
+ |
+import optparse |
+import os.path |
+import re |
+import sys |
+ |
+# |
+# Try to load the ply module, if not, then assume it is in the third_party |
+# directory, relative to ppapi |
+# |
+try: |
+ # pylint: disable=F0401 |
+ from ply import lex |
+except: |
+ module_path, module_name = os.path.split(__file__) |
+ third_party = os.path.join(module_path, '..', '..', 'third_party') |
+ sys.path.append(third_party) |
+ # pylint: disable=F0401 |
+ from ply import lex |
+ |
+# |
+# IDL Lexer |
+# |
+class IDLLexer(object): |
+ # 'tokens' is a value required by lex which specifies the complete list |
+ # of valid token types. |
+ tokens = [ |
+ # Data types |
+ 'float', |
+ 'integer', |
+ 'string', |
+ |
+ # Operators |
+ 'ELLIPSIS', |
+ 'LSHIFT', |
+ 'RSHIFT', |
+ |
+ # Symbol and keywords types |
+ 'COMMENT', |
+ 'identifier', |
+ |
+ # Pepper Extras |
+ 'INLINE', |
+ ] |
+ |
+ # 'keywords' is a map of string to token type. All tokens matching |
+ # KEYWORD_OR_SYMBOL are matched against keywords dictionary, to determine |
+ # if the token is actually a keyword. |
+ keywords = { |
+ 'any' : 'ANY', |
+ 'attribute' : 'ATTRIBUTE', |
+ 'boolean' : 'BOOLEAN', |
+ 'byte' : 'BYTE', |
+ 'callback' : 'CALLBACK', |
+ 'const' : 'CONST', |
+ 'creator' : 'CREATOR', |
+ 'Date' : 'DATE', |
+ 'deleter' : 'DELETER', |
+ 'dictionary' : 'DICTIONARY', |
+ 'DOMString' : 'DOMSTRING', |
+ 'double' : 'DOUBLE', |
+ 'enum' : 'ENUM', |
+ 'false' : 'FALSE', |
+ 'float' : 'FLOAT', |
+ 'exception' : 'EXCEPTION', |
+ 'getter': 'GETTER', |
+ 'implements' : 'IMPLEMENTS', |
+ 'Infinity' : 'INFINITY', |
+ 'inherit' : 'INHERIT', |
+ 'interface' : 'INTERFACE', |
+ 'label' : 'LABEL', |
+ 'legacycaller' : 'LEGACYCALLER', |
+ 'long' : 'LONG', |
+ 'namespace' : 'NAMESPACE', |
+ 'Nan' : 'NAN', |
+ 'null' : 'NULL', |
+ 'object' : 'OBJECT', |
+ 'octet' : 'OCTET', |
+ 'optional' : 'OPTIONAL', |
+ 'or' : 'OR', |
+ 'partial' : 'PARTIAL', |
+ 'readonly' : 'READONLY', |
+ 'sequence' : 'SEQUENCE', |
+ 'setter': 'SETTER', |
+ 'short' : 'SHORT', |
+ 'static' : 'STATIC', |
+ 'stringifier' : 'STRINGIFIER', |
+ 'struct' : 'STRUCT', |
+ 'typedef' : 'TYPEDEF', |
+ 'true' : 'TRUE', |
+ 'unsigned' : 'UNSIGNED', |
+ 'unrestricted' : 'UNRESTRICTED', |
+ 'void' : 'VOID' |
+ } |
+ |
+ # Add keywords |
+ for key in keywords: |
+ tokens.append(keywords[key]) |
+ |
+ # 'literals' is a value expected by lex which specifies a list of valid |
+ # literal tokens, meaning the token type and token value are identical. |
+ literals = '"*.(){}[],;:=+-/~|&^?<>' |
+ |
+ # Token definitions |
+ # |
+ # Lex assumes any value or function in the form of 't_<TYPE>' represents a |
+ # regular expression where a match will emit a token of type <TYPE>. In the |
+ # case of a function, the function is called when a match is made. These |
+ # definitions come from WebIDL. |
+ |
+ # 't_ignore' is a special match of items to ignore |
+ t_ignore = ' \t' |
+ |
+ # Constant values |
+ t_integer = r'-?(0[Xx][0-9A-Fa-f]+)|([0-9]+[uU]?)' |
+ t_float = r'-?(\d+\.\d*|\d*\.\d+)([Ee][+-]?\d+)?|-?\d+[Ee][+-]?\d+' |
+ |
+ # Special multi-character operators |
+ t_ELLIPSIS = r'\.\.\.' |
+ t_LSHIFT = r'<<' |
+ t_RSHIFT = r'>>' |
+ |
+ # A line ending '\n', we use this to increment the line number |
+ def t_LINE_END(self, t): |
+ r'\n+' |
+ self.AddLines(len(t.value)) |
+ |
+ # We do not process escapes in the IDL strings. Strings are exclusively |
+ # used for attributes and enums, and not used as typical 'C' constants. |
+ def t_string(self, t): |
+ r'"[^"]*"' |
+ t.value = t.value[1:-1] |
+ self.AddLines(t.value.count('\n')) |
+ return t |
+ |
+ # A C or C++ style comment: /* xxx */ or // |
+ def t_COMMENT(self, t): |
+ r'(/\*(.|\n)*?\*/)|(//.*(\n[ \t]*//.*)*)' |
sehr
2013/04/08 16:41:20
what is the "?" there for?
noelallen1
2013/04/11 21:52:09
I will add a new test to the lexer to vet comments
|
+ self.AddLines(t.value.count('\n')) |
+ return t |
+ |
+ # Return a "preprocessor" inline block |
+ def t_INLINE(self, t): |
+ r'\#inline (.|\n)*?\#endinl.*' |
+ self.AddLines(t.value.count('\n')) |
+ return t |
+ |
+ # A symbol or keyword. |
+ def t_KEYWORD_OR_SYMBOL(self, t): |
+ r'_?[A-Za-z][A-Za-z_0-9]*' |
+ |
+ # All non-keywords are assumed to be symbols |
+ t.type = self.keywords.get(t.value, 'identifier') |
+ |
+ # We strip leading underscores so that you can specify symbols with the same |
+ # value as a keywords (E.g. a dictionary named 'interface'). |
+ if t.value[0] == '_': |
+ t.value = t.value[1:] |
+ return t |
+ |
+ def t_ANY_error(self, t): |
+ msg = 'Unrecognized input' |
+ line = self.lexobj.lineno |
+ |
+ # If that line has not been accounted for, then we must have hit |
+ # EoF, so compute the beginning of the line that caused the problem. |
+ if line >= len(self.index): |
+ # Find the offset in the line of the first word causing the issue |
+ word = t.value.split()[0] |
+ offs = self.lines[line - 1].find(word) |
+ # Add the computed line's starting position |
+ self.index.append(self.lexobj.lexpos - offs) |
+ msg = 'Unexpected EoF reached after' |
+ |
+ pos = self.lexobj.lexpos - self.index[line] |
+ out = self.ErrorMessage(line, pos, msg) |
+ sys.stderr.write(out + '\n') |
+ self.lex_errors += 1 |
+ |
+ |
+ def AddLines(self, count): |
+ # Set the lexer position for the beginning of the next line. In the case |
+ # of multiple lines, tokens can not exist on any of the lines except the |
+ # last one, so the recorded value for previous lines are unused. We still |
+ # fill the array however, to make sure the line count is correct. |
+ self.lexobj.lineno += count |
+ # pylint: disable=W0612 |
+ for i in range(count): |
+ self.index.append(self.lexobj.lexpos) |
+ |
+ def FileLineMsg(self, line, msg): |
+ # Generate a message containing the file and line number of a token. |
+ filename = self.lexobj.filename |
+ if filename: |
+ return "%s(%d) : %s" % (filename, line + 1, msg) |
+ return "<BuiltIn> : %s" % msg |
+ |
+ def SourceLine(self, line, pos): |
+ # Create a source line marker |
+ caret = '\t^'.expandtabs(pos) |
sehr
2013/04/08 16:41:20
This is a "cute" way of generating pos * ' '. Is
noelallen1
2013/04/11 21:52:09
' ' * pos + '^' would also work.
So would using w
|
+ # We decrement the line number since the array is 0 based while the |
+ # line numbers are 1 based. |
+ return "%s\n%s" % (self.lines[line - 1], caret) |
+ |
+ def ErrorMessage(self, line, pos, msg): |
+ return "\n%s\n%s" % ( |
+ self.FileLineMsg(line, msg), |
+ self.SourceLine(line, pos)) |
+ |
+ def GetTokens(self): |
+ outlist = [] |
+ while 1: |
+ t = self.lexobj.token() |
+ if t is None: |
+ break |
+ outlist.append(t) |
+ return outlist |
+ |
+ def __init__(self, filename, data): |
+ self.index = [0] |
+ self.lex_errors = 0 |
+ self.lines = data.split('\n') |
+ self.lexobj = lex.lex(object=self, lextab=None, optimize=0) |
+ self.lexobj.filename = filename |
+ self.lexobj.input(data) |
+ |
+ |
+# |
+# FileToTokens |
+# |
+# From a source file generate a list of tokens. |
+# |
+def FileToTokens(filename): |
+ with open(filename, 'rb') as srcfile: |
+ lexer = IDLLexer(filename, srcfile.read()) |
+ return lexer.GetTokens() |
+ |
+ |
+# |
+# TextToTokens |
+# |
+# From a source file generate a list of tokens. |
+# |
+def TextToTokens(text): |
+ lexer = IDLLexer(None, text) |
+ return lexer.GetTokens() |
+ |
+ |
+# |
+# TestSameText |
+# |
+# From a set of tokens, generate a new source text by joining with a |
+# single space. The new source is then tokenized and compared against the |
sehr
2013/04/08 16:41:20
s/space/newline/ ?
noelallen1
2013/04/11 21:52:09
Done.
|
+# old set. |
+# |
+def TestSameText(filename): |
+ tokens1 = FileToTokens(filename) |
+ tokens2 = TextToTokens('\n'.join(tokens1)) |
+ |
+ count1 = len(tokens1) |
+ count2 = len(tokens2) |
+ if count1 != count2: |
+ print "Size mismatch original %d vs %d\n" % (count1, count2) |
+ if count1 > count2: |
+ count1 = count2 |
+ |
+ failed = 0 |
+ for i in range(count1): |
+ if tokens1[i] != tokens2[i]: |
+ print "%d >>%s<< >>%s<<" % (i, tokens1[i], tokens2[i]) |
+ failed = failed + 1 |
+ |
+ return failed |
+ |
+ |
+# |
+# TestExpectedText |
+# |
+# From a set of tokens pairs, verify the type field of the second matches |
+# the value of the first, so that: |
+# integer 123 float 1.1 |
+# will generate a passing test, where the first token has both the type and |
+# value of the keyword integer and the second has the type of integer and |
+# value of 123. |
+# |
+def TestExpect(tokens): |
+ count = len(tokens) |
+ index = 0 |
+ errors = 0 |
+ while index < count: |
+ expect_type = tokens[index].value |
+ actual_type = tokens[index + 1].type |
+ index += 2 |
+ |
+ if expect_type != actual_type: |
+ sys.stderr.write('Mismatch: Expected %s, but got %s = %s.\n' % |
+ (expect_type, actual_type, token.value)) |
+ errors += 1 |
+ |
+ return errors |
+ |
+ |
+def Main(args): |
+ parser = optparse.OptionParser() |
+ parser.add_option('--test', help='Run tests.', action='store_true') |
+ |
+ # If no arguments are provided, run tests. |
+ if len(args) == 0: |
+ args = ['--test', 'test_lexer/values.in', 'test_lexer/keywords.in'] |
sehr
2013/04/08 16:41:20
This is sort of ugly, baking in test input paths.
noelallen1
2013/04/11 21:52:09
This is fixed in a different CL
On 2013/04/08 16:4
|
+ |
+ global options |
+ options, filenames = parser.parse_args(args[1:]) |
+ |
+ if not filenames: |
+ parser.error('No files specified.') |
+ |
+ for filename in filenames: |
+ try: |
+ tokens = FileToTokens(filename) |
+ values = [tok.value for tok in tokens] |
+ errors = 0 |
+ |
+ if options.test: |
+ if TestSameText(values): |
+ sys.stderr.write('Failed text match on %s.\n' % filename) |
+ return -1 |
+ if TestExpect(tokens): |
+ sys.stderr.write('Failed expected type match on %s.\n' % filename) |
+ return -1 |
+ |
+ except lex.LexError as le: |
+ sys.stderr.write('%s\n' % str(le)) |
+ |
+ return 0 |
+ |
+ |
+if __name__ == '__main__': |
+ sys.exit(Main(sys.argv[1:])) |