summaryrefslogtreecommitdiffstats
path: root/contrib/tools/python3/Modules/_opcode.c
diff options
context:
space:
mode:
Diffstat (limited to 'contrib/tools/python3/Modules/_opcode.c')
-rw-r--r--contrib/tools/python3/Modules/_opcode.c346
1 files changed, 329 insertions, 17 deletions
diff --git a/contrib/tools/python3/Modules/_opcode.c b/contrib/tools/python3/Modules/_opcode.c
index b70d426fa29..4bf63932d22 100644
--- a/contrib/tools/python3/Modules/_opcode.c
+++ b/contrib/tools/python3/Modules/_opcode.c
@@ -1,6 +1,14 @@
+#ifndef Py_BUILD_CORE_BUILTIN
+# define Py_BUILD_CORE_MODULE 1
+#endif
+
#include "Python.h"
+#include "compile.h"
#include "opcode.h"
-#include "internal/pycore_code.h"
+#include "pycore_code.h"
+#include "pycore_compile.h"
+#include "pycore_intrinsics.h"
+#include "pycore_optimizer.h" // _Py_GetExecutor()
/*[clinic input]
module _opcode
@@ -27,25 +35,16 @@ _opcode_stack_effect_impl(PyObject *module, int opcode, PyObject *oparg,
PyObject *jump)
/*[clinic end generated code: output=64a18f2ead954dbb input=461c9d4a44851898]*/
{
- int effect;
int oparg_int = 0;
int jump_int;
- if (HAS_ARG(opcode)) {
- if (oparg == Py_None) {
- PyErr_SetString(PyExc_ValueError,
- "stack_effect: opcode requires oparg but oparg was not specified");
- return -1;
- }
+
+ if (oparg != Py_None) {
oparg_int = (int)PyLong_AsLong(oparg);
if ((oparg_int == -1) && PyErr_Occurred()) {
return -1;
}
}
- else if (oparg != Py_None) {
- PyErr_SetString(PyExc_ValueError,
- "stack_effect: opcode does not permit oparg but oparg was specified");
- return -1;
- }
+
if (jump == Py_None) {
jump_int = -1;
}
@@ -60,17 +59,158 @@ _opcode_stack_effect_impl(PyObject *module, int opcode, PyObject *oparg,
"stack_effect: jump must be False, True or None");
return -1;
}
- effect = PyCompile_OpcodeStackEffectWithJump(opcode, oparg_int, jump_int);
+ int effect = PyCompile_OpcodeStackEffectWithJump(opcode, oparg_int, jump_int);
if (effect == PY_INVALID_STACK_EFFECT) {
- PyErr_SetString(PyExc_ValueError,
- "invalid opcode or oparg");
- return -1;
+ PyErr_SetString(PyExc_ValueError, "invalid opcode or oparg");
+ return -1;
}
return effect;
}
/*[clinic input]
+_opcode.is_valid -> bool
+
+ opcode: int
+
+Return True if opcode is valid, False otherwise.
+[clinic start generated code]*/
+
+static int
+_opcode_is_valid_impl(PyObject *module, int opcode)
+/*[clinic end generated code: output=b0d918ea1d073f65 input=fe23e0aa194ddae0]*/
+{
+ return _PyCompile_OpcodeIsValid(opcode);
+}
+
+/*[clinic input]
+
+_opcode.has_arg -> bool
+
+ opcode: int
+
+Return True if the opcode uses its oparg, False otherwise.
+[clinic start generated code]*/
+
+static int
+_opcode_has_arg_impl(PyObject *module, int opcode)
+/*[clinic end generated code: output=7a062d3b2dcc0815 input=93d878ba6361db5f]*/
+{
+ return _PyCompile_OpcodeIsValid(opcode) &&
+ _PyCompile_OpcodeHasArg(opcode);
+}
+
+/*[clinic input]
+
+_opcode.has_const -> bool
+
+ opcode: int
+
+Return True if the opcode accesses a constant, False otherwise.
+[clinic start generated code]*/
+
+static int
+_opcode_has_const_impl(PyObject *module, int opcode)
+/*[clinic end generated code: output=c646d5027c634120 input=a6999e4cf13f9410]*/
+{
+ return _PyCompile_OpcodeIsValid(opcode) &&
+ _PyCompile_OpcodeHasConst(opcode);
+}
+
+/*[clinic input]
+
+_opcode.has_name -> bool
+
+ opcode: int
+
+Return True if the opcode accesses an attribute by name, False otherwise.
+[clinic start generated code]*/
+
+static int
+_opcode_has_name_impl(PyObject *module, int opcode)
+/*[clinic end generated code: output=b49a83555c2fa517 input=448aa5e4bcc947ba]*/
+{
+ return _PyCompile_OpcodeIsValid(opcode) &&
+ _PyCompile_OpcodeHasName(opcode);
+}
+
+/*[clinic input]
+
+_opcode.has_jump -> bool
+
+ opcode: int
+
+Return True if the opcode has a jump target, False otherwise.
+[clinic start generated code]*/
+
+static int
+_opcode_has_jump_impl(PyObject *module, int opcode)
+/*[clinic end generated code: output=e9c583c669f1c46a input=35f711274357a0c3]*/
+{
+ return _PyCompile_OpcodeIsValid(opcode) &&
+ _PyCompile_OpcodeHasJump(opcode);
+
+}
+
+/*[clinic input]
+
+_opcode.has_free -> bool
+
+ opcode: int
+
+Return True if the opcode accesses a free variable, False otherwise.
+
+Note that 'free' in this context refers to names in the current scope
+that are referenced by inner scopes or names in outer scopes that are
+referenced from this scope. It does not include references to global
+or builtin scopes.
+[clinic start generated code]*/
+
+static int
+_opcode_has_free_impl(PyObject *module, int opcode)
+/*[clinic end generated code: output=d81ae4d79af0ee26 input=117dcd5c19c1139b]*/
+{
+ return _PyCompile_OpcodeIsValid(opcode) &&
+ _PyCompile_OpcodeHasFree(opcode);
+
+}
+
+/*[clinic input]
+
+_opcode.has_local -> bool
+
+ opcode: int
+
+Return True if the opcode accesses a local variable, False otherwise.
+[clinic start generated code]*/
+
+static int
+_opcode_has_local_impl(PyObject *module, int opcode)
+/*[clinic end generated code: output=da5a8616b7a5097b input=9a798ee24aaef49d]*/
+{
+ return _PyCompile_OpcodeIsValid(opcode) &&
+ _PyCompile_OpcodeHasLocal(opcode);
+}
+
+/*[clinic input]
+
+_opcode.has_exc -> bool
+
+ opcode: int
+
+Return True if the opcode sets an exception handler, False otherwise.
+[clinic start generated code]*/
+
+static int
+_opcode_has_exc_impl(PyObject *module, int opcode)
+/*[clinic end generated code: output=41b68dff0ec82a52 input=db0e4bdb9bf13fa5]*/
+{
+ return _PyCompile_OpcodeIsValid(opcode) &&
+ _PyCompile_OpcodeHasExc(opcode);
+}
+
+/*[clinic input]
+
_opcode.get_specialization_stats
Return the specialization stats
@@ -87,15 +227,187 @@ _opcode_get_specialization_stats_impl(PyObject *module)
#endif
}
+/*[clinic input]
+
+_opcode.get_nb_ops
+
+Return array of symbols of binary ops.
+
+Indexed by the BINARY_OP oparg value.
+[clinic start generated code]*/
+
+static PyObject *
+_opcode_get_nb_ops_impl(PyObject *module)
+/*[clinic end generated code: output=d997d306cc15426f input=9462fc544c823176]*/
+{
+ PyObject *list = PyList_New(NB_OPARG_LAST + 1);
+ if (list == NULL) {
+ return NULL;
+ }
+#define ADD_NB_OP(NUM, STR) \
+ do { \
+ PyObject *pair = Py_BuildValue("ss", #NUM, STR); \
+ if (pair == NULL) { \
+ Py_DECREF(list); \
+ return NULL; \
+ } \
+ PyList_SET_ITEM(list, (NUM), pair); \
+ } while(0);
+
+ ADD_NB_OP(NB_ADD, "+");
+ ADD_NB_OP(NB_AND, "&");
+ ADD_NB_OP(NB_FLOOR_DIVIDE, "//");
+ ADD_NB_OP(NB_LSHIFT, "<<");
+ ADD_NB_OP(NB_MATRIX_MULTIPLY, "@");
+ ADD_NB_OP(NB_MULTIPLY, "*");
+ ADD_NB_OP(NB_REMAINDER, "%");
+ ADD_NB_OP(NB_OR, "|");
+ ADD_NB_OP(NB_POWER, "**");
+ ADD_NB_OP(NB_RSHIFT, ">>");
+ ADD_NB_OP(NB_SUBTRACT, "-");
+ ADD_NB_OP(NB_TRUE_DIVIDE, "/");
+ ADD_NB_OP(NB_XOR, "^");
+ ADD_NB_OP(NB_INPLACE_ADD, "+=");
+ ADD_NB_OP(NB_INPLACE_AND, "&=");
+ ADD_NB_OP(NB_INPLACE_FLOOR_DIVIDE, "//=");
+ ADD_NB_OP(NB_INPLACE_LSHIFT, "<<=");
+ ADD_NB_OP(NB_INPLACE_MATRIX_MULTIPLY, "@=");
+ ADD_NB_OP(NB_INPLACE_MULTIPLY, "*=");
+ ADD_NB_OP(NB_INPLACE_REMAINDER, "%=");
+ ADD_NB_OP(NB_INPLACE_OR, "|=");
+ ADD_NB_OP(NB_INPLACE_POWER, "**=");
+ ADD_NB_OP(NB_INPLACE_RSHIFT, ">>=");
+ ADD_NB_OP(NB_INPLACE_SUBTRACT, "-=");
+ ADD_NB_OP(NB_INPLACE_TRUE_DIVIDE, "/=");
+ ADD_NB_OP(NB_INPLACE_XOR, "^=");
+
+#undef ADD_NB_OP
+
+ for(int i = 0; i <= NB_OPARG_LAST; i++) {
+ if (PyList_GET_ITEM(list, i) == NULL) {
+ Py_DECREF(list);
+ PyErr_Format(PyExc_ValueError,
+ "Missing initialization for NB_OP %d",
+ i);
+ return NULL;
+ }
+ }
+ return list;
+}
+
+/*[clinic input]
+
+_opcode.get_intrinsic1_descs
+
+Return a list of names of the unary intrinsics.
+[clinic start generated code]*/
+
+static PyObject *
+_opcode_get_intrinsic1_descs_impl(PyObject *module)
+/*[clinic end generated code: output=bd1ddb6b4447d18b input=13b51c712618459b]*/
+{
+ PyObject *list = PyList_New(MAX_INTRINSIC_1 + 1);
+ if (list == NULL) {
+ return NULL;
+ }
+ for (int i=0; i <= MAX_INTRINSIC_1; i++) {
+ PyObject *name = _PyCompile_GetUnaryIntrinsicName(i);
+ if (name == NULL) {
+ Py_DECREF(list);
+ return NULL;
+ }
+ PyList_SET_ITEM(list, i, name);
+ }
+ return list;
+}
+
+
+/*[clinic input]
+
+_opcode.get_intrinsic2_descs
+
+Return a list of names of the binary intrinsics.
+[clinic start generated code]*/
+
+static PyObject *
+_opcode_get_intrinsic2_descs_impl(PyObject *module)
+/*[clinic end generated code: output=40e62bc27584c8a0 input=e83068f249f5471b]*/
+{
+ PyObject *list = PyList_New(MAX_INTRINSIC_2 + 1);
+ if (list == NULL) {
+ return NULL;
+ }
+ for (int i=0; i <= MAX_INTRINSIC_2; i++) {
+ PyObject *name = _PyCompile_GetBinaryIntrinsicName(i);
+ if (name == NULL) {
+ Py_DECREF(list);
+ return NULL;
+ }
+ PyList_SET_ITEM(list, i, name);
+ }
+ return list;
+}
+
+/*[clinic input]
+
+_opcode.get_executor
+
+ code: object
+ offset: int
+
+Return the executor object at offset in code if exists, None otherwise.
+[clinic start generated code]*/
+
+static PyObject *
+_opcode_get_executor_impl(PyObject *module, PyObject *code, int offset)
+/*[clinic end generated code: output=c035c7a47b16648f input=85eff93ea7aac282]*/
+{
+ if (!PyCode_Check(code)) {
+ PyErr_Format(PyExc_TypeError,
+ "expected a code object, not '%.100s'",
+ Py_TYPE(code)->tp_name);
+ return NULL;
+ }
+#ifdef _Py_TIER2
+ return (PyObject *)_Py_GetExecutor((PyCodeObject *)code, offset);
+#else
+ PyErr_Format(PyExc_RuntimeError,
+ "Executors are not available in this build");
+ return NULL;
+#endif
+}
+
static PyMethodDef
opcode_functions[] = {
_OPCODE_STACK_EFFECT_METHODDEF
+ _OPCODE_IS_VALID_METHODDEF
+ _OPCODE_HAS_ARG_METHODDEF
+ _OPCODE_HAS_CONST_METHODDEF
+ _OPCODE_HAS_NAME_METHODDEF
+ _OPCODE_HAS_JUMP_METHODDEF
+ _OPCODE_HAS_FREE_METHODDEF
+ _OPCODE_HAS_LOCAL_METHODDEF
+ _OPCODE_HAS_EXC_METHODDEF
_OPCODE_GET_SPECIALIZATION_STATS_METHODDEF
+ _OPCODE_GET_NB_OPS_METHODDEF
+ _OPCODE_GET_INTRINSIC1_DESCS_METHODDEF
+ _OPCODE_GET_INTRINSIC2_DESCS_METHODDEF
+ _OPCODE_GET_EXECUTOR_METHODDEF
{NULL, NULL, 0, NULL}
};
+int
+_opcode_exec(PyObject *m) {
+ if (PyModule_AddIntMacro(m, ENABLE_SPECIALIZATION) < 0) {
+ return -1;
+ }
+ return 0;
+}
+
static PyModuleDef_Slot module_slots[] = {
+ {Py_mod_exec, _opcode_exec},
{Py_mod_multiple_interpreters, Py_MOD_PER_INTERPRETER_GIL_SUPPORTED},
+ {Py_mod_gil, Py_MOD_GIL_NOT_USED},
{0, NULL}
};