mirror of
https://github.com/python/cpython.git
synced 2025-10-23 18:03:48 +00:00
This patch by Antoine Pitrou optimizes the bytecode for conditional branches by
merging the following "POP_TOP" instruction into the conditional jump. For
example, the list comprehension "[x for x in l if not x]" produced the
following bytecode:
1 0 BUILD_LIST 0
3 LOAD_FAST 0 (.0)
>> 6 FOR_ITER 23 (to 32)
9 STORE_FAST 1 (x)
12 LOAD_FAST 1 (x)
15 JUMP_IF_TRUE 10 (to 28)
18 POP_TOP
19 LOAD_FAST 1 (x)
22 LIST_APPEND 2
25 JUMP_ABSOLUTE 6
>> 28 POP_TOP
29 JUMP_ABSOLUTE 6
>> 32 RETURN_VALUE
but after the patch it produces the following bytecode:
1 0 BUILD_LIST 0
3 LOAD_FAST 0 (.0)
>> 6 FOR_ITER 18 (to 27)
9 STORE_FAST 1 (x)
12 LOAD_FAST 1 (x)
15 POP_JUMP_IF_TRUE 6
18 LOAD_FAST 1 (x)
21 LIST_APPEND 2
24 JUMP_ABSOLUTE 6
>> 27 RETURN_VALUE
Notice that not only the code is shorter, but the conditional jump
(POP_JUMP_IF_TRUE) jumps right to the start of the loop instead of going through
the JUMP_ABSOLUTE at the end. "continue" statements are helped
similarly.
Furthermore, the old jump opcodes (JUMP_IF_FALSE, JUMP_IF_TRUE) have been
replaced by two new opcodes:
- JUMP_IF_TRUE_OR_POP, which jumps if true and pops otherwise
- JUMP_IF_FALSE_OR_POP, which jumps if false and pops otherwise
258 lines
5.4 KiB
C
258 lines
5.4 KiB
C
static void *opcode_targets[256] = {
|
|
&&_unknown_opcode,
|
|
&&TARGET_POP_TOP,
|
|
&&TARGET_ROT_TWO,
|
|
&&TARGET_ROT_THREE,
|
|
&&TARGET_DUP_TOP,
|
|
&&TARGET_ROT_FOUR,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&TARGET_NOP,
|
|
&&TARGET_UNARY_POSITIVE,
|
|
&&TARGET_UNARY_NEGATIVE,
|
|
&&TARGET_UNARY_NOT,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&TARGET_UNARY_INVERT,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&TARGET_BINARY_POWER,
|
|
&&TARGET_BINARY_MULTIPLY,
|
|
&&_unknown_opcode,
|
|
&&TARGET_BINARY_MODULO,
|
|
&&TARGET_BINARY_ADD,
|
|
&&TARGET_BINARY_SUBTRACT,
|
|
&&TARGET_BINARY_SUBSCR,
|
|
&&TARGET_BINARY_FLOOR_DIVIDE,
|
|
&&TARGET_BINARY_TRUE_DIVIDE,
|
|
&&TARGET_INPLACE_FLOOR_DIVIDE,
|
|
&&TARGET_INPLACE_TRUE_DIVIDE,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&TARGET_STORE_MAP,
|
|
&&TARGET_INPLACE_ADD,
|
|
&&TARGET_INPLACE_SUBTRACT,
|
|
&&TARGET_INPLACE_MULTIPLY,
|
|
&&_unknown_opcode,
|
|
&&TARGET_INPLACE_MODULO,
|
|
&&TARGET_STORE_SUBSCR,
|
|
&&TARGET_DELETE_SUBSCR,
|
|
&&TARGET_BINARY_LSHIFT,
|
|
&&TARGET_BINARY_RSHIFT,
|
|
&&TARGET_BINARY_AND,
|
|
&&TARGET_BINARY_XOR,
|
|
&&TARGET_BINARY_OR,
|
|
&&TARGET_INPLACE_POWER,
|
|
&&TARGET_GET_ITER,
|
|
&&TARGET_STORE_LOCALS,
|
|
&&TARGET_PRINT_EXPR,
|
|
&&TARGET_LOAD_BUILD_CLASS,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&TARGET_INPLACE_LSHIFT,
|
|
&&TARGET_INPLACE_RSHIFT,
|
|
&&TARGET_INPLACE_AND,
|
|
&&TARGET_INPLACE_XOR,
|
|
&&TARGET_INPLACE_OR,
|
|
&&TARGET_BREAK_LOOP,
|
|
&&TARGET_WITH_CLEANUP,
|
|
&&_unknown_opcode,
|
|
&&TARGET_RETURN_VALUE,
|
|
&&TARGET_IMPORT_STAR,
|
|
&&_unknown_opcode,
|
|
&&TARGET_YIELD_VALUE,
|
|
&&TARGET_POP_BLOCK,
|
|
&&TARGET_END_FINALLY,
|
|
&&TARGET_POP_EXCEPT,
|
|
&&TARGET_STORE_NAME,
|
|
&&TARGET_DELETE_NAME,
|
|
&&TARGET_UNPACK_SEQUENCE,
|
|
&&TARGET_FOR_ITER,
|
|
&&TARGET_UNPACK_EX,
|
|
&&TARGET_STORE_ATTR,
|
|
&&TARGET_DELETE_ATTR,
|
|
&&TARGET_STORE_GLOBAL,
|
|
&&TARGET_DELETE_GLOBAL,
|
|
&&TARGET_DUP_TOPX,
|
|
&&TARGET_LOAD_CONST,
|
|
&&TARGET_LOAD_NAME,
|
|
&&TARGET_BUILD_TUPLE,
|
|
&&TARGET_BUILD_LIST,
|
|
&&TARGET_BUILD_SET,
|
|
&&TARGET_BUILD_MAP,
|
|
&&TARGET_LOAD_ATTR,
|
|
&&TARGET_COMPARE_OP,
|
|
&&TARGET_IMPORT_NAME,
|
|
&&TARGET_IMPORT_FROM,
|
|
&&TARGET_JUMP_FORWARD,
|
|
&&TARGET_JUMP_IF_FALSE_OR_POP,
|
|
&&TARGET_JUMP_IF_TRUE_OR_POP,
|
|
&&TARGET_JUMP_ABSOLUTE,
|
|
&&TARGET_POP_JUMP_IF_FALSE,
|
|
&&TARGET_POP_JUMP_IF_TRUE,
|
|
&&TARGET_LOAD_GLOBAL,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&TARGET_CONTINUE_LOOP,
|
|
&&TARGET_SETUP_LOOP,
|
|
&&TARGET_SETUP_EXCEPT,
|
|
&&TARGET_SETUP_FINALLY,
|
|
&&_unknown_opcode,
|
|
&&TARGET_LOAD_FAST,
|
|
&&TARGET_STORE_FAST,
|
|
&&TARGET_DELETE_FAST,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&TARGET_RAISE_VARARGS,
|
|
&&TARGET_CALL_FUNCTION,
|
|
&&TARGET_MAKE_FUNCTION,
|
|
&&TARGET_BUILD_SLICE,
|
|
&&TARGET_MAKE_CLOSURE,
|
|
&&TARGET_LOAD_CLOSURE,
|
|
&&TARGET_LOAD_DEREF,
|
|
&&TARGET_STORE_DEREF,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&TARGET_CALL_FUNCTION_VAR,
|
|
&&TARGET_CALL_FUNCTION_KW,
|
|
&&TARGET_CALL_FUNCTION_VAR_KW,
|
|
&&TARGET_EXTENDED_ARG,
|
|
&&_unknown_opcode,
|
|
&&TARGET_LIST_APPEND,
|
|
&&TARGET_SET_ADD,
|
|
&&TARGET_MAP_ADD,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode,
|
|
&&_unknown_opcode
|
|
};
|