mirror of
				https://github.com/python/cpython.git
				synced 2025-10-25 10:44:55 +00:00 
			
		
		
		
	bpo-22352: Adjust widths in the output of dis.dis() for large line numbers and (#1153)
instruction offsets. Add tests for widths of opcode names.
This commit is contained in:
		
							parent
							
								
									bf623ae884
								
							
						
					
					
						commit
						d90045f319
					
				
					 3 changed files with 77 additions and 8 deletions
				
			
		
							
								
								
									
										30
									
								
								Lib/dis.py
									
										
									
									
									
								
							
							
						
						
									
										30
									
								
								Lib/dis.py
									
										
									
									
									
								
							|  | @ -175,6 +175,9 @@ def show_code(co, *, file=None): | ||||||
| _Instruction.starts_line.__doc__ = "Line started by this opcode (if any), otherwise None" | _Instruction.starts_line.__doc__ = "Line started by this opcode (if any), otherwise None" | ||||||
| _Instruction.is_jump_target.__doc__ = "True if other code jumps to here, otherwise False" | _Instruction.is_jump_target.__doc__ = "True if other code jumps to here, otherwise False" | ||||||
| 
 | 
 | ||||||
|  | _OPNAME_WIDTH = 20 | ||||||
|  | _OPARG_WIDTH = 5 | ||||||
|  | 
 | ||||||
| class Instruction(_Instruction): | class Instruction(_Instruction): | ||||||
|     """Details for a bytecode operation |     """Details for a bytecode operation | ||||||
| 
 | 
 | ||||||
|  | @ -189,11 +192,12 @@ class Instruction(_Instruction): | ||||||
|          is_jump_target - True if other code jumps to here, otherwise False |          is_jump_target - True if other code jumps to here, otherwise False | ||||||
|     """ |     """ | ||||||
| 
 | 
 | ||||||
|     def _disassemble(self, lineno_width=3, mark_as_current=False): |     def _disassemble(self, lineno_width=3, mark_as_current=False, offset_width=4): | ||||||
|         """Format instruction details for inclusion in disassembly output |         """Format instruction details for inclusion in disassembly output | ||||||
| 
 | 
 | ||||||
|         *lineno_width* sets the width of the line number field (0 omits it) |         *lineno_width* sets the width of the line number field (0 omits it) | ||||||
|         *mark_as_current* inserts a '-->' marker arrow as part of the line |         *mark_as_current* inserts a '-->' marker arrow as part of the line | ||||||
|  |         *offset_width* sets the width of the instruction offset field | ||||||
|         """ |         """ | ||||||
|         fields = [] |         fields = [] | ||||||
|         # Column: Source code line number |         # Column: Source code line number | ||||||
|  | @ -214,12 +218,12 @@ def _disassemble(self, lineno_width=3, mark_as_current=False): | ||||||
|         else: |         else: | ||||||
|             fields.append('  ') |             fields.append('  ') | ||||||
|         # Column: Instruction offset from start of code sequence |         # Column: Instruction offset from start of code sequence | ||||||
|         fields.append(repr(self.offset).rjust(4)) |         fields.append(repr(self.offset).rjust(offset_width)) | ||||||
|         # Column: Opcode name |         # Column: Opcode name | ||||||
|         fields.append(self.opname.ljust(20)) |         fields.append(self.opname.ljust(_OPNAME_WIDTH)) | ||||||
|         # Column: Opcode argument |         # Column: Opcode argument | ||||||
|         if self.arg is not None: |         if self.arg is not None: | ||||||
|             fields.append(repr(self.arg).rjust(5)) |             fields.append(repr(self.arg).rjust(_OPARG_WIDTH)) | ||||||
|             # Column: Opcode argument details |             # Column: Opcode argument details | ||||||
|             if self.argrepr: |             if self.argrepr: | ||||||
|                 fields.append('(' + self.argrepr + ')') |                 fields.append('(' + self.argrepr + ')') | ||||||
|  | @ -339,8 +343,19 @@ def _disassemble_bytes(code, lasti=-1, varnames=None, names=None, | ||||||
|                        *, file=None, line_offset=0): |                        *, file=None, line_offset=0): | ||||||
|     # Omit the line number column entirely if we have no line number info |     # Omit the line number column entirely if we have no line number info | ||||||
|     show_lineno = linestarts is not None |     show_lineno = linestarts is not None | ||||||
|     # TODO?: Adjust width upwards if max(linestarts.values()) >= 1000? |     if show_lineno: | ||||||
|     lineno_width = 3 if show_lineno else 0 |         maxlineno = max(linestarts.values()) + line_offset | ||||||
|  |         if maxlineno >= 1000: | ||||||
|  |             lineno_width = len(str(maxlineno)) | ||||||
|  |         else: | ||||||
|  |             lineno_width = 3 | ||||||
|  |     else: | ||||||
|  |         lineno_width = 0 | ||||||
|  |     maxoffset = len(code) - 2 | ||||||
|  |     if maxoffset >= 10000: | ||||||
|  |         offset_width = len(str(maxoffset)) | ||||||
|  |     else: | ||||||
|  |         offset_width = 4 | ||||||
|     for instr in _get_instructions_bytes(code, varnames, names, |     for instr in _get_instructions_bytes(code, varnames, names, | ||||||
|                                          constants, cells, linestarts, |                                          constants, cells, linestarts, | ||||||
|                                          line_offset=line_offset): |                                          line_offset=line_offset): | ||||||
|  | @ -350,7 +365,8 @@ def _disassemble_bytes(code, lasti=-1, varnames=None, names=None, | ||||||
|         if new_source_line: |         if new_source_line: | ||||||
|             print(file=file) |             print(file=file) | ||||||
|         is_current_instr = instr.offset == lasti |         is_current_instr = instr.offset == lasti | ||||||
|         print(instr._disassemble(lineno_width, is_current_instr), file=file) |         print(instr._disassemble(lineno_width, is_current_instr, offset_width), | ||||||
|  |               file=file) | ||||||
| 
 | 
 | ||||||
| def _disassemble_str(source, *, file=None): | def _disassemble_str(source, *, file=None): | ||||||
|     """Compile the source string, then disassemble the code object.""" |     """Compile the source string, then disassemble the code object.""" | ||||||
|  |  | ||||||
|  | @ -175,6 +175,13 @@ def bug1333982(x=[]): | ||||||
|               6 RETURN_VALUE |               6 RETURN_VALUE | ||||||
| """ | """ | ||||||
| 
 | 
 | ||||||
|  | _BIG_LINENO_FORMAT2 = """\ | ||||||
|  | %4d           0 LOAD_GLOBAL              0 (spam) | ||||||
|  |                2 POP_TOP | ||||||
|  |                4 LOAD_CONST               0 (None) | ||||||
|  |                6 RETURN_VALUE | ||||||
|  | """ | ||||||
|  | 
 | ||||||
| dis_module_expected_results = """\ | dis_module_expected_results = """\ | ||||||
| Disassembly of f: | Disassembly of f: | ||||||
|   4           0 LOAD_CONST               0 (None) |   4           0 LOAD_CONST               0 (None) | ||||||
|  | @ -360,6 +367,17 @@ def test_boundaries(self): | ||||||
|         self.assertEqual(dis.opmap["EXTENDED_ARG"], dis.EXTENDED_ARG) |         self.assertEqual(dis.opmap["EXTENDED_ARG"], dis.EXTENDED_ARG) | ||||||
|         self.assertEqual(dis.opmap["STORE_NAME"], dis.HAVE_ARGUMENT) |         self.assertEqual(dis.opmap["STORE_NAME"], dis.HAVE_ARGUMENT) | ||||||
| 
 | 
 | ||||||
|  |     def test_widths(self): | ||||||
|  |         for opcode, opname in enumerate(dis.opname): | ||||||
|  |             if opname in ('BUILD_MAP_UNPACK_WITH_CALL', | ||||||
|  |                           'BUILD_TUPLE_UNPACK_WITH_CALL'): | ||||||
|  |                 continue | ||||||
|  |             with self.subTest(opname=opname): | ||||||
|  |                 width = dis._OPNAME_WIDTH | ||||||
|  |                 if opcode < dis.HAVE_ARGUMENT: | ||||||
|  |                     width += 1 + dis._OPARG_WIDTH | ||||||
|  |                 self.assertLessEqual(len(opname), width) | ||||||
|  | 
 | ||||||
|     def test_dis(self): |     def test_dis(self): | ||||||
|         self.do_disassembly_test(_f, dis_f) |         self.do_disassembly_test(_f, dis_f) | ||||||
| 
 | 
 | ||||||
|  | @ -387,13 +405,45 @@ def func(count): | ||||||
|             self.do_disassembly_test(func(i), expected) |             self.do_disassembly_test(func(i), expected) | ||||||
| 
 | 
 | ||||||
|         # Test some larger ranges too |         # Test some larger ranges too | ||||||
|         for i in range(300, 5000, 10): |         for i in range(300, 1000, 10): | ||||||
|             expected = _BIG_LINENO_FORMAT % (i + 2) |             expected = _BIG_LINENO_FORMAT % (i + 2) | ||||||
|             self.do_disassembly_test(func(i), expected) |             self.do_disassembly_test(func(i), expected) | ||||||
| 
 | 
 | ||||||
|  |         for i in range(1000, 5000, 10): | ||||||
|  |             expected = _BIG_LINENO_FORMAT2 % (i + 2) | ||||||
|  |             self.do_disassembly_test(func(i), expected) | ||||||
|  | 
 | ||||||
|         from test import dis_module |         from test import dis_module | ||||||
|         self.do_disassembly_test(dis_module, dis_module_expected_results) |         self.do_disassembly_test(dis_module, dis_module_expected_results) | ||||||
| 
 | 
 | ||||||
|  |     def test_big_offsets(self): | ||||||
|  |         def func(count): | ||||||
|  |             namespace = {} | ||||||
|  |             func = "def foo(x):\n " + ";".join(["x = x + 1"] * count) + "\n return x" | ||||||
|  |             exec(func, namespace) | ||||||
|  |             return namespace['foo'] | ||||||
|  | 
 | ||||||
|  |         def expected(count, w): | ||||||
|  |             s = ['''\ | ||||||
|  |            %*d LOAD_FAST                0 (x) | ||||||
|  |            %*d LOAD_CONST               1 (1) | ||||||
|  |            %*d BINARY_ADD | ||||||
|  |            %*d STORE_FAST               0 (x) | ||||||
|  | ''' % (w, 8*i, w, 8*i + 2, w, 8*i + 4, w, 8*i + 6) | ||||||
|  |                  for i in range(count)] | ||||||
|  |             s += ['''\ | ||||||
|  | 
 | ||||||
|  |   3        %*d LOAD_FAST                0 (x) | ||||||
|  |            %*d RETURN_VALUE | ||||||
|  | ''' % (w, 8*count, w, 8*count + 2)] | ||||||
|  |             s[0] = '  2' + s[0][3:] | ||||||
|  |             return ''.join(s) | ||||||
|  | 
 | ||||||
|  |         for i in range(1, 5): | ||||||
|  |             self.do_disassembly_test(func(i), expected(i, 4)) | ||||||
|  |         self.do_disassembly_test(func(1249), expected(1249, 4)) | ||||||
|  |         self.do_disassembly_test(func(1250), expected(1250, 5)) | ||||||
|  | 
 | ||||||
|     def test_disassemble_str(self): |     def test_disassemble_str(self): | ||||||
|         self.do_disassembly_test(expr_str, dis_expr_str) |         self.do_disassembly_test(expr_str, dis_expr_str) | ||||||
|         self.do_disassembly_test(simple_stmt_str, dis_simple_stmt_str) |         self.do_disassembly_test(simple_stmt_str, dis_simple_stmt_str) | ||||||
|  |  | ||||||
|  | @ -313,6 +313,9 @@ Extension Modules | ||||||
| Library | Library | ||||||
| ------- | ------- | ||||||
| 
 | 
 | ||||||
|  | - bpo-22352: Column widths in the output of dis.dis() are now adjusted for | ||||||
|  |   large line numbers and instruction offsets. | ||||||
|  | 
 | ||||||
| - bpo-30061: Fixed crashes in IOBase methods __next__() and readlines() when | - bpo-30061: Fixed crashes in IOBase methods __next__() and readlines() when | ||||||
|   readline() or __next__() respectively return non-sizeable object. |   readline() or __next__() respectively return non-sizeable object. | ||||||
|   Fixed possible other errors caused by not checking results of PyObject_Size(), |   Fixed possible other errors caused by not checking results of PyObject_Size(), | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Serhiy Storchaka
						Serhiy Storchaka