diff options
author | Mark Shannon <mark@hotpy.org> | 2021-07-15 16:46:55 (GMT) |
---|---|---|
committer | GitHub <noreply@github.com> | 2021-07-15 16:46:55 (GMT) |
commit | a86f7dae0acf918d54086cb85e5a0b0bedeedce7 (patch) | |
tree | 89ea31f995eb5a3899116d59452ea501a19413af /Python/compile.c | |
parent | 5007a4f23c551f8821483d15e76d0d15d5cb9945 (diff) | |
download | cpython-a86f7dae0acf918d54086cb85e5a0b0bedeedce7.zip cpython-a86f7dae0acf918d54086cb85e5a0b0bedeedce7.tar.gz cpython-a86f7dae0acf918d54086cb85e5a0b0bedeedce7.tar.bz2 |
bpo-44626: Merge basic blocks earlier to enable better handling of exit blocks without line numbers (GH-27138)
Diffstat (limited to 'Python/compile.c')
-rw-r--r-- | Python/compile.c | 64 |
1 files changed, 48 insertions, 16 deletions
diff --git a/Python/compile.c b/Python/compile.c index e21c7a5..3a20f6b 100644 --- a/Python/compile.c +++ b/Python/compile.c @@ -7521,10 +7521,11 @@ makecode(struct compiler *c, struct assembler *a, PyObject *constslist, /* For debugging purposes only */ #if 0 static void -dump_instr(const struct instr *i) +dump_instr(struct instr *i) { - const char *jrel = (is_relative_jump(instr)) ? "jrel " : ""; - const char *jabs = (is_jump(instr) && !is_relative_jump(instr))? "jabs " : ""; + const char *jrel = (is_relative_jump(i)) ? "jrel " : ""; + const char *jabs = (is_jump(i) && !is_relative_jump(i))? "jabs " : ""; + char arg[128]; *arg = '\0'; @@ -7561,6 +7562,9 @@ optimize_cfg(struct compiler *c, struct assembler *a, PyObject *consts); static int ensure_exits_have_lineno(struct compiler *c); +static int +extend_block(basicblock *bb); + static int * build_cellfixedoffsets(struct compiler *c) { @@ -7783,6 +7787,12 @@ assemble(struct compiler *c, int addNone) } } + for (basicblock *b = c->u->u_blocks; b != NULL; b = b->b_list) { + if (extend_block(b)) { + return NULL; + } + } + if (ensure_exits_have_lineno(c)) { return NULL; } @@ -7839,6 +7849,7 @@ assemble(struct compiler *c, int addNone) if (consts == NULL) { goto error; } + if (optimize_cfg(c, &a, consts)) { goto error; } @@ -8246,19 +8257,14 @@ optimize_basic_block(struct compiler *c, basicblock *bb, PyObject *consts) goto error; } break; - default: - if (inst->i_target->b_exit && inst->i_target->b_iused <= MAX_COPY_SIZE) { - basicblock *to_copy = inst->i_target; - inst->i_opcode = NOP; - for (i = 0; i < to_copy->b_iused; i++) { - int index = compiler_next_instr(bb); - if (index < 0) { - return -1; - } - bb->b_instr[index] = to_copy->b_instr[i]; - } - bb->b_exit = 1; - } + } + break; + case FOR_ITER: + assert (i == bb->b_iused-1); + if (target->i_opcode == JUMP_FORWARD) { + if (eliminate_jump_to_jump(bb, inst->i_opcode)) { + goto error; + } } break; case ROT_N: @@ -8288,6 +8294,32 @@ error: return -1; } +/* If this block ends with an unconditional jump to an exit block, + * then remove the jump and extend this block with the target. + */ +static int +extend_block(basicblock *bb) { + if (bb->b_iused == 0) { + return 0; + } + struct instr *last = &bb->b_instr[bb->b_iused-1]; + if (last->i_opcode != JUMP_ABSOLUTE && last->i_opcode != JUMP_FORWARD) { + return 0; + } + if (last->i_target->b_exit && last->i_target->b_iused <= MAX_COPY_SIZE) { + basicblock *to_copy = last->i_target; + last->i_opcode = NOP; + for (int i = 0; i < to_copy->b_iused; i++) { + int index = compiler_next_instr(bb); + if (index < 0) { + return -1; + } + bb->b_instr[index] = to_copy->b_instr[i]; + } + bb->b_exit = 1; + } + return 0; +} static void clean_basic_block(basicblock *bb) { |