mirror of
				https://github.com/python/cpython.git
				synced 2025-10-25 10:44:55 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			156 lines
		
	
	
	
		
			6.8 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			156 lines
		
	
	
	
		
			6.8 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| import dis
 | |
| from test.support.import_helper import import_module
 | |
| import unittest
 | |
| import opcode
 | |
| 
 | |
| _opcode = import_module("_opcode")
 | |
| from _opcode import stack_effect
 | |
| 
 | |
| 
 | |
| class OpcodeTests(unittest.TestCase):
 | |
| 
 | |
|     def check_bool_function_result(self, func, ops, expected):
 | |
|         for op in ops:
 | |
|             if isinstance(op, str):
 | |
|                 op = dis.opmap[op]
 | |
|             with self.subTest(opcode=op, func=func):
 | |
|                 self.assertIsInstance(func(op), bool)
 | |
|                 self.assertEqual(func(op), expected)
 | |
| 
 | |
|     def test_invalid_opcodes(self):
 | |
|         invalid = [-100, -1, 255, 512, 513, 1000]
 | |
|         self.check_bool_function_result(_opcode.is_valid, invalid, False)
 | |
|         self.check_bool_function_result(_opcode.has_arg, invalid, False)
 | |
|         self.check_bool_function_result(_opcode.has_const, invalid, False)
 | |
|         self.check_bool_function_result(_opcode.has_name, invalid, False)
 | |
|         self.check_bool_function_result(_opcode.has_jump, invalid, False)
 | |
| 
 | |
|     def test_is_valid(self):
 | |
|         names = [
 | |
|             'CACHE',
 | |
|             'POP_TOP',
 | |
|             'IMPORT_NAME',
 | |
|             'JUMP',
 | |
|             'INSTRUMENTED_RETURN_VALUE',
 | |
|         ]
 | |
|         opcodes = [dis.opmap[opname] for opname in names]
 | |
|         self.check_bool_function_result(_opcode.is_valid, opcodes, True)
 | |
| 
 | |
|     def test_has_arg(self):
 | |
|         has_arg = ['SWAP', 'LOAD_FAST', 'INSTRUMENTED_POP_JUMP_IF_TRUE', 'JUMP']
 | |
|         no_arg = ['SETUP_WITH', 'POP_TOP', 'NOP', 'CACHE']
 | |
|         self.check_bool_function_result(_opcode.has_arg, has_arg, True)
 | |
|         self.check_bool_function_result(_opcode.has_arg, no_arg, False)
 | |
| 
 | |
|     def test_has_const(self):
 | |
|         has_const = ['LOAD_CONST', 'RETURN_CONST', 'KW_NAMES']
 | |
|         no_const = ['SETUP_WITH', 'POP_TOP', 'NOP', 'CACHE']
 | |
|         self.check_bool_function_result(_opcode.has_const, has_const, True)
 | |
|         self.check_bool_function_result(_opcode.has_const, no_const, False)
 | |
| 
 | |
|     def test_has_name(self):
 | |
|         has_name = ['STORE_NAME', 'DELETE_ATTR', 'STORE_GLOBAL', 'IMPORT_FROM',
 | |
|                     'LOAD_FROM_DICT_OR_GLOBALS']
 | |
|         no_name = ['SETUP_WITH', 'POP_TOP', 'NOP', 'CACHE']
 | |
|         self.check_bool_function_result(_opcode.has_name, has_name, True)
 | |
|         self.check_bool_function_result(_opcode.has_name, no_name, False)
 | |
| 
 | |
|     def test_has_jump(self):
 | |
|         has_jump = ['FOR_ITER', 'JUMP_FORWARD', 'JUMP', 'POP_JUMP_IF_TRUE', 'SEND']
 | |
|         no_jump = ['SETUP_WITH', 'POP_TOP', 'NOP', 'CACHE']
 | |
|         self.check_bool_function_result(_opcode.has_jump, has_jump, True)
 | |
|         self.check_bool_function_result(_opcode.has_jump, no_jump, False)
 | |
| 
 | |
|     # the following test is part of the refactor, it will be removed soon
 | |
|     def test_against_legacy_bool_values(self):
 | |
|         # limiting to ops up to ENTER_EXECUTOR, because everything after that
 | |
|         # is not currently categorized correctly in opcode.py.
 | |
|         for op in range(0, opcode.opmap['ENTER_EXECUTOR']):
 | |
|             with self.subTest(op=op):
 | |
|                 if opcode.opname[op] != f'<{op}>':
 | |
|                     self.assertEqual(op in dis.hasarg, _opcode.has_arg(op))
 | |
|                     self.assertEqual(op in dis.hasconst, _opcode.has_const(op))
 | |
|                     self.assertEqual(op in dis.hasname, _opcode.has_name(op))
 | |
|                     self.assertEqual(op in dis.hasjrel, _opcode.has_jump(op))
 | |
| 
 | |
|     def test_stack_effect(self):
 | |
|         self.assertEqual(stack_effect(dis.opmap['POP_TOP']), -1)
 | |
|         self.assertEqual(stack_effect(dis.opmap['BUILD_SLICE'], 0), -1)
 | |
|         self.assertEqual(stack_effect(dis.opmap['BUILD_SLICE'], 1), -1)
 | |
|         self.assertEqual(stack_effect(dis.opmap['BUILD_SLICE'], 3), -2)
 | |
|         self.assertRaises(ValueError, stack_effect, 30000)
 | |
|         # All defined opcodes
 | |
|         has_arg = dis.hasarg
 | |
|         for name, code in filter(lambda item: item[0] not in dis.deoptmap, dis.opmap.items()):
 | |
|             if code >= opcode.MIN_INSTRUMENTED_OPCODE:
 | |
|                 continue
 | |
|             with self.subTest(opname=name):
 | |
|                 stack_effect(code)
 | |
|                 stack_effect(code, 0)
 | |
|         # All not defined opcodes
 | |
|         for code in set(range(256)) - set(dis.opmap.values()):
 | |
|             with self.subTest(opcode=code):
 | |
|                 self.assertRaises(ValueError, stack_effect, code)
 | |
|                 self.assertRaises(ValueError, stack_effect, code, 0)
 | |
| 
 | |
|     def test_stack_effect_jump(self):
 | |
|         FOR_ITER = dis.opmap['FOR_ITER']
 | |
|         self.assertEqual(stack_effect(FOR_ITER, 0), 1)
 | |
|         self.assertEqual(stack_effect(FOR_ITER, 0, jump=True), 1)
 | |
|         self.assertEqual(stack_effect(FOR_ITER, 0, jump=False), 1)
 | |
|         JUMP_FORWARD = dis.opmap['JUMP_FORWARD']
 | |
|         self.assertEqual(stack_effect(JUMP_FORWARD, 0), 0)
 | |
|         self.assertEqual(stack_effect(JUMP_FORWARD, 0, jump=True), 0)
 | |
|         self.assertEqual(stack_effect(JUMP_FORWARD, 0, jump=False), 0)
 | |
|         # All defined opcodes
 | |
|         has_arg = dis.hasarg
 | |
|         has_exc = dis.hasexc
 | |
|         has_jump = dis.hasjabs + dis.hasjrel
 | |
|         for name, code in filter(lambda item: item[0] not in dis.deoptmap, dis.opmap.items()):
 | |
|             if code >= opcode.MIN_INSTRUMENTED_OPCODE:
 | |
|                 continue
 | |
|             with self.subTest(opname=name):
 | |
|                 if code not in has_arg:
 | |
|                     common = stack_effect(code)
 | |
|                     jump = stack_effect(code, jump=True)
 | |
|                     nojump = stack_effect(code, jump=False)
 | |
|                 else:
 | |
|                     common = stack_effect(code, 0)
 | |
|                     jump = stack_effect(code, 0, jump=True)
 | |
|                     nojump = stack_effect(code, 0, jump=False)
 | |
|                 if code in has_jump or code in has_exc:
 | |
|                     self.assertEqual(common, max(jump, nojump))
 | |
|                 else:
 | |
|                     self.assertEqual(jump, common)
 | |
|                     self.assertEqual(nojump, common)
 | |
| 
 | |
| 
 | |
| class SpecializationStatsTests(unittest.TestCase):
 | |
|     def test_specialization_stats(self):
 | |
|         stat_names = ["success", "failure", "hit", "deferred", "miss", "deopt"]
 | |
|         specialized_opcodes = [
 | |
|             op.lower()
 | |
|             for op in opcode._specializations
 | |
|             if opcode._inline_cache_entries[opcode.opmap[op]]
 | |
|         ]
 | |
|         self.assertIn('load_attr', specialized_opcodes)
 | |
|         self.assertIn('binary_subscr', specialized_opcodes)
 | |
| 
 | |
|         stats = _opcode.get_specialization_stats()
 | |
|         if stats is not None:
 | |
|             self.assertIsInstance(stats, dict)
 | |
|             self.assertCountEqual(stats.keys(), specialized_opcodes)
 | |
|             self.assertCountEqual(
 | |
|                 stats['load_attr'].keys(),
 | |
|                 stat_names + ['failure_kinds'])
 | |
|             for sn in stat_names:
 | |
|                 self.assertIsInstance(stats['load_attr'][sn], int)
 | |
|             self.assertIsInstance(
 | |
|                 stats['load_attr']['failure_kinds'],
 | |
|                 tuple)
 | |
|             for v in stats['load_attr']['failure_kinds']:
 | |
|                 self.assertIsInstance(v, int)
 | |
| 
 | |
| 
 | |
| if __name__ == "__main__":
 | |
|     unittest.main()
 | 
