| #! /usr/bin/env python | 
 |  | 
 | # objgraph | 
 | # | 
 | # Read "nm -o" input (on IRIX: "nm -Bo") of a set of libraries or modules | 
 | # and print various interesting listings, such as: | 
 | # | 
 | # - which names are used but not defined in the set (and used where), | 
 | # - which names are defined in the set (and where), | 
 | # - which modules use which other modules, | 
 | # - which modules are used by which other modules. | 
 | # | 
 | # Usage: objgraph [-cdu] [file] ... | 
 | # -c: print callers per objectfile | 
 | # -d: print callees per objectfile | 
 | # -u: print usage of undefined symbols | 
 | # If none of -cdu is specified, all are assumed. | 
 | # Use "nm -o" to generate the input (on IRIX: "nm -Bo"), | 
 | # e.g.: nm -o /lib/libc.a | objgraph | 
 |  | 
 |  | 
 | import sys | 
 | import string | 
 | import os | 
 | import getopt | 
 | import regex | 
 |  | 
 | # Types of symbols. | 
 | # | 
 | definitions = 'TRGDSBAEC' | 
 | externals = 'UV' | 
 | ignore = 'Nntrgdsbavuc' | 
 |  | 
 | # Regular expression to parse "nm -o" output. | 
 | # | 
 | matcher = regex.compile('\(.*\):\t?........ \(.\) \(.*\)$') | 
 |  | 
 | # Store "item" in "dict" under "key". | 
 | # The dictionary maps keys to lists of items. | 
 | # If there is no list for the key yet, it is created. | 
 | # | 
 | def store(dict, key, item): | 
 | 	if dict.has_key(key): | 
 | 		dict[key].append(item) | 
 | 	else: | 
 | 		dict[key] = [item] | 
 |  | 
 | # Return a flattened version of a list of strings: the concatenation | 
 | # of its elements with intervening spaces. | 
 | # | 
 | def flat(list): | 
 | 	s = '' | 
 | 	for item in list: | 
 | 		s = s + ' ' + item | 
 | 	return s[1:] | 
 |  | 
 | # Global variables mapping defined/undefined names to files and back. | 
 | # | 
 | file2undef = {} | 
 | def2file = {} | 
 | file2def = {} | 
 | undef2file = {} | 
 |  | 
 | # Read one input file and merge the data into the tables. | 
 | # Argument is an open file. | 
 | # | 
 | def readinput(file): | 
 | 	while 1: | 
 | 		s = file.readline() | 
 | 		if not s: | 
 | 			break | 
 | 		# If you get any output from this line, | 
 | 		# it is probably caused by an unexpected input line: | 
 | 		if matcher.search(s) < 0: s; continue # Shouldn't happen | 
 | 		(ra, rb), (r1a, r1b), (r2a, r2b), (r3a, r3b) = matcher.regs[:4] | 
 | 		fn, name, type = s[r1a:r1b], s[r3a:r3b], s[r2a:r2b] | 
 | 		if type in definitions: | 
 | 			store(def2file, name, fn) | 
 | 			store(file2def, fn, name) | 
 | 		elif type in externals: | 
 | 			store(file2undef, fn, name) | 
 | 			store(undef2file, name, fn) | 
 | 		elif not type in ignore: | 
 | 			print fn + ':' + name + ': unknown type ' + type | 
 |  | 
 | # Print all names that were undefined in some module and where they are | 
 | # defined. | 
 | # | 
 | def printcallee(): | 
 | 	flist = file2undef.keys() | 
 | 	flist.sort() | 
 | 	for file in flist: | 
 | 		print file + ':' | 
 | 		elist = file2undef[file] | 
 | 		elist.sort() | 
 | 		for ext in elist: | 
 | 			if len(ext) >= 8: | 
 | 				tabs = '\t' | 
 | 			else: | 
 | 				tabs = '\t\t' | 
 | 			if not def2file.has_key(ext): | 
 | 				print '\t' + ext + tabs + ' *undefined' | 
 | 			else: | 
 | 				print '\t' + ext + tabs + flat(def2file[ext]) | 
 |  | 
 | # Print for each module the names of the other modules that use it. | 
 | # | 
 | def printcaller(): | 
 | 	files = file2def.keys() | 
 | 	files.sort() | 
 | 	for file in files: | 
 | 		callers = [] | 
 | 		for label in file2def[file]: | 
 | 			if undef2file.has_key(label): | 
 | 				callers = callers + undef2file[label] | 
 | 		if callers: | 
 | 			callers.sort() | 
 | 			print file + ':' | 
 | 			lastfn = '' | 
 | 			for fn in callers: | 
 | 				if fn <> lastfn: | 
 | 					print '\t' + fn | 
 | 				lastfn = fn | 
 | 		else: | 
 | 			print file + ': unused' | 
 |  | 
 | # Print undefine names and where they are used. | 
 | # | 
 | def printundef(): | 
 | 	undefs = {} | 
 | 	for file in file2undef.keys(): | 
 | 		for ext in file2undef[file]: | 
 | 			if not def2file.has_key(ext): | 
 | 				store(undefs, ext, file) | 
 | 	elist = undefs.keys() | 
 | 	elist.sort() | 
 | 	for ext in elist: | 
 | 		print ext + ':' | 
 | 		flist = undefs[ext] | 
 | 		flist.sort() | 
 | 		for file in flist: | 
 | 			print '\t' + file | 
 |  | 
 | # Print warning messages about names defined in more than one file. | 
 | # | 
 | def warndups(): | 
 | 	savestdout = sys.stdout | 
 | 	sys.stdout = sys.stderr | 
 | 	names = def2file.keys() | 
 | 	names.sort() | 
 | 	for name in names: | 
 | 		if len(def2file[name]) > 1: | 
 | 			print 'warning:', name, 'multiply defined:', | 
 | 			print flat(def2file[name]) | 
 | 	sys.stdout = savestdout | 
 |  | 
 | # Main program | 
 | # | 
 | def main(): | 
 | 	try: | 
 | 		optlist, args = getopt.getopt(sys.argv[1:], 'cdu') | 
 | 	except getopt.error: | 
 | 		sys.stdout = sys.stderr | 
 | 		print 'Usage:', os.path.basename(sys.argv[0]), | 
 | 		print           '[-cdu] [file] ...' | 
 | 		print '-c: print callers per objectfile' | 
 | 		print '-d: print callees per objectfile' | 
 | 		print '-u: print usage of undefined symbols' | 
 | 		print 'If none of -cdu is specified, all are assumed.' | 
 | 		print 'Use "nm -o" to generate the input (on IRIX: "nm -Bo"),' | 
 | 		print 'e.g.: nm -o /lib/libc.a | objgraph' | 
 | 		return 1 | 
 | 	optu = optc = optd = 0 | 
 | 	for opt, void in optlist: | 
 | 		if opt == '-u': | 
 | 			optu = 1 | 
 | 		elif opt == '-c': | 
 | 			optc = 1 | 
 | 		elif opt == '-d': | 
 | 			optd = 1 | 
 | 	if optu == optc == optd == 0: | 
 | 		optu = optc = optd = 1 | 
 | 	if not args: | 
 | 		args = ['-'] | 
 | 	for file in args: | 
 | 		if file == '-': | 
 | 			readinput(sys.stdin) | 
 | 		else: | 
 | 			readinput(open(file, 'r')) | 
 | 	# | 
 | 	warndups() | 
 | 	# | 
 | 	more = (optu + optc + optd > 1) | 
 | 	if optd: | 
 | 		if more: | 
 | 			print '---------------All callees------------------' | 
 | 		printcallee() | 
 | 	if optu: | 
 | 		if more: | 
 | 			print '---------------Undefined callees------------' | 
 | 		printundef() | 
 | 	if optc: | 
 | 		if more: | 
 | 			print '---------------All Callers------------------' | 
 | 		printcaller() | 
 | 	return 0 | 
 |  | 
 | # Call the main program. | 
 | # Use its return value as exit status. | 
 | # Catch interrupts to avoid stack trace. | 
 | # | 
 | try: | 
 | 	sys.exit(main()) | 
 | except KeyboardInterrupt: | 
 | 	sys.exit(1) |