2013-12-22 18:55:48 +01:00
|
|
|
#!/usr/bin/python
|
|
|
|
import os
|
|
|
|
import sys
|
|
|
|
import re
|
2017-10-03 15:19:07 +02:00
|
|
|
import copy
|
2013-12-22 18:55:48 +01:00
|
|
|
|
|
|
|
import monkTools as tools
|
|
|
|
|
|
|
|
sys.path.append(tools.get_current_path(__file__) + "/ply/ply/")
|
|
|
|
sys.path.append(tools.get_current_path(__file__) + "/codeBB/")
|
2017-10-03 15:19:07 +02:00
|
|
|
sys.path.append(tools.get_current_path(__file__) + "/codeMarkDown/")
|
2013-12-22 18:55:48 +01:00
|
|
|
sys.path.append(tools.get_current_path(__file__) + "/codeHL/")
|
|
|
|
|
|
|
|
import lex
|
|
|
|
|
|
|
|
import inspect
|
|
|
|
import monkDebug as debug
|
|
|
|
import monkClass as Class
|
|
|
|
import monkNamespace as Namespace
|
|
|
|
import monkStruct as Struct
|
|
|
|
import monkUnion as Union
|
|
|
|
import monkMethode as Methode
|
|
|
|
import monkEnum as Enum
|
|
|
|
import monkVariable as Variable
|
|
|
|
import monkNode as Node
|
2016-03-25 22:52:03 +01:00
|
|
|
import monkUsing as Using
|
|
|
|
import monkTypedef as Typedef
|
2013-12-22 18:55:48 +01:00
|
|
|
|
|
|
|
tokens = [
|
|
|
|
'NUMBER',
|
2014-06-16 21:53:04 +02:00
|
|
|
'TEMPLATE',
|
2013-12-22 18:55:48 +01:00
|
|
|
'NAME',
|
|
|
|
'OPEN_PAREN',
|
|
|
|
'CLOSE_PAREN',
|
|
|
|
'OPEN_BRACE',
|
|
|
|
'CLOSE_BRACE',
|
|
|
|
'OPEN_SQUARE_BRACKET',
|
|
|
|
'CLOSE_SQUARE_BRACKET',
|
|
|
|
'COLON',
|
|
|
|
'SEMI_COLON',
|
|
|
|
'COMMA',
|
|
|
|
'TAB',
|
|
|
|
'BACKSLASH',
|
|
|
|
'PIPE',
|
|
|
|
'PERCENT',
|
|
|
|
'EXCLAMATION',
|
|
|
|
'CARET',
|
|
|
|
'COMMENT_SINGLELINE_DOC_PREVIOUS',
|
|
|
|
'COMMENT_SINGLELINE_DOC',
|
|
|
|
'COMMENT_SINGLELINE',
|
|
|
|
'COMMENT_MULTILINE_DOC',
|
|
|
|
'COMMENT_MULTILINE',
|
|
|
|
'PRECOMP',
|
|
|
|
'ASTERISK',
|
|
|
|
'AMPERSTAND',
|
|
|
|
'EQUALS',
|
|
|
|
'MINUS',
|
|
|
|
'PLUS',
|
|
|
|
'DIVIDE',
|
|
|
|
'CHAR_LITERAL',
|
|
|
|
'STRING_LITERAL',
|
|
|
|
'NEW_LINE',
|
|
|
|
'SQUOTE',
|
|
|
|
]
|
|
|
|
|
|
|
|
t_ignore = " \r.?@\f"
|
2014-06-16 21:53:04 +02:00
|
|
|
t_TEMPLATE = r'template'
|
|
|
|
t_NUMBER = r'[0-9][0-9XxA-Fa-f]*L?'
|
2013-12-22 18:55:48 +01:00
|
|
|
t_NAME = r'[<>A-Za-z_~][A-Za-z0-9_]*'
|
|
|
|
t_OPEN_PAREN = r'\('
|
|
|
|
t_CLOSE_PAREN = r'\)'
|
|
|
|
t_OPEN_BRACE = r'{'
|
|
|
|
t_CLOSE_BRACE = r'}'
|
|
|
|
t_OPEN_SQUARE_BRACKET = r'\['
|
|
|
|
t_CLOSE_SQUARE_BRACKET = r'\]'
|
|
|
|
t_SEMI_COLON = r';'
|
|
|
|
t_COLON = r':'
|
|
|
|
t_COMMA = r','
|
|
|
|
def t_TAB(t):
|
|
|
|
r'\t'
|
|
|
|
t_BACKSLASH = r'\\'
|
|
|
|
t_PIPE = r'\|'
|
|
|
|
t_PERCENT = r'%'
|
|
|
|
t_CARET = r'\^'
|
|
|
|
t_EXCLAMATION = r'!'
|
|
|
|
def t_PRECOMP(t):
|
|
|
|
r'\#.*?\n'
|
|
|
|
t.value = re.sub(r'\#\#multiline\#\#', "\\\n", t.value)
|
2017-10-03 15:19:07 +02:00
|
|
|
t.lexer.lineno += len(list(filter(lambda a: a=="\n", t.value)))
|
2013-12-22 18:55:48 +01:00
|
|
|
return t
|
|
|
|
def t_COMMENT_SINGLELINE_DOC_PREVIOUS(t):
|
|
|
|
r'//(/|!)<.*?\n'
|
|
|
|
t.lexer.lineno += 1
|
|
|
|
t.value = t.value[4:]
|
|
|
|
while t.value[0] in ['\n', '\t', ' ']:
|
|
|
|
if len(t.value) <= 2:
|
|
|
|
break
|
|
|
|
t.value = t.value[1:]
|
|
|
|
while t.value[-1] in ['\n', '\t', ' ']:
|
|
|
|
if len(t.value) <= 2:
|
|
|
|
break
|
|
|
|
t.value = t.value[:-1]
|
|
|
|
return t
|
|
|
|
def t_COMMENT_SINGLELINE_DOC(t):
|
|
|
|
r'//(/|!).*?\n'
|
|
|
|
t.lexer.lineno += 1
|
|
|
|
t.value = t.value[3:]
|
|
|
|
while t.value[0] in ['\n', '\t', ' ']:
|
|
|
|
if len(t.value) <= 2:
|
|
|
|
break
|
|
|
|
t.value = t.value[1:]
|
|
|
|
while t.value[-1] in ['\n', '\t', ' ']:
|
|
|
|
if len(t.value) <= 2:
|
|
|
|
break
|
|
|
|
t.value = t.value[:-1]
|
|
|
|
return t
|
|
|
|
def t_COMMENT_SINGLELINE(t):
|
|
|
|
r'\/\/.*\n'
|
|
|
|
t.lexer.lineno += 1
|
|
|
|
t_ASTERISK = r'\*'
|
|
|
|
t_MINUS = r'\-'
|
|
|
|
t_PLUS = r'\+'
|
|
|
|
t_DIVIDE = r'/(?!/)'
|
|
|
|
t_AMPERSTAND = r'&'
|
|
|
|
t_EQUALS = r'='
|
|
|
|
t_CHAR_LITERAL = "'.'"
|
|
|
|
t_SQUOTE = "'"
|
|
|
|
#found at http://wordaligned.org/articles/string-literals-and-regular-expressions
|
|
|
|
#TODO: This does not work with the string "bla \" bla"
|
|
|
|
t_STRING_LITERAL = r'"([^"\\]|\\.)*"'
|
|
|
|
#Found at http://ostermiller.org/findcomment.html
|
|
|
|
def t_COMMENT_MULTILINE_DOC(t):
|
|
|
|
r'/\*(\*|!)(\n|.)*?\*/'
|
2017-10-03 15:19:07 +02:00
|
|
|
filter_result = list(filter(lambda a: a=="\n", t.value))
|
|
|
|
t.lexer.lineno += len(filter_result)
|
2013-12-22 18:55:48 +01:00
|
|
|
t.value = re.sub("( |\t)*\*", "", t.value[3:-2])
|
|
|
|
while t.value[0] == '\n':
|
|
|
|
if len(t.value) <= 2:
|
|
|
|
break
|
|
|
|
t.value = t.value[1:]
|
|
|
|
while t.value[-1] in ['\n', '\t', ' ']:
|
|
|
|
if len(t.value) <= 2:
|
|
|
|
break
|
|
|
|
t.value = t.value[:-1]
|
|
|
|
removeLen = 9999
|
|
|
|
|
|
|
|
listElement = t.value.split('\n')
|
|
|
|
for line in listElement:
|
|
|
|
tmpLen = 0
|
|
|
|
for element in line:
|
|
|
|
if element == ' ':
|
|
|
|
tmpLen += 1
|
|
|
|
else:
|
|
|
|
break;
|
|
|
|
if removeLen > tmpLen:
|
|
|
|
removeLen = tmpLen
|
|
|
|
if removeLen == 9999:
|
|
|
|
return t
|
|
|
|
ret = ""
|
|
|
|
isFirst = True
|
|
|
|
for line in listElement:
|
|
|
|
if isFirst == False:
|
|
|
|
ret += '\n'
|
|
|
|
isFirst = False
|
|
|
|
ret += line[removeLen:]
|
|
|
|
t.value = ret
|
|
|
|
return t
|
|
|
|
def t_COMMENT_MULTILINE(t):
|
|
|
|
r'/\*(\n|.)*?\*/'
|
2017-10-03 15:19:07 +02:00
|
|
|
t.lexer.lineno += len(list(filter(lambda a: a=="\n", t.value)))
|
2013-12-22 18:55:48 +01:00
|
|
|
def t_NEWLINE(t):
|
|
|
|
r'\n+'
|
|
|
|
t.lexer.lineno += len(t.value)
|
|
|
|
|
|
|
|
def t_error(v):
|
|
|
|
print( "Lex error: ", v )
|
|
|
|
|
|
|
|
lex.lex()
|
|
|
|
|
|
|
|
##
|
|
|
|
## @brief Join the class name element : ['class', 'Bar', ':', ':', 'Foo'] -> ['class', 'Bar::Foo']
|
|
|
|
## @param table Input table to convert. ex: [':', '\t', 'class', 'Bar', ':', ':', 'Foo']
|
|
|
|
## @return The new table. ex: ['class', 'Bar::Foo']
|
|
|
|
##
|
|
|
|
def create_compleate_class_name(table):
|
2016-03-25 22:52:03 +01:00
|
|
|
debug.verbose("table = " + str(table))
|
|
|
|
# separate ["<XXX"] in ["<", "XXX"]
|
|
|
|
out = []
|
|
|
|
for name in table:
|
|
|
|
if len(name) > 1 \
|
|
|
|
and name[0] == "<":
|
|
|
|
out.append('<')
|
|
|
|
out.append(name[1:])
|
|
|
|
else:
|
|
|
|
out.append(name)
|
2013-12-22 18:55:48 +01:00
|
|
|
table = out
|
2016-03-25 22:52:03 +01:00
|
|
|
# convert [":", ":"] in ["::"]:
|
|
|
|
out = []
|
|
|
|
for name in table:
|
|
|
|
if len(out) == 0:
|
|
|
|
out.append(name)
|
|
|
|
elif name == ":" \
|
|
|
|
and out[-1] == ":":
|
|
|
|
out[-1] += name
|
|
|
|
else:
|
|
|
|
out.append(name)
|
|
|
|
table = out
|
|
|
|
# convert ["|", "|"] in ["||"]:
|
|
|
|
out = []
|
|
|
|
for name in table:
|
|
|
|
if len(out) == 0:
|
|
|
|
out.append(name)
|
|
|
|
elif name == "|" \
|
|
|
|
and out[-1] == "|":
|
|
|
|
out[-1] += name
|
|
|
|
else:
|
|
|
|
out.append(name)
|
|
|
|
table = out
|
|
|
|
# convert ["&", "&"] in ["&&"]:
|
|
|
|
out = []
|
|
|
|
for name in table:
|
|
|
|
if len(out) == 0:
|
|
|
|
out.append(name)
|
|
|
|
elif name == "&" \
|
|
|
|
and out[-1] == "&&":
|
|
|
|
out[-1] += name
|
|
|
|
else:
|
|
|
|
out.append(name)
|
|
|
|
table = out
|
|
|
|
#
|
|
|
|
|
|
|
|
# join operator ...
|
2013-12-22 18:55:48 +01:00
|
|
|
if 'operator' not in "".join(table):
|
|
|
|
out = table
|
|
|
|
else:
|
|
|
|
out = []
|
|
|
|
for name in table:
|
|
|
|
if len(out) == 0:
|
|
|
|
out.append(name)
|
2013-12-23 21:00:43 +01:00
|
|
|
elif out[-1][:8] == 'operator' \
|
|
|
|
and name != '(':
|
|
|
|
if out[-1] == 'operator':
|
|
|
|
out[-1] += ' '
|
|
|
|
if out[-1][-1] not in [' ', '<','>','=', '-', '!', '+', '*', '&', '|', '/']:
|
|
|
|
out[-1] += ' '
|
2013-12-22 18:55:48 +01:00
|
|
|
out[-1] += name
|
|
|
|
else:
|
|
|
|
out.append(name)
|
2016-03-25 22:52:03 +01:00
|
|
|
debug.verbose(" ==> out = " + str(out))
|
2013-12-22 18:55:48 +01:00
|
|
|
return out
|
|
|
|
|
|
|
|
|
|
|
|
class parse_file():
|
|
|
|
|
|
|
|
def gen_debug_space(self):
|
2016-03-24 21:58:34 +01:00
|
|
|
ret = "[" + str(len(self.brace_depth_type)+1) + "]"
|
|
|
|
for iii in range(0,len(self.brace_depth_type)):
|
2013-12-22 18:55:48 +01:00
|
|
|
ret += " "
|
|
|
|
return ret
|
|
|
|
|
|
|
|
def fusion(self, baseNode):
|
2016-03-24 21:58:34 +01:00
|
|
|
baseNode.fusion(self.main_node)
|
2013-12-22 18:55:48 +01:00
|
|
|
return baseNode
|
|
|
|
|
|
|
|
def __init__(self, fileName):
|
2016-03-24 21:58:34 +01:00
|
|
|
self.main_node = Node.MainNode("main-node", "tmp")
|
|
|
|
self.element_parse_stack = []
|
2013-12-22 18:55:48 +01:00
|
|
|
debug.debug("Parse file : '" + fileName + "'")
|
|
|
|
|
2016-03-24 21:58:34 +01:00
|
|
|
self.header_file_name = fileName
|
2013-12-22 18:55:48 +01:00
|
|
|
|
|
|
|
self.anon_union_counter = [-1, 0]
|
|
|
|
# load all the file data :
|
|
|
|
headerFileStr = tools.file_read_data(fileName)
|
|
|
|
|
|
|
|
# Strip out template declarations
|
|
|
|
# TODO : What is the real need ???
|
2014-06-16 21:53:04 +02:00
|
|
|
#headerFileStr = re.sub("template[\t ]*<[^>]*>", "", headerFileStr)
|
2013-12-22 18:55:48 +01:00
|
|
|
# remove all needed \r unneeded ==> this simplify next resExp ...
|
|
|
|
headerFileStr = re.sub("\r", "\r\n", headerFileStr)
|
|
|
|
headerFileStr = re.sub("\r\n\n", "\r\n", headerFileStr)
|
|
|
|
headerFileStr = re.sub("\r", "", headerFileStr)
|
|
|
|
# TODO : Can generate some error ...
|
|
|
|
headerFileStr = re.sub("\#if 0(.*?)(\#endif|\#else)", "", headerFileStr, flags=re.DOTALL)
|
|
|
|
headerFileafter = re.sub("\@interface(.*?)\@end", "", headerFileStr, flags=re.DOTALL)
|
|
|
|
if headerFileStr != headerFileafter :
|
|
|
|
debug.debug(" Objective C interface ... ==> not supported")
|
|
|
|
return
|
|
|
|
|
|
|
|
#debug.verbose(headerFileStr)
|
|
|
|
|
|
|
|
#Filter out Extern "C" statements. These are order dependent
|
|
|
|
headerFileStr = re.sub(r'extern( |\t)+"[Cc]"( |\t)*{', "{", headerFileStr)
|
|
|
|
headerFileStr = re.sub(r'\\\n', "##multiline##", headerFileStr)
|
|
|
|
headerFileStr += '\n'
|
|
|
|
debug.debug(headerFileStr)
|
|
|
|
|
|
|
|
###### debug.info(headerFileStr)
|
|
|
|
self.stack = [] # token stack to find the namespace and the element name ...
|
2013-12-27 07:57:16 +01:00
|
|
|
self.previous = None
|
2016-03-24 21:58:34 +01:00
|
|
|
self.name_stack = [] #
|
|
|
|
self.brace_depth = 0
|
|
|
|
self.brace_depth_type = []
|
|
|
|
self.last_comment = []
|
|
|
|
self.sub_module_count_brace = 0;
|
2013-12-22 18:55:48 +01:00
|
|
|
lex.lex()
|
|
|
|
lex.input(headerFileStr)
|
2016-03-24 21:58:34 +01:00
|
|
|
self.cur_line = 0
|
|
|
|
self.cur_char = 0
|
2016-03-25 22:52:03 +01:00
|
|
|
self.count_pthese = 0
|
2013-12-22 18:55:48 +01:00
|
|
|
while True:
|
|
|
|
tok = lex.token()
|
|
|
|
if not tok:
|
|
|
|
break
|
2014-09-01 07:07:08 +02:00
|
|
|
debug.verbose("TOK: " + str(tok))
|
2013-12-22 18:55:48 +01:00
|
|
|
self.stack.append( tok.value )
|
2016-03-24 21:58:34 +01:00
|
|
|
self.cur_line = tok.lineno
|
|
|
|
self.cur_char = tok.lexpos
|
2013-12-22 18:55:48 +01:00
|
|
|
# special case to remove internal function define in header:
|
|
|
|
if self.previous_is('function') == True:
|
|
|
|
if tok.type == 'OPEN_BRACE':
|
2016-03-24 21:58:34 +01:00
|
|
|
self.sub_module_count_brace += 1
|
|
|
|
debug.verbose("openBrace " + str(self.sub_module_count_brace))
|
2013-12-22 18:55:48 +01:00
|
|
|
elif tok.type == 'CLOSE_BRACE':
|
2016-03-24 21:58:34 +01:00
|
|
|
self.sub_module_count_brace -= 1
|
|
|
|
debug.verbose("closeBrace " + str(self.sub_module_count_brace) + " line:" + str(self.cur_line))
|
|
|
|
if self.sub_module_count_brace <= 0:
|
2013-12-22 18:55:48 +01:00
|
|
|
self.brace_type_pop()
|
2016-03-24 21:58:34 +01:00
|
|
|
self.last_comment = []
|
2013-12-22 18:55:48 +01:00
|
|
|
continue
|
|
|
|
# normal case:
|
|
|
|
if tok.type == 'PRECOMP':
|
|
|
|
debug.debug("PRECOMP: " + str(tok))
|
|
|
|
self.stack = []
|
2016-03-24 21:58:34 +01:00
|
|
|
self.name_stack = []
|
|
|
|
self.last_comment = []
|
2013-12-22 18:55:48 +01:00
|
|
|
# Do nothing for macro ==> many time not needed ...
|
|
|
|
continue
|
|
|
|
if tok.type == 'COMMENT_SINGLELINE_DOC_PREVIOUS':
|
2013-12-27 07:57:16 +01:00
|
|
|
if self.previous_is('enum') == True:
|
2016-03-24 21:58:34 +01:00
|
|
|
if self.name_stack[-1] == ",":
|
|
|
|
self.name_stack[-1] = "//!< " + tok.value
|
|
|
|
self.name_stack.append(",")
|
2013-12-27 07:57:16 +01:00
|
|
|
else:
|
2016-03-24 21:58:34 +01:00
|
|
|
self.name_stack.append("//!< " + tok.value)
|
2013-12-28 12:44:01 +01:00
|
|
|
elif self.previous != None \
|
|
|
|
and self.previous.get_node_type() == 'variable':
|
|
|
|
self.previous.add_doc([tok.value])
|
2013-12-27 07:57:16 +01:00
|
|
|
else:
|
2016-03-24 21:58:34 +01:00
|
|
|
#self.last_comment.append(tok.value)
|
2013-12-27 07:57:16 +01:00
|
|
|
pass
|
2013-12-22 18:55:48 +01:00
|
|
|
if tok.type == 'COMMENT_MULTILINE_DOC':
|
2016-03-24 21:58:34 +01:00
|
|
|
self.last_comment.append(tok.value)
|
2013-12-22 18:55:48 +01:00
|
|
|
if tok.type == 'COMMENT_SINGLELINE_DOC':
|
2016-03-24 21:58:34 +01:00
|
|
|
self.last_comment.append(tok.value)
|
2013-12-22 18:55:48 +01:00
|
|
|
if tok.type == 'OPEN_BRACE':
|
2016-03-25 22:52:03 +01:00
|
|
|
if self.count_pthese >= 1:
|
|
|
|
# special case of lamba declaration inside initialisation of constructor
|
|
|
|
self.name_stack.append(tok.value)
|
|
|
|
debug.info("plop 0 " +str(self.count_pthese))
|
2013-12-22 18:55:48 +01:00
|
|
|
else:
|
2016-03-25 22:52:03 +01:00
|
|
|
# When we open a brace, this is the time to parse the stack ...
|
|
|
|
# Clean the stack : (remove \t\r\n , and concatenate the 'xx', ':', ':', 'yy' in 'xx::yy',
|
|
|
|
self.name_stack = create_compleate_class_name(self.name_stack)
|
|
|
|
if len(self.name_stack) <= 0:
|
|
|
|
#open brace with no name ...
|
|
|
|
self.brace_type_push('empty', [])
|
|
|
|
elif is_a_function(self.name_stack):
|
|
|
|
# need to parse sub function internal description...
|
|
|
|
self.sub_module_count_brace = 1
|
|
|
|
self.brace_type_push('function', self.name_stack)
|
|
|
|
debug.verbose("openBrace *** " + str(self.name_stack))
|
|
|
|
elif 'namespace' in self.name_stack:
|
|
|
|
self.brace_type_push('namespace', self.name_stack)
|
|
|
|
elif 'class' in self.name_stack:
|
|
|
|
self.brace_type_push('class', self.name_stack)
|
|
|
|
elif 'enum' in self.name_stack:
|
|
|
|
self.brace_type_push('enum', self.name_stack)
|
|
|
|
elif 'struct' in self.name_stack:
|
|
|
|
self.brace_type_push('struct', self.name_stack)
|
|
|
|
elif 'typedef' in self.name_stack:
|
|
|
|
self.brace_type_push('typedef', self.name_stack)
|
|
|
|
elif 'using' in self.name_stack:
|
|
|
|
self.brace_type_push('using', self.name_stack)
|
|
|
|
elif 'union' in self.name_stack:
|
|
|
|
self.brace_type_push('union', self.name_stack)
|
2013-12-22 18:55:48 +01:00
|
|
|
else:
|
2016-03-25 22:52:03 +01:00
|
|
|
self.brace_type_push('unknow', self.name_stack)
|
|
|
|
self.stack = []
|
|
|
|
self.name_stack = []
|
|
|
|
self.last_comment = []
|
|
|
|
elif tok.type == 'CLOSE_BRACE':
|
|
|
|
if self.count_pthese >= 1:
|
|
|
|
debug.info("plop 2 " +str(self.count_pthese))
|
|
|
|
# special case of lamba declaration inside initialisation of constructor
|
|
|
|
self.name_stack.append(tok.value)
|
|
|
|
else:
|
|
|
|
if len(self.name_stack) != 0:
|
|
|
|
if self.previous_is('enum') == True:
|
|
|
|
self.brace_type_append('enum list', self.name_stack);
|
|
|
|
else:
|
|
|
|
debug.warning(self.gen_debug_space() + "end brace DROP : " + str(self.name_stack));
|
|
|
|
self.stack = []
|
|
|
|
self.name_stack = []
|
|
|
|
self.last_comment = []
|
|
|
|
self.brace_type_pop()
|
|
|
|
self.name_stack = create_compleate_class_name(self.name_stack)
|
2013-12-22 18:55:48 +01:00
|
|
|
if tok.type == 'OPEN_PAREN':
|
2016-03-25 22:52:03 +01:00
|
|
|
self.count_pthese += 1
|
2016-03-24 21:58:34 +01:00
|
|
|
self.name_stack.append(tok.value)
|
2013-12-22 18:55:48 +01:00
|
|
|
elif tok.type == 'CLOSE_PAREN':
|
2016-03-25 22:52:03 +01:00
|
|
|
self.count_pthese -= 1
|
2016-03-24 21:58:34 +01:00
|
|
|
self.name_stack.append(tok.value)
|
2013-12-22 18:55:48 +01:00
|
|
|
elif tok.type == 'OPEN_SQUARE_BRACKET':
|
2016-03-24 21:58:34 +01:00
|
|
|
self.name_stack.append(tok.value)
|
2013-12-22 18:55:48 +01:00
|
|
|
elif tok.type == 'CLOSE_SQUARE_BRACKET':
|
2016-03-24 21:58:34 +01:00
|
|
|
self.name_stack.append(tok.value)
|
2013-12-22 18:55:48 +01:00
|
|
|
elif tok.type == 'EQUALS':
|
2016-03-24 21:58:34 +01:00
|
|
|
self.name_stack.append(tok.value)
|
2013-12-22 18:55:48 +01:00
|
|
|
elif tok.type == 'COMMA':
|
2016-03-24 21:58:34 +01:00
|
|
|
self.name_stack.append(tok.value)
|
2013-12-22 18:55:48 +01:00
|
|
|
elif tok.type == 'BACKSLASH':
|
2016-03-24 21:58:34 +01:00
|
|
|
self.name_stack.append(tok.value)
|
2013-12-22 18:55:48 +01:00
|
|
|
elif tok.type == 'PIPE':
|
2016-03-24 21:58:34 +01:00
|
|
|
self.name_stack.append(tok.value)
|
2013-12-22 18:55:48 +01:00
|
|
|
elif tok.type == 'PERCENT':
|
2016-03-24 21:58:34 +01:00
|
|
|
self.name_stack.append(tok.value)
|
2013-12-22 18:55:48 +01:00
|
|
|
elif tok.type == 'CARET':
|
2016-03-24 21:58:34 +01:00
|
|
|
self.name_stack.append(tok.value)
|
2013-12-22 18:55:48 +01:00
|
|
|
elif tok.type == 'EXCLAMATION':
|
2016-03-24 21:58:34 +01:00
|
|
|
self.name_stack.append(tok.value)
|
2013-12-22 18:55:48 +01:00
|
|
|
elif tok.type == 'SQUOTE':
|
2016-03-24 21:58:34 +01:00
|
|
|
self.name_stack.append(tok.value)
|
2013-12-22 18:55:48 +01:00
|
|
|
elif tok.type == 'NUMBER':
|
2016-03-24 21:58:34 +01:00
|
|
|
self.name_stack.append(tok.value)
|
2013-12-22 18:55:48 +01:00
|
|
|
elif tok.type == 'MINUS':
|
2016-03-24 21:58:34 +01:00
|
|
|
self.name_stack.append(tok.value)
|
2013-12-22 18:55:48 +01:00
|
|
|
elif tok.type == 'PLUS':
|
2016-03-24 21:58:34 +01:00
|
|
|
self.name_stack.append(tok.value)
|
2013-12-22 18:55:48 +01:00
|
|
|
elif tok.type == 'STRING_LITERAL':
|
2016-03-24 21:58:34 +01:00
|
|
|
self.name_stack.append(tok.value)
|
2013-12-22 18:55:48 +01:00
|
|
|
elif tok.type == 'NAME' \
|
|
|
|
or tok.type == 'AMPERSTAND' \
|
|
|
|
or tok.type == 'ASTERISK' \
|
|
|
|
or tok.type == 'CHAR_LITERAL':
|
2016-03-24 21:58:34 +01:00
|
|
|
self.name_stack.append(tok.value)
|
2013-12-22 18:55:48 +01:00
|
|
|
elif tok.type == 'COLON':
|
2016-03-25 22:52:03 +01:00
|
|
|
if self.name_stack[0] in Node.access_list:
|
2016-03-24 21:58:34 +01:00
|
|
|
debug.debug(self.gen_debug_space() + "change visibility : " + self.name_stack[0]);
|
|
|
|
self.brace_type_change_access(self.name_stack[0])
|
|
|
|
self.name_stack = []
|
2013-12-22 18:55:48 +01:00
|
|
|
self.stack = []
|
|
|
|
else :
|
2016-03-24 21:58:34 +01:00
|
|
|
self.name_stack.append(tok.value)
|
2013-12-22 18:55:48 +01:00
|
|
|
elif tok.type == 'SEMI_COLON':
|
2016-03-25 22:52:03 +01:00
|
|
|
if self.count_pthese >= 1:
|
|
|
|
debug.info("plop 3 " +str(self.count_pthese))
|
|
|
|
# special case of lamba declaration inside initialisation of constructor
|
|
|
|
self.name_stack.append(tok.value)
|
|
|
|
else:
|
|
|
|
if len(self.name_stack) != 0:
|
|
|
|
self.name_stack = create_compleate_class_name(self.name_stack)
|
|
|
|
if is_a_function(self.name_stack):
|
|
|
|
self.brace_type_append('function', self.name_stack);
|
|
|
|
elif 'namespace' in self.name_stack:
|
|
|
|
debug.debug(self.gen_debug_space() + "find a namespace DECLARATION : " + str(self.name_stack));
|
|
|
|
elif 'class' in self.name_stack:
|
|
|
|
debug.debug(self.gen_debug_space() + "find a class DECLARATION : " + str(self.name_stack));
|
|
|
|
elif 'enum' in self.name_stack:
|
|
|
|
debug.debug(self.gen_debug_space() + "find a enum DECLARATION : " + str(self.name_stack));
|
|
|
|
elif 'struct' in self.name_stack:
|
|
|
|
debug.debug(self.gen_debug_space() + "find a struct DECLARATION : " + str(self.name_stack));
|
|
|
|
elif 'typedef' in self.name_stack:
|
|
|
|
debug.warning(self.gen_debug_space() + "find a typedef DECLARATION : " + str(self.name_stack) + " ==> lose it ...");
|
|
|
|
#self.brace_type_push('typedef', self.name_stack);
|
|
|
|
elif 'using' in self.name_stack:
|
|
|
|
debug.info(self.gen_debug_space() + "find a using DECLARATION : " + str(self.name_stack));
|
|
|
|
self.brace_type_append('using', self.name_stack);
|
|
|
|
elif 'union' in self.name_stack:
|
|
|
|
debug.debug(self.gen_debug_space() + "find a union DECLARATION : " + str(self.name_stack));
|
2013-12-22 18:55:48 +01:00
|
|
|
else:
|
2016-03-25 22:52:03 +01:00
|
|
|
if self.previous_is('enum') == True:
|
|
|
|
self.brace_type_append('enum list', self.name_stack);
|
|
|
|
else:
|
|
|
|
# TODO : Check if it is true in all case :
|
|
|
|
self.brace_type_append('variable', self.name_stack);
|
|
|
|
#debug.warning(self.gen_debug_space() + "variable : " + str(self.name_stack));
|
|
|
|
self.stack = []
|
|
|
|
self.name_stack = []
|
|
|
|
self.last_comment = []
|
2013-12-22 18:55:48 +01:00
|
|
|
#self.debug_display();
|
|
|
|
|
|
|
|
def debug_display(self):
|
|
|
|
debug.info("Debug display :")
|
2016-03-24 21:58:34 +01:00
|
|
|
self.main_node.debug_display(1)
|
2013-12-22 18:55:48 +01:00
|
|
|
|
|
|
|
def create_element(self, type, stack):
|
|
|
|
ret = None
|
2013-12-28 12:44:01 +01:00
|
|
|
self.previous = None
|
2013-12-22 18:55:48 +01:00
|
|
|
if type == 'empty' \
|
|
|
|
or type == 'enum list':
|
|
|
|
pass
|
|
|
|
elif type == 'namespace':
|
2016-03-24 21:58:34 +01:00
|
|
|
ret = Namespace.Namespace(stack, self.header_file_name, self.cur_line, self.last_comment)
|
2013-12-22 18:55:48 +01:00
|
|
|
elif type == 'class':
|
2016-03-24 21:58:34 +01:00
|
|
|
ret = Class.Class(stack, self.header_file_name, self.cur_line, self.last_comment)
|
2013-12-22 18:55:48 +01:00
|
|
|
elif type == 'struct':
|
2016-03-24 21:58:34 +01:00
|
|
|
ret = Struct.Struct(stack, self.header_file_name, self.cur_line, self.last_comment)
|
2013-12-22 18:55:48 +01:00
|
|
|
elif type == 'typedef':
|
2016-03-25 22:52:03 +01:00
|
|
|
ret = Typedef.Typedef(stack, self.header_file_name, self.cur_line)
|
|
|
|
elif type == 'using':
|
|
|
|
ret = Using.Using(stack, self.header_file_name, self.cur_line)
|
2013-12-22 18:55:48 +01:00
|
|
|
elif type == 'union':
|
2016-03-24 21:58:34 +01:00
|
|
|
ret = Union.Union(stack, self.header_file_name, self.cur_line, self.last_comment)
|
2013-12-22 18:55:48 +01:00
|
|
|
elif type == 'function':
|
2016-03-24 21:58:34 +01:00
|
|
|
#debug.info(str(self.last_comment))
|
2013-12-23 21:00:43 +01:00
|
|
|
if self.get_last_type() == 'class':
|
2016-03-24 21:58:34 +01:00
|
|
|
ret = Methode.Methode(stack, self.header_file_name, self.cur_line, self.last_comment, self.brace_depth_type[len(self.brace_depth_type)-1]['node'].get_name())
|
2013-12-23 21:00:43 +01:00
|
|
|
else:
|
2016-03-24 21:58:34 +01:00
|
|
|
ret = Methode.Methode(stack, self.header_file_name, self.cur_line, self.last_comment)
|
2013-12-22 18:55:48 +01:00
|
|
|
elif type == 'enum':
|
2016-03-24 21:58:34 +01:00
|
|
|
ret = Enum.Enum(stack, self.header_file_name, self.cur_line, self.last_comment)
|
2013-12-22 18:55:48 +01:00
|
|
|
elif type == 'variable':
|
2016-03-24 21:58:34 +01:00
|
|
|
ret = Variable.Variable(stack, self.header_file_name, self.cur_line, self.last_comment)
|
2013-12-28 12:44:01 +01:00
|
|
|
self.previous = ret
|
2013-12-22 18:55:48 +01:00
|
|
|
else:
|
|
|
|
debug.error("unknow type ...")
|
|
|
|
return ret
|
|
|
|
|
|
|
|
def brace_type_push(self, type, stack):
|
|
|
|
debug.debug(self.gen_debug_space() + "find a <<" + type + ">> : " + str(stack));
|
|
|
|
myClassElement = self.create_element(type, stack)
|
|
|
|
element = { 'type' : type,
|
|
|
|
'stack' : stack,
|
|
|
|
'node' : myClassElement
|
|
|
|
}
|
2016-03-24 21:58:34 +01:00
|
|
|
self.brace_depth_type.append(element)
|
2013-12-22 18:55:48 +01:00
|
|
|
#debug.info ("append : " + str(element))
|
|
|
|
|
|
|
|
def brace_type_append_current(self, element, id = -50):
|
|
|
|
if id == -50:
|
2016-03-24 21:58:34 +01:00
|
|
|
id = len(self.brace_depth_type)-1
|
2013-12-22 18:55:48 +01:00
|
|
|
if id >= 0:
|
2016-03-24 21:58:34 +01:00
|
|
|
while self.brace_depth_type[id]['node'] == None:
|
2013-12-22 18:55:48 +01:00
|
|
|
# special case for empty brace, just add it to the upper
|
|
|
|
id -=1
|
|
|
|
if id < 0:
|
|
|
|
break;
|
|
|
|
if id < 0:
|
2016-03-24 21:58:34 +01:00
|
|
|
self.main_node.append(element)
|
2013-12-22 18:55:48 +01:00
|
|
|
else:
|
2016-03-24 21:58:34 +01:00
|
|
|
self.brace_depth_type[id]['node'].append(element)
|
2013-12-22 18:55:48 +01:00
|
|
|
|
|
|
|
def brace_type_append(self, type, stack):
|
|
|
|
debug.debug(self.gen_debug_space() + " append a <<" + type + ">> : " + str(stack));
|
|
|
|
lastType = self.get_last_type()
|
|
|
|
newType = self.create_element(type, stack)
|
|
|
|
if newType != None:
|
|
|
|
self.brace_type_append_current(newType)
|
|
|
|
return
|
|
|
|
# enum sub list:
|
|
|
|
if lastType == 'enum' \
|
|
|
|
and type == 'enum list':
|
2016-03-24 21:58:34 +01:00
|
|
|
id = len(self.brace_depth_type)-1
|
|
|
|
self.brace_depth_type[id]['node'].enum_append(stack)
|
2013-12-22 18:55:48 +01:00
|
|
|
return
|
|
|
|
debug.info("TODO : Parse the special type")
|
|
|
|
|
|
|
|
def brace_type_pop(self):
|
2016-03-24 21:58:34 +01:00
|
|
|
id = len(self.brace_depth_type)-1
|
2013-12-22 18:55:48 +01:00
|
|
|
if id < 0:
|
|
|
|
debug.warning("Try to pop the stack with No more element ...")
|
|
|
|
return
|
2016-03-24 21:58:34 +01:00
|
|
|
if self.brace_depth_type[id]['node'] == None:
|
2013-12-22 18:55:48 +01:00
|
|
|
# nothing to add at the upper ...
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
# add it on the previous
|
2016-03-24 21:58:34 +01:00
|
|
|
self.brace_type_append_current(self.brace_depth_type[id]['node'], id-1)
|
|
|
|
self.brace_depth_type.pop()
|
2013-12-22 18:55:48 +01:00
|
|
|
|
|
|
|
def brace_type_change_access(self, newOne):
|
2016-03-25 22:52:03 +01:00
|
|
|
if newOne not in Node.access_list:
|
2013-12-22 18:55:48 +01:00
|
|
|
debug.error("unknow access type : " + newOne)
|
|
|
|
return
|
2016-03-24 21:58:34 +01:00
|
|
|
id = len(self.brace_depth_type)-1
|
2013-12-22 18:55:48 +01:00
|
|
|
if id >= 0:
|
2016-03-24 21:58:34 +01:00
|
|
|
while self.brace_depth_type[id]['node'] == None:
|
2013-12-22 18:55:48 +01:00
|
|
|
# special case for empty brace, just add it to the upper
|
|
|
|
id -=1
|
|
|
|
if id < 0:
|
|
|
|
break;
|
|
|
|
if id < 0:
|
|
|
|
debug.warning("can not change the main access on the library")
|
|
|
|
else:
|
2016-03-24 21:58:34 +01:00
|
|
|
if self.brace_depth_type[id]['node'].get_access() == None:
|
|
|
|
debug.error("Can not set access in other as : 'class' or 'struct' :" + str(self.brace_depth_type[id]))
|
2013-12-22 18:55:48 +01:00
|
|
|
return
|
2016-03-24 21:58:34 +01:00
|
|
|
self.brace_depth_type[id]['node'].set_access(newOne)
|
2013-12-22 18:55:48 +01:00
|
|
|
|
|
|
|
def previous_is(self, type):
|
|
|
|
if self.get_last_type() == type:
|
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
|
|
|
def get_last_type(self):
|
2016-03-24 21:58:34 +01:00
|
|
|
if len(self.brace_depth_type) > 0:
|
|
|
|
return self.brace_depth_type[len(self.brace_depth_type)-1]['type']
|
2013-12-22 18:55:48 +01:00
|
|
|
return None
|
|
|
|
|
|
|
|
def is_a_function(stack) :
|
|
|
|
# in a function we need to have functionName + ( + )
|
2017-10-03 15:19:07 +02:00
|
|
|
tmp = copy.deepcopy(stack);
|
|
|
|
while len(tmp) > 0\
|
|
|
|
and ( tmp[-1] == 'const'\
|
|
|
|
or tmp[-1] == 'noexcept'\
|
|
|
|
or tmp[-1] == 'override'\
|
|
|
|
or tmp[-1] == 'volatile'):
|
|
|
|
tmp = tmp[:-1]
|
|
|
|
if len(tmp) < 3:
|
2013-12-22 18:55:48 +01:00
|
|
|
return False
|
2017-10-03 15:19:07 +02:00
|
|
|
if ':' in tmp:
|
2013-12-22 18:55:48 +01:00
|
|
|
res = []
|
2017-10-03 15:19:07 +02:00
|
|
|
for element in tmp:
|
2013-12-22 18:55:48 +01:00
|
|
|
if element != ':':
|
|
|
|
res.append(element)
|
|
|
|
else:
|
|
|
|
break
|
2017-10-03 15:19:07 +02:00
|
|
|
tmp = res
|
|
|
|
if tmp[-2] == '=' \
|
|
|
|
and tmp[-1] == '0':
|
|
|
|
tmp = tmp[:-2]
|
|
|
|
#can end with 2 possibilities : ')', 'const' or ')'
|
|
|
|
if tmp[-1] == ')':
|
2013-12-22 18:55:48 +01:00
|
|
|
return True
|
2017-10-03 15:19:07 +02:00
|
|
|
return False
|