forked from v-p-b/rabbithole
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathrabbithole.py
121 lines (92 loc) · 3.09 KB
/
rabbithole.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
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
# Mark functions with their cumulative cyclomatic complexity
#@author buherator
#@category _NEW_
#@keybinding
#@menupath
#@toolbar
from ghidra.program.util import CyclomaticComplexity
from ghidra.util.task import TaskMonitor
from ghidra.program.model.symbol import SourceType
from docking.widgets import OptionDialog
from ghidra.app.tablechooser import TableChooserExecutor, AddressableRowObject, ColumnDisplay
CYCLO_MAP = {}
CCs = []
cyclomaticComplexity = CyclomaticComplexity();
class RabbitHoleExecutor(TableChooserExecutor):
def execute(self, rowObject):
self.lastRowExecuted = rowObject
return True
def getButtonName(self):
return "I'm late!"
class RabbitHoleRow(AddressableRowObject):
def __init__(self, func, cc):
self.function = func
#self.address = self.function.getEntryPoint()
self.cc = cc
def getAddress(self):
return self.function.getEntryPoint()
def getCycloComplexity(self):
return self.cc
def getName(self):
return self.function.getName()
class RabbitHoleCCColumn(ColumnDisplay):
def getColumnValue(self,rowObj):
return rowObj.getCycloComplexity()
def getColumnName(self):
return "Cumulative Complexity"
def getColumnClass(self):
return str
def compare(self, r1, r2):
c1 = r1.getCycloComplexity()
c2 = r2.getCycloComplexity()
return c1-c2
class RabbitHoleNameColumn(ColumnDisplay):
def getColumnValue(self,rowObj):
return rowObj.getName()
def getColumnName(self):
return "Function name"
def getColumnClass(self):
return str
def compare(self, r1, r2):
n1 = r1.getName()
n2 = r2.getName()
if n1 < n2:
return -1
elif n2 > n1:
return 1
else:
return 0
def recurse_cyclo(func, visited):
entry=func.getEntryPoint().getOffset()
# Avoid loops
if entry in visited:
return 0
else:
visited.add(entry)
# Use the cache
if (entry in CYCLO_MAP):
return CYCLO_MAP[entry]
cc = cyclomaticComplexity.calculateCyclomaticComplexity(func, TaskMonitor.DUMMY)
for f in func.getCalledFunctions(TaskMonitor.DUMMY):
cc += recurse_cyclo(f, visited)
CYCLO_MAP[entry] = cc
return cc
func = getFirstFunction()
userDialog=OptionDialog.showOptionDialog(None, "Rabbit Hole", "Hello Alice! \nDo you want your functions renamed?", "Yes", "No")
if userDialog == 0:
exit()
tableDialog=createTableChooserDialog("Rabbit Hole", RabbitHoleExecutor())
tableDialog.addCustomColumn(RabbitHoleNameColumn())
tableDialog.addCustomColumn(RabbitHoleCCColumn())
while func is not None:
cc = recurse_cyclo(func,set())
CCs.append(cc)
print("%s\t%d" % (func.getName(), cc))
tableDialog.add(RabbitHoleRow(func, cc))
if userDialog == 1:
func.setName("%s_cc%d" % (func.getName(), cc), SourceType.USER_DEFINED)
func=getFunctionAfter(func)
tableDialog.show()
print("Rabbit Hole statistics:")
print("Average CC: %f " % (sum(CCs)/len(CCs)))
print("Median CC: %d" % (CCs[int(len(CCs)/2)]))