| 
									
										
										
										
											1997-11-18 15:47:55 +00:00
										 |  |  | """Disassembler of Python byte code into mnemonics.""" | 
					
						
							| 
									
										
										
										
											1990-12-26 15:40:07 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | import sys | 
					
						
							| 
									
										
										
										
											1997-03-14 04:15:43 +00:00
										 |  |  | import types | 
					
						
							| 
									
										
										
										
											1990-12-26 15:40:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-02-27 21:29:27 +00:00
										 |  |  | from opcode import * | 
					
						
							|  |  |  | from opcode import __all__ as _opcodes_all | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | __all__ = ["dis","disassemble","distb","disco"] + _opcodes_all | 
					
						
							|  |  |  | del _opcodes_all | 
					
						
							| 
									
										
										
										
											2001-01-20 19:54:20 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-01-17 20:05:04 +00:00
										 |  |  | def dis(x=None): | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  |     """Disassemble classes, methods, functions, or code.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     With no argument, disassemble the last traceback. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2002-06-01 00:57:55 +00:00
										 |  |  |     if x is None: | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  |         distb() | 
					
						
							|  |  |  |         return | 
					
						
							|  |  |  |     if type(x) is types.InstanceType: | 
					
						
							|  |  |  |         x = x.__class__ | 
					
						
							| 
									
										
										
										
											2001-01-19 02:41:41 +00:00
										 |  |  |     if hasattr(x, 'im_func'): | 
					
						
							|  |  |  |         x = x.im_func | 
					
						
							|  |  |  |     if hasattr(x, 'func_code'): | 
					
						
							|  |  |  |         x = x.func_code | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  |     if hasattr(x, '__dict__'): | 
					
						
							|  |  |  |         items = x.__dict__.items() | 
					
						
							|  |  |  |         items.sort() | 
					
						
							|  |  |  |         for name, x1 in items: | 
					
						
							|  |  |  |             if type(x1) in (types.MethodType, | 
					
						
							|  |  |  |                             types.FunctionType, | 
					
						
							| 
									
										
										
										
											2003-02-27 21:29:27 +00:00
										 |  |  |                             types.CodeType, | 
					
						
							|  |  |  |                             types.ClassType): | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  |                 print "Disassembly of %s:" % name | 
					
						
							|  |  |  |                 try: | 
					
						
							|  |  |  |                     dis(x1) | 
					
						
							|  |  |  |                 except TypeError, msg: | 
					
						
							|  |  |  |                     print "Sorry:", msg | 
					
						
							|  |  |  |                 print | 
					
						
							| 
									
										
										
										
											2001-01-19 02:41:41 +00:00
										 |  |  |     elif hasattr(x, 'co_code'): | 
					
						
							|  |  |  |         disassemble(x) | 
					
						
							| 
									
										
										
										
											2003-02-27 21:29:27 +00:00
										 |  |  |     elif isinstance(x, str): | 
					
						
							|  |  |  |         disassemble_string(x) | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  |     else: | 
					
						
							| 
									
										
										
										
											2001-01-19 02:41:41 +00:00
										 |  |  |         raise TypeError, \ | 
					
						
							|  |  |  |               "don't know how to disassemble %s objects" % \ | 
					
						
							|  |  |  |               type(x).__name__ | 
					
						
							| 
									
										
										
										
											1990-12-26 15:40:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-01-17 20:05:04 +00:00
										 |  |  | def distb(tb=None): | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  |     """Disassemble a traceback (default: last traceback).""" | 
					
						
							| 
									
										
										
										
											2002-06-01 00:57:55 +00:00
										 |  |  |     if tb is None: | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  |         try: | 
					
						
							|  |  |  |             tb = sys.last_traceback | 
					
						
							|  |  |  |         except AttributeError: | 
					
						
							|  |  |  |             raise RuntimeError, "no last traceback to disassemble" | 
					
						
							|  |  |  |         while tb.tb_next: tb = tb.tb_next | 
					
						
							|  |  |  |     disassemble(tb.tb_frame.f_code, tb.tb_lasti) | 
					
						
							| 
									
										
										
										
											1990-12-26 15:40:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-01-17 20:05:04 +00:00
										 |  |  | def disassemble(co, lasti=-1): | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  |     """Disassemble a code object.""" | 
					
						
							|  |  |  |     code = co.co_code | 
					
						
							|  |  |  |     labels = findlabels(code) | 
					
						
							| 
									
										
										
										
											2003-10-28 12:17:25 +00:00
										 |  |  |     linestarts = dict(findlinestarts(co)) | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  |     n = len(code) | 
					
						
							|  |  |  |     i = 0 | 
					
						
							|  |  |  |     extended_arg = 0 | 
					
						
							| 
									
										
										
										
											2001-01-25 20:08:47 +00:00
										 |  |  |     free = None | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  |     while i < n: | 
					
						
							|  |  |  |         c = code[i] | 
					
						
							|  |  |  |         op = ord(c) | 
					
						
							| 
									
										
										
										
											2003-10-28 12:17:25 +00:00
										 |  |  |         if i in linestarts: | 
					
						
							| 
									
										
										
										
											2002-08-15 14:59:02 +00:00
										 |  |  |             if i > 0: | 
					
						
							|  |  |  |                 print | 
					
						
							| 
									
										
										
										
											2003-10-28 12:17:25 +00:00
										 |  |  |             print "%3d" % linestarts[i], | 
					
						
							| 
									
										
										
										
											2002-08-15 14:59:02 +00:00
										 |  |  |         else: | 
					
						
							|  |  |  |             print '   ', | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  |         if i == lasti: print '-->', | 
					
						
							|  |  |  |         else: print '   ', | 
					
						
							|  |  |  |         if i in labels: print '>>', | 
					
						
							|  |  |  |         else: print '  ', | 
					
						
							| 
									
										
										
										
											2004-02-12 17:35:32 +00:00
										 |  |  |         print repr(i).rjust(4), | 
					
						
							| 
									
										
										
										
											2001-02-09 08:25:29 +00:00
										 |  |  |         print opname[op].ljust(20), | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  |         i = i+1 | 
					
						
							|  |  |  |         if op >= HAVE_ARGUMENT: | 
					
						
							|  |  |  |             oparg = ord(code[i]) + ord(code[i+1])*256 + extended_arg | 
					
						
							|  |  |  |             extended_arg = 0 | 
					
						
							|  |  |  |             i = i+2 | 
					
						
							|  |  |  |             if op == EXTENDED_ARG: | 
					
						
							|  |  |  |                 extended_arg = oparg*65536L | 
					
						
							| 
									
										
										
										
											2004-02-12 17:35:32 +00:00
										 |  |  |             print repr(oparg).rjust(5), | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  |             if op in hasconst: | 
					
						
							| 
									
										
										
										
											2004-02-12 17:35:32 +00:00
										 |  |  |                 print '(' + repr(co.co_consts[oparg]) + ')', | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  |             elif op in hasname: | 
					
						
							|  |  |  |                 print '(' + co.co_names[oparg] + ')', | 
					
						
							|  |  |  |             elif op in hasjrel: | 
					
						
							| 
									
										
										
										
											2004-02-12 17:35:32 +00:00
										 |  |  |                 print '(to ' + repr(i + oparg) + ')', | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  |             elif op in haslocal: | 
					
						
							|  |  |  |                 print '(' + co.co_varnames[oparg] + ')', | 
					
						
							|  |  |  |             elif op in hascompare: | 
					
						
							|  |  |  |                 print '(' + cmp_op[oparg] + ')', | 
					
						
							| 
									
										
										
										
											2001-01-25 20:08:47 +00:00
										 |  |  |             elif op in hasfree: | 
					
						
							|  |  |  |                 if free is None: | 
					
						
							|  |  |  |                     free = co.co_cellvars + co.co_freevars | 
					
						
							|  |  |  |                 print '(' + free[oparg] + ')', | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  |         print | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-02-27 21:29:27 +00:00
										 |  |  | def disassemble_string(code, lasti=-1, varnames=None, names=None, | 
					
						
							|  |  |  |                        constants=None): | 
					
						
							| 
									
										
										
										
											2003-03-07 17:30:48 +00:00
										 |  |  |     labels = findlabels(code) | 
					
						
							|  |  |  |     n = len(code) | 
					
						
							|  |  |  |     i = 0 | 
					
						
							|  |  |  |     while i < n: | 
					
						
							|  |  |  |         c = code[i] | 
					
						
							|  |  |  |         op = ord(c) | 
					
						
							|  |  |  |         if i == lasti: print '-->', | 
					
						
							|  |  |  |         else: print '   ', | 
					
						
							|  |  |  |         if i in labels: print '>>', | 
					
						
							|  |  |  |         else: print '  ', | 
					
						
							| 
									
										
										
										
											2004-02-12 17:35:32 +00:00
										 |  |  |         print repr(i).rjust(4), | 
					
						
							| 
									
										
										
										
											2003-03-07 17:30:48 +00:00
										 |  |  |         print opname[op].ljust(15), | 
					
						
							|  |  |  |         i = i+1 | 
					
						
							|  |  |  |         if op >= HAVE_ARGUMENT: | 
					
						
							|  |  |  |             oparg = ord(code[i]) + ord(code[i+1])*256 | 
					
						
							|  |  |  |             i = i+2 | 
					
						
							| 
									
										
										
										
											2004-02-12 17:35:32 +00:00
										 |  |  |             print repr(oparg).rjust(5), | 
					
						
							| 
									
										
										
										
											2003-03-07 17:30:48 +00:00
										 |  |  |             if op in hasconst: | 
					
						
							|  |  |  |                 if constants: | 
					
						
							| 
									
										
										
										
											2004-02-12 17:35:32 +00:00
										 |  |  |                     print '(' + repr(constants[oparg]) + ')', | 
					
						
							| 
									
										
										
										
											2003-03-07 17:30:48 +00:00
										 |  |  |                 else: | 
					
						
							|  |  |  |                     print '(%d)'%oparg, | 
					
						
							|  |  |  |             elif op in hasname: | 
					
						
							|  |  |  |                 if names is not None: | 
					
						
							|  |  |  |                     print '(' + names[oparg] + ')', | 
					
						
							|  |  |  |                 else: | 
					
						
							|  |  |  |                     print '(%d)'%oparg, | 
					
						
							|  |  |  |             elif op in hasjrel: | 
					
						
							| 
									
										
										
										
											2004-02-12 17:35:32 +00:00
										 |  |  |                 print '(to ' + repr(i + oparg) + ')', | 
					
						
							| 
									
										
										
										
											2003-03-07 17:30:48 +00:00
										 |  |  |             elif op in haslocal: | 
					
						
							|  |  |  |                 if varnames: | 
					
						
							|  |  |  |                     print '(' + varnames[oparg] + ')', | 
					
						
							|  |  |  |                 else: | 
					
						
							|  |  |  |                     print '(%d)' % oparg, | 
					
						
							|  |  |  |             elif op in hascompare: | 
					
						
							|  |  |  |                 print '(' + cmp_op[oparg] + ')', | 
					
						
							|  |  |  |         print | 
					
						
							| 
									
										
										
										
											2003-02-27 21:29:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  | disco = disassemble                     # XXX For backwards compatibility | 
					
						
							| 
									
										
										
										
											1997-01-17 20:05:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1990-12-26 15:40:07 +00:00
										 |  |  | def findlabels(code): | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  |     """Detect all offsets in a byte code which are jump targets.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Return the list of offsets. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     labels = [] | 
					
						
							|  |  |  |     n = len(code) | 
					
						
							|  |  |  |     i = 0 | 
					
						
							|  |  |  |     while i < n: | 
					
						
							|  |  |  |         c = code[i] | 
					
						
							|  |  |  |         op = ord(c) | 
					
						
							|  |  |  |         i = i+1 | 
					
						
							|  |  |  |         if op >= HAVE_ARGUMENT: | 
					
						
							|  |  |  |             oparg = ord(code[i]) + ord(code[i+1])*256 | 
					
						
							|  |  |  |             i = i+2 | 
					
						
							|  |  |  |             label = -1 | 
					
						
							|  |  |  |             if op in hasjrel: | 
					
						
							|  |  |  |                 label = i+oparg | 
					
						
							|  |  |  |             elif op in hasjabs: | 
					
						
							|  |  |  |                 label = oparg | 
					
						
							|  |  |  |             if label >= 0: | 
					
						
							|  |  |  |                 if label not in labels: | 
					
						
							|  |  |  |                     labels.append(label) | 
					
						
							|  |  |  |     return labels | 
					
						
							| 
									
										
										
										
											1990-12-26 15:40:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-10-28 12:17:25 +00:00
										 |  |  | def findlinestarts(code): | 
					
						
							|  |  |  |     """Find the offsets in a byte code which are start of lines in the source.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Generate pairs (offset, lineno) as described in Python/compile.c. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     byte_increments = [ord(c) for c in code.co_lnotab[0::2]] | 
					
						
							|  |  |  |     line_increments = [ord(c) for c in code.co_lnotab[1::2]] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     lastlineno = None | 
					
						
							|  |  |  |     lineno = code.co_firstlineno | 
					
						
							|  |  |  |     addr = 0 | 
					
						
							|  |  |  |     for byte_incr, line_incr in zip(byte_increments, line_increments): | 
					
						
							|  |  |  |         if byte_incr: | 
					
						
							|  |  |  |             if lineno != lastlineno: | 
					
						
							|  |  |  |                 yield (addr, lineno) | 
					
						
							|  |  |  |                 lastlineno = lineno | 
					
						
							|  |  |  |             addr += byte_incr | 
					
						
							|  |  |  |         lineno += line_incr | 
					
						
							|  |  |  |     if lineno != lastlineno: | 
					
						
							|  |  |  |         yield (addr, lineno) | 
					
						
							| 
									
										
										
										
											2000-02-04 17:47:55 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | def _test(): | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  |     """Simple test program to disassemble a file.""" | 
					
						
							|  |  |  |     if sys.argv[1:]: | 
					
						
							|  |  |  |         if sys.argv[2:]: | 
					
						
							|  |  |  |             sys.stderr.write("usage: python dis.py [-|file]\n") | 
					
						
							|  |  |  |             sys.exit(2) | 
					
						
							|  |  |  |         fn = sys.argv[1] | 
					
						
							|  |  |  |         if not fn or fn == "-": | 
					
						
							|  |  |  |             fn = None | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         fn = None | 
					
						
							| 
									
										
										
										
											2002-06-01 00:57:55 +00:00
										 |  |  |     if fn is None: | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  |         f = sys.stdin | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         f = open(fn) | 
					
						
							|  |  |  |     source = f.read() | 
					
						
							| 
									
										
										
										
											2002-06-01 00:57:55 +00:00
										 |  |  |     if fn is not None: | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  |         f.close() | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         fn = "<stdin>" | 
					
						
							|  |  |  |     code = compile(source, fn, "exec") | 
					
						
							|  |  |  |     dis(code) | 
					
						
							| 
									
										
										
										
											2000-02-04 17:47:55 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | if __name__ == "__main__": | 
					
						
							| 
									
										
										
										
											2001-01-14 23:36:06 +00:00
										 |  |  |     _test() |