-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathcall_tree.py
executable file
·98 lines (78 loc) · 2.72 KB
/
call_tree.py
1
2
3
4
5
6
7
8
9
10
11
12
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
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
#!/usr/bin/env python3
from clang.cindex import CursorKind, TranslationUnit
from collections import defaultdict
class CallTree():
"""
Represents the call tree for a program. Each function maps to a set
of the functions that it calls. The call tree is a directed graph which
may contain cycles.
"""
def __init__( self ):
self.tree = defaultdict( lambda: set() )
def add( self, caller, callee ):
self.tree[ caller ].add( callee )
def addAll( self, caller, callees ):
self.tree[ caller ] |= callees
def functions( self ):
return list( self.tree.keys() )
def calls( self, caller ):
return self.tree[ caller ]
def augment_with_overrides( self, overrides ):
"""
Mutate |call_tree| such that calls to a parent method also imply
a call to the child method
"""
for func in self.functions():
for callee in list( self.calls( func ) ):
self.addAll( func, overrides[ callee ] )
def size( self ):
return sum( [ len( val ) for val in self.tree.values() ] )
def merge_call_trees( subtrees ):
"""
Union two call trees into a single merged call tree
"""
result = CallTree()
for tree in subtrees:
for caller in tree.functions():
result.addAll( caller, tree.calls( caller ) )
return result
def build_call_tree( translation_unit ):
"""
Build a call tree for the given translation unit. This is a wrapper
function around _rec_build_call_tree which mutates the in-parameter
in place.
"""
result = CallTree()
_rec_build_call_tree(
translation_unit.cursor,
translation_unit,
result,
)
return result
def _rec_build_call_tree( node, caller, call_tree ):
"""
Build a call tree by recursively visiting all the nodes in a given
context. This function mutates |call_tree| in place.
"""
if ( node.kind in [
CursorKind.FUNCTION_DECL,
CursorKind.CXX_METHOD,
CursorKind.CONSTRUCTOR,
CursorKind.DESTRUCTOR ] ):
caller = node
elif ( node.kind == CursorKind.CALL_EXPR ):
if node.referenced:
func = node.referenced
if type( caller ) == TranslationUnit:
caller_str = '_start'
else:
caller_str = caller.get_usr()
call_tree.add( caller_str, func.get_usr() )
for c in node.get_children():
_rec_build_call_tree( c, caller, call_tree )
if __name__ == '__main__':
from ast_helpers import get_translation_unit
import sys
from pprint import pprint
pprint( dict( build_call_tree(
get_translation_unit( sys.argv[1] ) ).tree ) )