| 
									
										
										
										
											2004-08-23 23:37:48 +00:00
										 |  |  | import dis | 
					
						
							|  |  |  | import sys | 
					
						
							|  |  |  | from cStringIO import StringIO | 
					
						
							|  |  |  | import unittest | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def disassemble(func): | 
					
						
							|  |  |  |     f = StringIO() | 
					
						
							|  |  |  |     tmp = sys.stdout | 
					
						
							|  |  |  |     sys.stdout = f | 
					
						
							|  |  |  |     dis.dis(func) | 
					
						
							|  |  |  |     sys.stdout = tmp | 
					
						
							|  |  |  |     result = f.getvalue() | 
					
						
							|  |  |  |     f.close() | 
					
						
							|  |  |  |     return result | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def dis_single(line): | 
					
						
							|  |  |  |     return disassemble(compile(line, '', 'single')) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class TestTranforms(unittest.TestCase): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_unot(self): | 
					
						
							|  |  |  |         # UNARY_NOT JUMP_IF_FALSE POP_TOP  -->  JUMP_IF_TRUE POP_TOP' | 
					
						
							|  |  |  |         def unot(x): | 
					
						
							|  |  |  |             if not x == 2: | 
					
						
							|  |  |  |                 del x | 
					
						
							|  |  |  |         asm = disassemble(unot) | 
					
						
							|  |  |  |         for elem in ('UNARY_NOT', 'JUMP_IF_FALSE'): | 
					
						
							|  |  |  |             self.assert_(elem not in asm) | 
					
						
							|  |  |  |         for elem in ('JUMP_IF_TRUE', 'POP_TOP'): | 
					
						
							|  |  |  |             self.assert_(elem in asm) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_elim_inversion_of_is_or_in(self): | 
					
						
							|  |  |  |         for line, elem in ( | 
					
						
							|  |  |  |             ('not a is b', '(is not)',), | 
					
						
							|  |  |  |             ('not a in b', '(not in)',), | 
					
						
							|  |  |  |             ('not a is not b', '(is)',), | 
					
						
							|  |  |  |             ('not a not in b', '(in)',), | 
					
						
							|  |  |  |             ): | 
					
						
							|  |  |  |             asm = dis_single(line) | 
					
						
							|  |  |  |             self.assert_(elem in asm) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_none_as_constant(self): | 
					
						
							|  |  |  |         # LOAD_GLOBAL None  -->  LOAD_CONST None | 
					
						
							|  |  |  |         def f(x): | 
					
						
							| 
									
										
										
										
											2004-08-26 05:23:19 +00:00
										 |  |  |             None | 
					
						
							|  |  |  |             return x | 
					
						
							| 
									
										
										
										
											2004-08-23 23:37:48 +00:00
										 |  |  |         asm = disassemble(f) | 
					
						
							|  |  |  |         for elem in ('LOAD_GLOBAL',): | 
					
						
							|  |  |  |             self.assert_(elem not in asm) | 
					
						
							|  |  |  |         for elem in ('LOAD_CONST', '(None)'): | 
					
						
							|  |  |  |             self.assert_(elem in asm) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_while_one(self): | 
					
						
							|  |  |  |         # Skip over:  LOAD_CONST trueconst  JUMP_IF_FALSE xx  POP_TOP | 
					
						
							|  |  |  |         def f(): | 
					
						
							| 
									
										
										
										
											2004-08-26 05:23:19 +00:00
										 |  |  |             while 1: | 
					
						
							|  |  |  |                 pass | 
					
						
							|  |  |  |             return list | 
					
						
							| 
									
										
										
										
											2004-08-23 23:37:48 +00:00
										 |  |  |         asm = disassemble(f) | 
					
						
							|  |  |  |         for elem in ('LOAD_CONST', 'JUMP_IF_FALSE'): | 
					
						
							|  |  |  |             self.assert_(elem not in asm) | 
					
						
							|  |  |  |         for elem in ('JUMP_ABSOLUTE',): | 
					
						
							|  |  |  |             self.assert_(elem in asm) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_pack_unpack(self): | 
					
						
							|  |  |  |         for line, elem in ( | 
					
						
							| 
									
										
										
										
											2004-09-22 18:44:21 +00:00
										 |  |  |             ('a, = a,', 'LOAD_CONST',), | 
					
						
							|  |  |  |             ('a, b = a, b', 'ROT_TWO',), | 
					
						
							|  |  |  |             ('a, b, c = a, b, c', 'ROT_THREE',), | 
					
						
							| 
									
										
										
										
											2004-08-23 23:37:48 +00:00
										 |  |  |             ): | 
					
						
							|  |  |  |             asm = dis_single(line) | 
					
						
							|  |  |  |             self.assert_(elem in asm) | 
					
						
							|  |  |  |             self.assert_('BUILD_TUPLE' not in asm) | 
					
						
							|  |  |  |             self.assert_('UNPACK_TUPLE' not in asm) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-09-22 18:44:21 +00:00
										 |  |  |     def test_folding_of_tuples_of_constants(self): | 
					
						
							|  |  |  |         for line, elem in ( | 
					
						
							| 
									
										
										
										
											2004-11-02 04:20:10 +00:00
										 |  |  |             ('a = 1,2,3', '((1, 2, 3))'), | 
					
						
							|  |  |  |             ('("a","b","c")', "(('a', 'b', 'c'))"), | 
					
						
							|  |  |  |             ('a,b,c = 1,2,3', '((1, 2, 3))'), | 
					
						
							|  |  |  |             ('(None, 1, None)', '((None, 1, None))'), | 
					
						
							|  |  |  |             ('((1, 2), 3, 4)', '(((1, 2), 3, 4))'), | 
					
						
							| 
									
										
										
										
											2004-09-22 18:44:21 +00:00
										 |  |  |             ): | 
					
						
							|  |  |  |             asm = dis_single(line) | 
					
						
							|  |  |  |             self.assert_(elem in asm) | 
					
						
							|  |  |  |             self.assert_('BUILD_TUPLE' not in asm) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-10-26 08:59:14 +00:00
										 |  |  |         # Bug 1053819:  Tuple of constants misidentified when presented with: | 
					
						
							|  |  |  |         # . . . opcode_with_arg 100   unary_opcode   BUILD_TUPLE 1  . . . | 
					
						
							|  |  |  |         # The following would segfault upon compilation | 
					
						
							|  |  |  |         def crater(): | 
					
						
							|  |  |  |             (~[ | 
					
						
							|  |  |  |                 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, | 
					
						
							|  |  |  |                 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, | 
					
						
							|  |  |  |                 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, | 
					
						
							|  |  |  |                 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, | 
					
						
							|  |  |  |                 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, | 
					
						
							|  |  |  |                 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, | 
					
						
							|  |  |  |                 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, | 
					
						
							|  |  |  |                 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, | 
					
						
							|  |  |  |                 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, | 
					
						
							|  |  |  |                 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, | 
					
						
							|  |  |  |             ],) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-01-02 06:17:33 +00:00
										 |  |  |     def test_folding_of_binops_on_constants(self): | 
					
						
							|  |  |  |         for line, elem in ( | 
					
						
							|  |  |  |             ('a = 2+3+4', '(9)'),                   # chained fold | 
					
						
							|  |  |  |             ('"@"*4', "('@@@@')"),                  # check string ops | 
					
						
							|  |  |  |             ('a="abc" + "def"', "('abcdef')"),      # check string ops | 
					
						
							|  |  |  |             ('a = 3**4', '(81)'),                   # binary power | 
					
						
							|  |  |  |             ('a = 3*4', '(12)'),                    # binary multiply | 
					
						
							|  |  |  |             ('a = 13//4', '(3)'),                   # binary floor divide | 
					
						
							|  |  |  |             ('a = 14%4', '(2)'),                    # binary modulo | 
					
						
							|  |  |  |             ('a = 2+3', '(5)'),                     # binary add | 
					
						
							|  |  |  |             ('a = 13-4', '(9)'),                    # binary subtract | 
					
						
							|  |  |  |             ('a = (12,13)[1]', '(13)'),             # binary subscr | 
					
						
							|  |  |  |             ('a = 13 << 2', '(52)'),                # binary lshift | 
					
						
							|  |  |  |             ('a = 13 >> 2', '(3)'),                 # binary rshift | 
					
						
							|  |  |  |             ('a = 13 & 7', '(5)'),                  # binary and | 
					
						
							|  |  |  |             ('a = 13 ^ 7', '(10)'),                 # binary xor | 
					
						
							|  |  |  |             ('a = 13 | 7', '(15)'),                 # binary or | 
					
						
							|  |  |  |             ): | 
					
						
							|  |  |  |             asm = dis_single(line) | 
					
						
							|  |  |  |             self.assert_(elem in asm, asm) | 
					
						
							|  |  |  |             self.assert_('BINARY_' not in asm) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Verify that unfoldables are skipped | 
					
						
							|  |  |  |         asm = dis_single('a=2+"b"') | 
					
						
							|  |  |  |         self.assert_('(2)' in asm) | 
					
						
							|  |  |  |         self.assert_("('b')" in asm) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-01-26 12:50:05 +00:00
										 |  |  |         # Verify that large sequences do not result from folding | 
					
						
							|  |  |  |         asm = dis_single('a="x"*1000') | 
					
						
							|  |  |  |         self.assert_('(1000)' in asm) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-08-23 23:37:48 +00:00
										 |  |  |     def test_elim_extra_return(self): | 
					
						
							|  |  |  |         # RETURN LOAD_CONST None RETURN  -->  RETURN | 
					
						
							|  |  |  |         def f(x): | 
					
						
							|  |  |  |             return x | 
					
						
							|  |  |  |         asm = disassemble(f) | 
					
						
							|  |  |  |         self.assert_('LOAD_CONST' not in asm) | 
					
						
							|  |  |  |         self.assert_('(None)' not in asm) | 
					
						
							|  |  |  |         self.assertEqual(asm.split().count('RETURN_VALUE'), 1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def test_main(verbose=None): | 
					
						
							|  |  |  |     import sys | 
					
						
							|  |  |  |     from test import test_support | 
					
						
							|  |  |  |     test_classes = (TestTranforms,) | 
					
						
							|  |  |  |     test_support.run_unittest(*test_classes) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # verify reference counting | 
					
						
							|  |  |  |     if verbose and hasattr(sys, "gettotalrefcount"): | 
					
						
							|  |  |  |         import gc | 
					
						
							|  |  |  |         counts = [None] * 5 | 
					
						
							|  |  |  |         for i in xrange(len(counts)): | 
					
						
							|  |  |  |             test_support.run_unittest(*test_classes) | 
					
						
							|  |  |  |             gc.collect() | 
					
						
							|  |  |  |             counts[i] = sys.gettotalrefcount() | 
					
						
							|  |  |  |         print counts | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if __name__ == "__main__": | 
					
						
							|  |  |  |     test_main(verbose=True) |