test__opcode.py 4.1 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495
  1. import dis
  2. from test.support.import_helper import import_module
  3. import unittest
  4. import opcode
  5. _opcode = import_module("_opcode")
  6. from _opcode import stack_effect
  7. class OpcodeTests(unittest.TestCase):
  8. def test_stack_effect(self):
  9. self.assertEqual(stack_effect(dis.opmap['POP_TOP']), -1)
  10. self.assertEqual(stack_effect(dis.opmap['BUILD_SLICE'], 0), -1)
  11. self.assertEqual(stack_effect(dis.opmap['BUILD_SLICE'], 1), -1)
  12. self.assertEqual(stack_effect(dis.opmap['BUILD_SLICE'], 3), -2)
  13. self.assertRaises(ValueError, stack_effect, 30000)
  14. self.assertRaises(ValueError, stack_effect, dis.opmap['BUILD_SLICE'])
  15. self.assertRaises(ValueError, stack_effect, dis.opmap['POP_TOP'], 0)
  16. # All defined opcodes
  17. for name, code in filter(lambda item: item[0] not in dis.deoptmap, dis.opmap.items()):
  18. with self.subTest(opname=name):
  19. if code < dis.HAVE_ARGUMENT:
  20. stack_effect(code)
  21. self.assertRaises(ValueError, stack_effect, code, 0)
  22. else:
  23. stack_effect(code, 0)
  24. self.assertRaises(ValueError, stack_effect, code)
  25. # All not defined opcodes
  26. for code in set(range(256)) - set(dis.opmap.values()):
  27. with self.subTest(opcode=code):
  28. self.assertRaises(ValueError, stack_effect, code)
  29. self.assertRaises(ValueError, stack_effect, code, 0)
  30. def test_stack_effect_jump(self):
  31. JUMP_IF_TRUE_OR_POP = dis.opmap['JUMP_IF_TRUE_OR_POP']
  32. self.assertEqual(stack_effect(JUMP_IF_TRUE_OR_POP, 0), 0)
  33. self.assertEqual(stack_effect(JUMP_IF_TRUE_OR_POP, 0, jump=True), 0)
  34. self.assertEqual(stack_effect(JUMP_IF_TRUE_OR_POP, 0, jump=False), -1)
  35. FOR_ITER = dis.opmap['FOR_ITER']
  36. self.assertEqual(stack_effect(FOR_ITER, 0), 1)
  37. self.assertEqual(stack_effect(FOR_ITER, 0, jump=True), -1)
  38. self.assertEqual(stack_effect(FOR_ITER, 0, jump=False), 1)
  39. JUMP_FORWARD = dis.opmap['JUMP_FORWARD']
  40. self.assertEqual(stack_effect(JUMP_FORWARD, 0), 0)
  41. self.assertEqual(stack_effect(JUMP_FORWARD, 0, jump=True), 0)
  42. self.assertEqual(stack_effect(JUMP_FORWARD, 0, jump=False), 0)
  43. # All defined opcodes
  44. has_jump = dis.hasjabs + dis.hasjrel
  45. for name, code in filter(lambda item: item[0] not in dis.deoptmap, dis.opmap.items()):
  46. with self.subTest(opname=name):
  47. if code < dis.HAVE_ARGUMENT:
  48. common = stack_effect(code)
  49. jump = stack_effect(code, jump=True)
  50. nojump = stack_effect(code, jump=False)
  51. else:
  52. common = stack_effect(code, 0)
  53. jump = stack_effect(code, 0, jump=True)
  54. nojump = stack_effect(code, 0, jump=False)
  55. if code in has_jump:
  56. self.assertEqual(common, max(jump, nojump))
  57. else:
  58. self.assertEqual(jump, common)
  59. self.assertEqual(nojump, common)
  60. class SpecializationStatsTests(unittest.TestCase):
  61. def test_specialization_stats(self):
  62. stat_names = opcode._specialization_stats
  63. specialized_opcodes = [
  64. op[:-len("_ADAPTIVE")].lower() for
  65. op in opcode._specialized_instructions
  66. if op.endswith("_ADAPTIVE")]
  67. self.assertIn('load_attr', specialized_opcodes)
  68. self.assertIn('binary_subscr', specialized_opcodes)
  69. stats = _opcode.get_specialization_stats()
  70. if stats is not None:
  71. self.assertIsInstance(stats, dict)
  72. self.assertCountEqual(stats.keys(), specialized_opcodes)
  73. self.assertCountEqual(
  74. stats['load_attr'].keys(),
  75. stat_names + ['failure_kinds'])
  76. for sn in stat_names:
  77. self.assertIsInstance(stats['load_attr'][sn], int)
  78. self.assertIsInstance(
  79. stats['load_attr']['failure_kinds'],
  80. tuple)
  81. for v in stats['load_attr']['failure_kinds']:
  82. self.assertIsInstance(v, int)
  83. if __name__ == "__main__":
  84. unittest.main()