2023-03-31 14:17:59 -03:00
|
|
|
#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"
|
2023-04-11 07:15:09 -03:00
|
|
|
#include "pycore_compile.h"
|
2023-03-31 14:17:59 -03:00
|
|
|
|
|
|
|
|
|
|
|
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
|
2023-04-11 07:15:09 -03:00
|
|
|
block in this list, not to be confused with b_next, which is next by
|
|
|
|
control flow. */
|
2023-03-31 14:17:59 -03:00
|
|
|
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,
|
2023-04-11 07:15:09 -03:00
|
|
|
int code_flags, int nlocals, int nparams, int firstlineno);
|
2023-03-31 14:17:59 -03:00
|
|
|
int _PyCfg_Stackdepth(_PyCfgBasicblock *entryblock, int code_flags);
|
2023-05-09 14:02:14 -03:00
|
|
|
void _PyCfg_ConvertPseudoOps(_PyCfgBasicblock *entryblock);
|
2023-03-31 14:17:59 -03:00
|
|
|
int _PyCfg_ResolveJumps(_PyCfgBuilder *g);
|
|
|
|
|
|
|
|
|
|
|
|
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))
|
|
|
|
|
2023-04-11 07:15:09 -03:00
|
|
|
PyCodeObject *
|
|
|
|
_PyAssemble_MakeCodeObject(_PyCompile_CodeUnitMetadata *u, PyObject *const_cache,
|
2023-04-29 08:06:04 -03:00
|
|
|
PyObject *consts, int maxdepth, _PyCompile_InstructionSequence *instrs,
|
2023-04-11 07:15:09 -03:00
|
|
|
int nlocalsplus, int code_flags, PyObject *filename);
|
2023-03-31 14:17:59 -03:00
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#endif /* !Py_INTERNAL_CFG_H */
|