blob: f578942ccd53f8b1f23b060507349c1ab14977a3 [file] [log] [blame]
eli.bendersky38165b72011-02-04 08:13:39 +02001import sys
Eli Bendersky3921e8e2010-05-21 09:05:39 +03002from pycparser.c_ast import *
3from pycparser.c_parser import CParser, Coord, ParseError
eli.bendersky38165b72011-02-04 08:13:39 +02004from pycparser.c_lexer import CLexer
Eli Bendersky3921e8e2010-05-21 09:05:39 +03005
6
7def expand_decl(decl):
8 """ Converts the declaration into a nested list.
9 """
10 typ = type(decl)
11
12 if typ == TypeDecl:
13 return ['TypeDecl', expand_decl(decl.type)]
14 elif typ == IdentifierType:
15 return ['IdentifierType', decl.names]
16 elif typ == ID:
17 return ['ID', decl.name]
18 elif typ in [Struct, Union]:
19 decls = [expand_decl(d) for d in decl.decls or []]
20 return [typ.__name__, decl.name, decls]
21 else:
22 nested = expand_decl(decl.type)
23
24 if typ == Decl:
25 if decl.quals:
26 return ['Decl', decl.quals, decl.name, nested]
27 else:
28 return ['Decl', decl.name, nested]
29 elif typ == Typename: # for function parameters
30 if decl.quals:
31 return ['Typename', decl.quals, nested]
32 else:
33 return ['Typename', nested]
34 elif typ == ArrayDecl:
35 dimval = decl.dim.value if decl.dim else ''
36 return ['ArrayDecl', dimval, nested]
37 elif typ == PtrDecl:
38 return ['PtrDecl', nested]
39 elif typ == Typedef:
40 return ['Typedef', decl.name, nested]
41 elif typ == FuncDecl:
42 if decl.args:
43 params = [expand_decl(param) for param in decl.args.params]
44 else:
45 params = []
46 return ['FuncDecl', params, nested]
47
48#-----------------------------------------------------------------
eli.bendersky724b1cc2011-03-05 10:45:08 +020049class NodeVisitor(object):
50 def __init__(self):
51 self.node_stack = []
52
53 def visit(self, node):
54 """ Visit a node.
55 """
56 method = 'visit_' + node.__class__.__name__
57 visitor = getattr(self, method, self.generic_visit)
58 return visitor(node)
59
60 def generic_visit(self, node):
61 """ Called if no explicit visitor function exists for a
62 node. Implements preorder visiting of the node.
63 """
64 print(node)
65 print(self.node_stack)
66 self.node_stack.append(node)
67 for c in node.children():
68 self.visit(c)
69 self.node_stack.pop()
70
Eli Bendersky3921e8e2010-05-21 09:05:39 +030071
eli.bendersky0e0a71f2010-10-09 08:32:00 +020072if __name__ == "__main__":
73 source_code = """
eli.bendersky724b1cc2011-03-05 10:45:08 +020074 typedef int Node, Hash;
Eli Bendersky3921e8e2010-05-21 09:05:39 +030075
eli.bendersky724b1cc2011-03-05 10:45:08 +020076 void HashPrint(Hash* hash, void (*PrintFunc)(char*, char*))
eli.bendersky75092b32011-02-04 09:03:28 +020077 {
eli.bendersky724b1cc2011-03-05 10:45:08 +020078 unsigned int i;
79
80 if (hash == NULL || hash->heads == NULL)
81 return;
82
83 for (i = 0; i < hash->table_size; ++i)
84 {
85 Node* temp = hash->heads[i];
86
87 while (temp != NULL)
88 {
89 temp = temp->next;
90 PrintFunc(temp->entry->key, temp->entry->value);
91 }
92 }
93 }
94"""
eli.bendersky38165b72011-02-04 08:13:39 +020095
96 #--------------- Lexing
eli.bendersky38165b72011-02-04 08:13:39 +020097 #~ def errfoo(msg, a, b):
98 #~ printme(msg)
99 #~ sys.exit()
100 #~ clex = CLexer(errfoo, lambda t: False)
101 #~ clex.build()
102 #~ clex.input(source_code)
103
104 #~ while 1:
105 #~ tok = clex.token()
106 #~ if not tok: break
107
108 #~ printme([tok.value, tok.type, tok.lineno, clex.filename, tok.lexpos])
109
110 #--------------- Parsing
eli.bendersky0e0a71f2010-10-09 08:32:00 +0200111 parser = CParser()
eli.bendersky724b1cc2011-03-05 10:45:08 +0200112 ast = parser.parse(source_code, filename='zz')
113 ast.show(showcoord=True)
114 nv=NodeVisitor()
115 nv.visit(ast)
116
Eli Bendersky3921e8e2010-05-21 09:05:39 +0300117