mirror of
https://github.com/python/cpython.git
synced 2025-11-25 04:34:37 +00:00
gh-101101: Unstable C API tier (PEP 689) (GH-101102)
This commit is contained in:
parent
c41af812c9
commit
6b2d7c0ddb
18 changed files with 358 additions and 29 deletions
115
Modules/_testcapi/code.c
Normal file
115
Modules/_testcapi/code.c
Normal file
|
|
@ -0,0 +1,115 @@
|
|||
#include "parts.h"
|
||||
|
||||
static Py_ssize_t
|
||||
get_code_extra_index(PyInterpreterState* interp) {
|
||||
Py_ssize_t result = -1;
|
||||
|
||||
static const char *key = "_testcapi.frame_evaluation.code_index";
|
||||
|
||||
PyObject *interp_dict = PyInterpreterState_GetDict(interp); // borrowed
|
||||
assert(interp_dict); // real users would handle missing dict... somehow
|
||||
|
||||
PyObject *index_obj = PyDict_GetItemString(interp_dict, key); // borrowed
|
||||
Py_ssize_t index = 0;
|
||||
if (!index_obj) {
|
||||
if (PyErr_Occurred()) {
|
||||
goto finally;
|
||||
}
|
||||
index = PyUnstable_Eval_RequestCodeExtraIndex(NULL);
|
||||
if (index < 0 || PyErr_Occurred()) {
|
||||
goto finally;
|
||||
}
|
||||
index_obj = PyLong_FromSsize_t(index); // strong ref
|
||||
if (!index_obj) {
|
||||
goto finally;
|
||||
}
|
||||
int res = PyDict_SetItemString(interp_dict, key, index_obj);
|
||||
Py_DECREF(index_obj);
|
||||
if (res < 0) {
|
||||
goto finally;
|
||||
}
|
||||
}
|
||||
else {
|
||||
index = PyLong_AsSsize_t(index_obj);
|
||||
if (index == -1 && PyErr_Occurred()) {
|
||||
goto finally;
|
||||
}
|
||||
}
|
||||
|
||||
result = index;
|
||||
finally:
|
||||
return result;
|
||||
}
|
||||
|
||||
static PyObject *
|
||||
test_code_extra(PyObject* self, PyObject *Py_UNUSED(callable))
|
||||
{
|
||||
PyObject *result = NULL;
|
||||
PyObject *test_module = NULL;
|
||||
PyObject *test_func = NULL;
|
||||
|
||||
// Get or initialize interpreter-specific code object storage index
|
||||
PyInterpreterState *interp = PyInterpreterState_Get();
|
||||
if (!interp) {
|
||||
return NULL;
|
||||
}
|
||||
Py_ssize_t code_extra_index = get_code_extra_index(interp);
|
||||
if (PyErr_Occurred()) {
|
||||
goto finally;
|
||||
}
|
||||
|
||||
// Get a function to test with
|
||||
// This can be any Python function. Use `test.test_misc.testfunction`.
|
||||
test_module = PyImport_ImportModule("test.test_capi.test_misc");
|
||||
if (!test_module) {
|
||||
goto finally;
|
||||
}
|
||||
test_func = PyObject_GetAttrString(test_module, "testfunction");
|
||||
if (!test_func) {
|
||||
goto finally;
|
||||
}
|
||||
PyObject *test_func_code = PyFunction_GetCode(test_func); // borrowed
|
||||
if (!test_func_code) {
|
||||
goto finally;
|
||||
}
|
||||
|
||||
// Check the value is initially NULL
|
||||
void *extra;
|
||||
int res = PyUnstable_Code_GetExtra(test_func_code, code_extra_index, &extra);
|
||||
if (res < 0) {
|
||||
goto finally;
|
||||
}
|
||||
assert (extra == NULL);
|
||||
|
||||
// Set another code extra value
|
||||
res = PyUnstable_Code_SetExtra(test_func_code, code_extra_index, (void*)(uintptr_t)77);
|
||||
if (res < 0) {
|
||||
goto finally;
|
||||
}
|
||||
// Assert it was set correctly
|
||||
res = PyUnstable_Code_GetExtra(test_func_code, code_extra_index, &extra);
|
||||
if (res < 0) {
|
||||
goto finally;
|
||||
}
|
||||
assert ((uintptr_t)extra == 77);
|
||||
|
||||
result = Py_NewRef(Py_None);
|
||||
finally:
|
||||
Py_XDECREF(test_module);
|
||||
Py_XDECREF(test_func);
|
||||
return result;
|
||||
}
|
||||
|
||||
static PyMethodDef TestMethods[] = {
|
||||
{"test_code_extra", test_code_extra, METH_NOARGS},
|
||||
{NULL},
|
||||
};
|
||||
|
||||
int
|
||||
_PyTestCapi_Init_Code(PyObject *m) {
|
||||
if (PyModule_AddFunctions(m, TestMethods) < 0) {
|
||||
return -1;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
Loading…
Add table
Add a link
Reference in a new issue