aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorNick Mathewson <nickm@torproject.org>2004-05-12 20:24:56 +0000
committerNick Mathewson <nickm@torproject.org>2004-05-12 20:24:56 +0000
commit44aecf8628f3961033a2fe9e4304402003ba03f5 (patch)
treee0a14b5ada74cc177dbd078b879091f19d494d87
parent7ee6194f3adf4b818509734a2e8fb256ab86f68b (diff)
downloadtor-44aecf8628f3961033a2fe9e4304402003ba03f5.tar.gz
tor-44aecf8628f3961033a2fe9e4304402003ba03f5.zip
Add nasty kludgy broken call-graph generator
svn:r1855
-rwxr-xr-xcontrib/mdd.py158
1 files changed, 158 insertions, 0 deletions
diff --git a/contrib/mdd.py b/contrib/mdd.py
new file mode 100755
index 0000000000..01bcb8d61a
--- /dev/null
+++ b/contrib/mdd.py
@@ -0,0 +1,158 @@
+#!/home/nickm/bin/python2.3
+
+import re, sys
+import textwrap
+
+files = sys.argv[1:]
+funcDeclaredIn = {}
+fileDeclares = {}
+functionCalls = {}
+funcCalledByFile = {}
+funcCalledByFunc = {}
+
+for fname in files:
+ f = open(fname, 'r')
+ curFunc = "???"
+ functionCalls.setdefault(curFunc,{})
+ lineno = 0
+ for line in f.xreadlines():
+ lineno += 1
+ m = re.match(r'^[^\s/].*\s(\w+)\([^;]*$', line)
+ if m:
+ #print line, "->", m.group(1)
+ curFunc = m.group(1)
+ functionCalls.setdefault(curFunc,{})
+ funcDeclaredIn[m.group(1)] = fname
+ fileDeclares.setdefault(fname, {})[m.group(1)] = 1
+ continue
+ m = re.match(r'^(\w+)\([^;]', line)
+ if m:
+ #print line, "->", m.group(1)
+ curFunc = m.group(1)
+ functionCalls.setdefault(curFunc,{})
+ funcDeclaredIn[m.group(1)] = fname
+ fileDeclares.setdefault(fname, {})[m.group(1)] = 1
+ continue
+ while line:
+ m = re.search(r'(\w+)\(', line)
+ if not m: break
+ #print line, "->", m.group(1)
+ functionCalls[curFunc][m.group(1)] = 1
+ #if curFunc == "???":
+ # print ">>!!!!! at %s:%s"%(fname,lineno)
+ funcCalledByFunc.setdefault(m.group(1), {})[curFunc]=1
+ funcCalledByFile.setdefault(m.group(1), {})[fname]=1
+ line = line[m.end():]
+
+ f.close()
+
+fileUsers = {}
+fileUses = {}
+
+for fname in files:
+ print "%s:"%fname
+ users = {}
+ for func in fileDeclares[fname]:
+ cb = funcCalledByFile.get(func,{}).keys()
+ for f in cb: users[f] = 1
+ #print "users[%s] = %s"%(f,users[f])
+ users = users.keys()
+ users.sort()
+ fileUsers[fname] = users
+ for user in users:
+ fileUses.setdefault(user,[]).append(fname)
+ if user == fname: continue
+ print " from %s:"%user
+ for func in fileDeclares[fname]:
+ if funcCalledByFile.get(func,{}).get(user,0):
+ print " %s()"%func
+
+def wrap(s, pre):
+ return textwrap.fill(s,
+ width=77, initial_indent=pre,
+ subsequent_indent=" "*len(pre))
+
+for fname in files:
+ print
+ print "===== %s"%fname
+ print wrap(" ".join(fileUses[fname]),
+ " Calls: ")
+ print wrap(" ".join(fileUsers[fname]),
+ " Called by: ")
+
+print "=============================="
+
+
+funcnames = functionCalls.keys()
+funcnames.sort()
+
+if 1:
+ for func in funcnames:
+ print "===== %s"%func
+ callers = [c for c in funcCalledByFunc.get(func,{}).keys()
+ if c != "???"]
+ callers.sort()
+ called = [c for c in functionCalls[func].keys() if c != "???"]
+ called.sort()
+ print wrap(" ".join(callers),
+ " Called by:")
+ print wrap(" ".join(called),
+ " Calls:")
+
+# simple topological sort.
+functionDepth = {}
+while 1:
+ BIG = 1000000
+ any = 0
+ for func in funcnames:
+ if functionDepth.has_key(func):
+ continue
+ called = [c for c in functionCalls[func] if c != func and
+ functionCalls.has_key(c)]
+ if len(called) == 0:
+ functionDepth[func] = 0
+ #print "Depth(%s)=%s"%(func,0)
+ any = 1
+ continue
+ calledDepths = [ functionDepth.get(c,BIG) for c in called ]
+ if max(calledDepths) < BIG:
+ d = functionDepth[func] = max(calledDepths)+1
+ #print "Depth(%s)=%s"%(func,d)
+ any = 1
+ continue
+ if not any:
+ break
+
+# compute lexical closure.
+cycCalls = {}
+for func in funcnames:
+ if not functionDepth.has_key(func):
+ calls = [ c for c in functionCalls[func] if c != func and
+ functionCalls.has_key(c) and not functionDepth.has_key(c)]
+ cycCalls[func] = d = {}
+ for c in calls:
+ d[c]=1
+
+
+cycNames = cycCalls.keys()
+while 1:
+ any = 0
+ for func in cycNames:
+ L = len(cycCalls[func])
+ for called in cycCalls[func].keys():
+ cycCalls[func].update(cycCalls[called])
+ if L != len(cycCalls[func]):
+ any = 1
+ if not any:
+ break
+
+depthList = [ (v,k) for k,v in functionDepth.items() ]
+depthList.sort()
+cycList = [ (len(v),k) for k,v in cycCalls.items() ]
+cycList.sort()
+for depth,name in depthList:
+ print "Depth[%s]=%s"%(name,depth)
+for bredth,name in cycList:
+ print "Width[%s]=%s"%(name,bredth)
+
+print "Sorted %s / %s"%(len(functionDepth),len(funcnames))