mirror of https://github.com/python/cpython
gh-87092: move CFG related code from compile.c to flowgraph.c (#103021)
This commit is contained in:
parent
b0422e140d
commit
80163e17d3
|
@ -33,6 +33,16 @@ extern int _PyAST_Optimize(
|
|||
struct _arena *arena,
|
||||
_PyASTOptimizeState *state);
|
||||
|
||||
/* Utility for a number of growing arrays used in the compiler */
|
||||
int _PyCompile_EnsureArrayLargeEnough(
|
||||
int idx,
|
||||
void **array,
|
||||
int *alloc,
|
||||
int default_alloc,
|
||||
size_t item_size);
|
||||
|
||||
int _PyCompile_ConstCacheMergeOne(PyObject *const_cache, PyObject **obj);
|
||||
|
||||
/* Access compiler internals for unit testing */
|
||||
|
||||
PyAPI_FUNC(PyObject*) _PyCompile_CodeGen(
|
||||
|
|
|
@ -0,0 +1,117 @@
|
|||
#ifndef Py_INTERNAL_CFG_H
|
||||
#define Py_INTERNAL_CFG_H
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
#ifndef Py_BUILD_CORE
|
||||
# error "this header requires Py_BUILD_CORE define"
|
||||
#endif
|
||||
|
||||
#include "pycore_opcode_utils.h"
|
||||
|
||||
static const _PyCompilerSrcLocation NO_LOCATION = {-1, -1, -1, -1};
|
||||
|
||||
typedef struct {
|
||||
int i_opcode;
|
||||
int i_oparg;
|
||||
_PyCompilerSrcLocation i_loc;
|
||||
struct _PyCfgBasicblock_ *i_target; /* target block (if jump instruction) */
|
||||
struct _PyCfgBasicblock_ *i_except; /* target block when exception is raised */
|
||||
} _PyCfgInstruction;
|
||||
|
||||
typedef struct {
|
||||
int id;
|
||||
} _PyCfgJumpTargetLabel;
|
||||
|
||||
|
||||
typedef struct {
|
||||
struct _PyCfgBasicblock_ *handlers[CO_MAXBLOCKS+1];
|
||||
int depth;
|
||||
} _PyCfgExceptStack;
|
||||
|
||||
typedef struct _PyCfgBasicblock_ {
|
||||
/* Each basicblock in a compilation unit is linked via b_list in the
|
||||
reverse order that the block are allocated. b_list points to the next
|
||||
block, not to be confused with b_next, which is next by control flow. */
|
||||
struct _PyCfgBasicblock_ *b_list;
|
||||
/* The label of this block if it is a jump target, -1 otherwise */
|
||||
_PyCfgJumpTargetLabel b_label;
|
||||
/* Exception stack at start of block, used by assembler to create the exception handling table */
|
||||
_PyCfgExceptStack *b_exceptstack;
|
||||
/* pointer to an array of instructions, initially NULL */
|
||||
_PyCfgInstruction *b_instr;
|
||||
/* If b_next is non-NULL, it is a pointer to the next
|
||||
block reached by normal control flow. */
|
||||
struct _PyCfgBasicblock_ *b_next;
|
||||
/* number of instructions used */
|
||||
int b_iused;
|
||||
/* length of instruction array (b_instr) */
|
||||
int b_ialloc;
|
||||
/* Used by add_checks_for_loads_of_unknown_variables */
|
||||
uint64_t b_unsafe_locals_mask;
|
||||
/* Number of predecessors that a block has. */
|
||||
int b_predecessors;
|
||||
/* depth of stack upon entry of block, computed by stackdepth() */
|
||||
int b_startdepth;
|
||||
/* instruction offset for block, computed by assemble_jump_offsets() */
|
||||
int b_offset;
|
||||
/* Basic block is an exception handler that preserves lasti */
|
||||
unsigned b_preserve_lasti : 1;
|
||||
/* Used by compiler passes to mark whether they have visited a basic block. */
|
||||
unsigned b_visited : 1;
|
||||
/* b_except_handler is used by the cold-detection algorithm to mark exception targets */
|
||||
unsigned b_except_handler : 1;
|
||||
/* b_cold is true if this block is not perf critical (like an exception handler) */
|
||||
unsigned b_cold : 1;
|
||||
/* b_warm is used by the cold-detection algorithm to mark blocks which are definitely not cold */
|
||||
unsigned b_warm : 1;
|
||||
} _PyCfgBasicblock;
|
||||
|
||||
int _PyBasicblock_InsertInstruction(_PyCfgBasicblock *block, int pos, _PyCfgInstruction *instr);
|
||||
|
||||
typedef struct cfg_builder_ {
|
||||
/* The entryblock, at which control flow begins. All blocks of the
|
||||
CFG are reachable through the b_next links */
|
||||
_PyCfgBasicblock *g_entryblock;
|
||||
/* Pointer to the most recently allocated block. By following
|
||||
b_list links, you can reach all allocated blocks. */
|
||||
_PyCfgBasicblock *g_block_list;
|
||||
/* pointer to the block currently being constructed */
|
||||
_PyCfgBasicblock *g_curblock;
|
||||
/* label for the next instruction to be placed */
|
||||
_PyCfgJumpTargetLabel g_current_label;
|
||||
} _PyCfgBuilder;
|
||||
|
||||
int _PyCfgBuilder_UseLabel(_PyCfgBuilder *g, _PyCfgJumpTargetLabel lbl);
|
||||
int _PyCfgBuilder_Addop(_PyCfgBuilder *g, int opcode, int oparg, _PyCompilerSrcLocation loc);
|
||||
|
||||
int _PyCfgBuilder_Init(_PyCfgBuilder *g);
|
||||
void _PyCfgBuilder_Fini(_PyCfgBuilder *g);
|
||||
|
||||
_PyCfgInstruction* _PyCfg_BasicblockLastInstr(const _PyCfgBasicblock *b);
|
||||
int _PyCfg_OptimizeCodeUnit(_PyCfgBuilder *g, PyObject *consts, PyObject *const_cache,
|
||||
int code_flags, int nlocals, int nparams);
|
||||
int _PyCfg_Stackdepth(_PyCfgBasicblock *entryblock, int code_flags);
|
||||
void _PyCfg_ConvertExceptionHandlersToNops(_PyCfgBasicblock *entryblock);
|
||||
int _PyCfg_ResolveLineNumbers(_PyCfgBuilder *g, int firstlineno);
|
||||
int _PyCfg_ResolveJumps(_PyCfgBuilder *g);
|
||||
int _PyCfg_InstrSize(_PyCfgInstruction *instruction);
|
||||
|
||||
|
||||
static inline int
|
||||
basicblock_nofallthrough(const _PyCfgBasicblock *b) {
|
||||
_PyCfgInstruction *last = _PyCfg_BasicblockLastInstr(b);
|
||||
return (last &&
|
||||
(IS_SCOPE_EXIT_OPCODE(last->i_opcode) ||
|
||||
IS_UNCONDITIONAL_JUMP_OPCODE(last->i_opcode)));
|
||||
}
|
||||
|
||||
#define BB_NO_FALLTHROUGH(B) (basicblock_nofallthrough(B))
|
||||
#define BB_HAS_FALLTHROUGH(B) (!basicblock_nofallthrough(B))
|
||||
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
||||
#endif /* !Py_INTERNAL_CFG_H */
|
|
@ -12,12 +12,16 @@ extern "C" {
|
|||
|
||||
#include "opcode.h"
|
||||
|
||||
extern const uint32_t _PyOpcode_RelativeJump[9];
|
||||
|
||||
extern const uint32_t _PyOpcode_Jump[9];
|
||||
|
||||
extern const uint8_t _PyOpcode_Caches[256];
|
||||
|
||||
extern const uint8_t _PyOpcode_Deopt[256];
|
||||
|
||||
#ifdef NEED_OPCODE_TABLES
|
||||
static const uint32_t _PyOpcode_RelativeJump[9] = {
|
||||
const uint32_t _PyOpcode_RelativeJump[9] = {
|
||||
0U,
|
||||
0U,
|
||||
536870912U,
|
||||
|
@ -28,7 +32,7 @@ static const uint32_t _PyOpcode_RelativeJump[9] = {
|
|||
0U,
|
||||
48U,
|
||||
};
|
||||
static const uint32_t _PyOpcode_Jump[9] = {
|
||||
const uint32_t _PyOpcode_Jump[9] = {
|
||||
0U,
|
||||
0U,
|
||||
536870912U,
|
||||
|
|
|
@ -0,0 +1,95 @@
|
|||
#ifndef Py_INTERNAL_OPCODE_UTILS_H
|
||||
#define Py_INTERNAL_OPCODE_UTILS_H
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
#ifndef Py_BUILD_CORE
|
||||
# error "this header requires Py_BUILD_CORE define"
|
||||
#endif
|
||||
|
||||
#include "pycore_opcode.h" // _PyOpcode_RelativeJump
|
||||
|
||||
|
||||
#define MAX_REAL_OPCODE 254
|
||||
|
||||
#define IS_WITHIN_OPCODE_RANGE(opcode) \
|
||||
(((opcode) >= 0 && (opcode) <= MAX_REAL_OPCODE) || \
|
||||
IS_PSEUDO_OPCODE(opcode))
|
||||
|
||||
#define IS_JUMP_OPCODE(opcode) \
|
||||
is_bit_set_in_table(_PyOpcode_Jump, opcode)
|
||||
|
||||
#define IS_BLOCK_PUSH_OPCODE(opcode) \
|
||||
((opcode) == SETUP_FINALLY || \
|
||||
(opcode) == SETUP_WITH || \
|
||||
(opcode) == SETUP_CLEANUP)
|
||||
|
||||
#define HAS_TARGET(opcode) \
|
||||
(IS_JUMP_OPCODE(opcode) || IS_BLOCK_PUSH_OPCODE(opcode))
|
||||
|
||||
/* opcodes that must be last in the basicblock */
|
||||
#define IS_TERMINATOR_OPCODE(opcode) \
|
||||
(IS_JUMP_OPCODE(opcode) || IS_SCOPE_EXIT_OPCODE(opcode))
|
||||
|
||||
/* opcodes which are not emitted in codegen stage, only by the assembler */
|
||||
#define IS_ASSEMBLER_OPCODE(opcode) \
|
||||
((opcode) == JUMP_FORWARD || \
|
||||
(opcode) == JUMP_BACKWARD || \
|
||||
(opcode) == JUMP_BACKWARD_NO_INTERRUPT)
|
||||
|
||||
#define IS_BACKWARDS_JUMP_OPCODE(opcode) \
|
||||
((opcode) == JUMP_BACKWARD || \
|
||||
(opcode) == JUMP_BACKWARD_NO_INTERRUPT)
|
||||
|
||||
#define IS_UNCONDITIONAL_JUMP_OPCODE(opcode) \
|
||||
((opcode) == JUMP || \
|
||||
(opcode) == JUMP_NO_INTERRUPT || \
|
||||
(opcode) == JUMP_FORWARD || \
|
||||
(opcode) == JUMP_BACKWARD || \
|
||||
(opcode) == JUMP_BACKWARD_NO_INTERRUPT)
|
||||
|
||||
#define IS_SCOPE_EXIT_OPCODE(opcode) \
|
||||
((opcode) == RETURN_VALUE || \
|
||||
(opcode) == RETURN_CONST || \
|
||||
(opcode) == RAISE_VARARGS || \
|
||||
(opcode) == RERAISE)
|
||||
|
||||
#define IS_SUPERINSTRUCTION_OPCODE(opcode) \
|
||||
((opcode) == LOAD_FAST__LOAD_FAST || \
|
||||
(opcode) == LOAD_FAST__LOAD_CONST || \
|
||||
(opcode) == LOAD_CONST__LOAD_FAST || \
|
||||
(opcode) == STORE_FAST__LOAD_FAST || \
|
||||
(opcode) == STORE_FAST__STORE_FAST)
|
||||
|
||||
|
||||
#define LOG_BITS_PER_INT 5
|
||||
#define MASK_LOW_LOG_BITS 31
|
||||
|
||||
static inline int
|
||||
is_bit_set_in_table(const uint32_t *table, int bitindex) {
|
||||
/* Is the relevant bit set in the relevant word? */
|
||||
/* 512 bits fit into 9 32-bits words.
|
||||
* Word is indexed by (bitindex>>ln(size of int in bits)).
|
||||
* Bit within word is the low bits of bitindex.
|
||||
*/
|
||||
if (bitindex >= 0 && bitindex < 512) {
|
||||
uint32_t word = table[bitindex >> LOG_BITS_PER_INT];
|
||||
return (word >> (bitindex & MASK_LOW_LOG_BITS)) & 1;
|
||||
}
|
||||
else {
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
|
||||
#undef LOG_BITS_PER_INT
|
||||
#undef MASK_LOW_LOG_BITS
|
||||
|
||||
#define IS_RELATIVE_JUMP(opcode) (is_bit_set_in_table(_PyOpcode_RelativeJump, opcode))
|
||||
|
||||
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
||||
#endif /* !Py_INTERNAL_OPCODE_UTILS_H */
|
|
@ -374,6 +374,7 @@ PYTHON_OBJS= \
|
|||
Python/ast_unparse.o \
|
||||
Python/bltinmodule.o \
|
||||
Python/ceval.o \
|
||||
Python/flowgraph.o \
|
||||
Python/codecs.o \
|
||||
Python/compile.o \
|
||||
Python/context.o \
|
||||
|
@ -1702,6 +1703,8 @@ PYTHON_HEADERS= \
|
|||
$(srcdir)/Include/internal/pycore_object_state.h \
|
||||
$(srcdir)/Include/internal/pycore_obmalloc.h \
|
||||
$(srcdir)/Include/internal/pycore_obmalloc_init.h \
|
||||
$(srcdir)/Include/internal/pycore_opcode.h \
|
||||
$(srcdir)/Include/internal/pycore_opcode_utils.h \
|
||||
$(srcdir)/Include/internal/pycore_pathconfig.h \
|
||||
$(srcdir)/Include/internal/pycore_pyarena.h \
|
||||
$(srcdir)/Include/internal/pycore_pyerrors.h \
|
||||
|
|
|
@ -183,6 +183,7 @@
|
|||
<ClCompile Include="..\Python\bltinmodule.c" />
|
||||
<ClCompile Include="..\Python\bootstrap_hash.c" />
|
||||
<ClCompile Include="..\Python\ceval.c" />
|
||||
<ClCompile Include="..\Python\flowgraph.c" />
|
||||
<ClCompile Include="..\Python\codecs.c" />
|
||||
<ClCompile Include="..\Python\compile.c" />
|
||||
<ClCompile Include="..\Python\context.c" />
|
||||
|
|
|
@ -76,6 +76,9 @@
|
|||
<ClCompile Include="..\Python\ceval.c">
|
||||
<Filter>Source Files</Filter>
|
||||
</ClCompile>
|
||||
<ClCompile Include="..\Python\flowgraph.c">
|
||||
<Filter>Source Files</Filter>
|
||||
</ClCompile>
|
||||
<ClCompile Include="..\Objects\classobject.c">
|
||||
<Filter>Source Files</Filter>
|
||||
</ClCompile>
|
||||
|
|
|
@ -205,6 +205,7 @@
|
|||
<ClInclude Include="..\Include\internal\pycore_call.h" />
|
||||
<ClInclude Include="..\Include\internal\pycore_ceval.h" />
|
||||
<ClInclude Include="..\Include\internal\pycore_ceval_state.h" />
|
||||
<ClInclude Include="..\Include\internal\pycore_cfg.h" />
|
||||
<ClInclude Include="..\Include\internal\pycore_code.h" />
|
||||
<ClInclude Include="..\Include\internal\pycore_compile.h" />
|
||||
<ClInclude Include="..\Include\internal\pycore_condvar.h" />
|
||||
|
@ -504,6 +505,7 @@
|
|||
<ClCompile Include="..\Python\bltinmodule.c" />
|
||||
<ClCompile Include="..\Python\bootstrap_hash.c" />
|
||||
<ClCompile Include="..\Python\ceval.c" />
|
||||
<ClCompile Include="..\Python\flowgraph.c" />
|
||||
<ClCompile Include="..\Python\codecs.c" />
|
||||
<ClCompile Include="..\Python\compile.c" />
|
||||
<ClCompile Include="..\Python\context.c" />
|
||||
|
|
|
@ -1106,6 +1106,9 @@
|
|||
<ClCompile Include="..\Python\ceval.c">
|
||||
<Filter>Python</Filter>
|
||||
</ClCompile>
|
||||
<ClCompile Include="..\Python\flowgraph.c">
|
||||
<Filter>Python</Filter>
|
||||
</ClCompile>
|
||||
<ClCompile Include="..\Python\codecs.c">
|
||||
<Filter>Python</Filter>
|
||||
</ClCompile>
|
||||
|
|
2451
Python/compile.c
2451
Python/compile.c
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
|
@ -3,7 +3,7 @@
|
|||
// Python/bytecodes.c
|
||||
// Do not edit!
|
||||
|
||||
#ifndef NEED_OPCODE_TABLES
|
||||
#ifndef NEED_OPCODE_METADATA
|
||||
extern int _PyOpcode_num_popped(int opcode, int oparg, bool jump);
|
||||
#else
|
||||
int
|
||||
|
@ -349,7 +349,7 @@ _PyOpcode_num_popped(int opcode, int oparg, bool jump) {
|
|||
}
|
||||
#endif
|
||||
|
||||
#ifndef NEED_OPCODE_TABLES
|
||||
#ifndef NEED_OPCODE_METADATA
|
||||
extern int _PyOpcode_num_pushed(int opcode, int oparg, bool jump);
|
||||
#else
|
||||
int
|
||||
|
@ -701,7 +701,7 @@ struct opcode_metadata {
|
|||
enum InstructionFormat instr_format;
|
||||
};
|
||||
|
||||
#ifndef NEED_OPCODE_TABLES
|
||||
#ifndef NEED_OPCODE_METADATA
|
||||
extern const struct opcode_metadata _PyOpcode_opcode_metadata[256];
|
||||
#else
|
||||
const struct opcode_metadata _PyOpcode_opcode_metadata[256] = {
|
||||
|
|
|
@ -60,7 +60,7 @@ def write_int_array_from_ops(name, ops, out):
|
|||
bits = 0
|
||||
for op in ops:
|
||||
bits |= 1<<op
|
||||
out.write(f"static const uint32_t {name}[9] = {{\n")
|
||||
out.write(f"const uint32_t {name}[9] = {{\n")
|
||||
for i in range(9):
|
||||
out.write(f" {bits & UINT32_MASK}U,\n")
|
||||
bits >>= 32
|
||||
|
@ -130,6 +130,8 @@ def main(opcode_py, outfile='Include/opcode.h', internaloutfile='Include/interna
|
|||
for name, op in specialized_opmap.items():
|
||||
fobj.write(DEFINE.format(name, op))
|
||||
|
||||
iobj.write("\nextern const uint32_t _PyOpcode_RelativeJump[9];\n")
|
||||
iobj.write("\nextern const uint32_t _PyOpcode_Jump[9];\n")
|
||||
iobj.write("\nextern const uint8_t _PyOpcode_Caches[256];\n")
|
||||
iobj.write("\nextern const uint8_t _PyOpcode_Deopt[256];\n")
|
||||
iobj.write("\n#ifdef NEED_OPCODE_TABLES\n")
|
||||
|
|
|
@ -930,7 +930,7 @@ class Analyzer:
|
|||
direction: str, data: list[tuple[AnyInstruction, str]]
|
||||
) -> None:
|
||||
self.out.emit("")
|
||||
self.out.emit("#ifndef NEED_OPCODE_TABLES")
|
||||
self.out.emit("#ifndef NEED_OPCODE_METADATA")
|
||||
self.out.emit(f"extern int _PyOpcode_num_{direction}(int opcode, int oparg, bool jump);")
|
||||
self.out.emit("#else")
|
||||
self.out.emit("int")
|
||||
|
@ -999,7 +999,7 @@ class Analyzer:
|
|||
self.out.emit("")
|
||||
|
||||
# Write metadata array declaration
|
||||
self.out.emit("#ifndef NEED_OPCODE_TABLES")
|
||||
self.out.emit("#ifndef NEED_OPCODE_METADATA")
|
||||
self.out.emit("extern const struct opcode_metadata _PyOpcode_opcode_metadata[256];")
|
||||
self.out.emit("#else")
|
||||
self.out.emit("const struct opcode_metadata _PyOpcode_opcode_metadata[256] = {")
|
||||
|
|
Loading…
Reference in New Issue