mirror of
https://github.com/python/cpython.git
synced 2025-10-25 10:44:55 +00:00
This PR sets up tagged pointers for CPython. The general idea is to create a separate struct _PyStackRef for everything on the evaluation stack to store the bits. This forces the C compiler to warn us if we try to cast things or pull things out of the struct directly. Only for free threading: We tag the low bit if something is deferred - that means we skip incref and decref operations on it. This behavior may change in the future if Mark's plans to defer all objects in the interpreter loop pans out. This implies a strict stack reference discipline is required. ALL incref and decref operations on stackrefs must use the stackref variants. It is unsafe to untag something then do normal incref/decref ops on it. The new incref and decref variants are called dup and close. They mimic a "handle" API operating on these stackrefs. Please read Include/internal/pycore_stackref.h for more information! --------- Co-authored-by: Mark Shannon <9448417+markshannon@users.noreply.github.com>
272 lines
7.3 KiB
C
272 lines
7.3 KiB
C
#ifndef Py_INTERNAL_STACKREF_H
|
|
#define Py_INTERNAL_STACKREF_H
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
#ifndef Py_BUILD_CORE
|
|
# error "this header requires Py_BUILD_CORE define"
|
|
#endif
|
|
|
|
#include "pycore_object_deferred.h"
|
|
|
|
#include <stddef.h>
|
|
|
|
/*
|
|
This file introduces a new API for handling references on the stack, called
|
|
_PyStackRef. This API is inspired by HPy.
|
|
|
|
There are 3 main operations, that convert _PyStackRef to PyObject* and
|
|
vice versa:
|
|
|
|
1. Borrow (discouraged)
|
|
2. Steal
|
|
3. New
|
|
|
|
Borrow means that the reference is converted without any change in ownership.
|
|
This is discouraged because it makes verification much harder. It also makes
|
|
unboxed integers harder in the future.
|
|
|
|
Steal means that ownership is transferred to something else. The total
|
|
number of references to the object stays the same.
|
|
|
|
New creates a new reference from the old reference. The old reference
|
|
is still valid.
|
|
|
|
With these 3 API, a strict stack discipline must be maintained. All
|
|
_PyStackRef must be operated on by the new reference operations:
|
|
|
|
1. DUP
|
|
2. CLOSE
|
|
|
|
DUP is roughly equivalent to Py_NewRef. It creates a new reference from an old
|
|
reference. The old reference remains unchanged.
|
|
|
|
CLOSE is roughly equivalent to Py_DECREF. It destroys a reference.
|
|
|
|
Note that it is unsafe to borrow a _PyStackRef and then do normal
|
|
CPython refcounting operations on it!
|
|
*/
|
|
|
|
typedef union {
|
|
uintptr_t bits;
|
|
} _PyStackRef;
|
|
|
|
|
|
#define Py_TAG_DEFERRED (1)
|
|
|
|
#define Py_TAG_PTR (0)
|
|
#define Py_TAG_BITS (1)
|
|
|
|
#ifdef Py_GIL_DISABLED
|
|
static const _PyStackRef PyStackRef_NULL = { .bits = 0 | Py_TAG_DEFERRED};
|
|
#else
|
|
static const _PyStackRef PyStackRef_NULL = { .bits = 0 };
|
|
#endif
|
|
|
|
#define PyStackRef_IsNull(stackref) ((stackref).bits == PyStackRef_NULL.bits)
|
|
|
|
|
|
#ifdef Py_GIL_DISABLED
|
|
# define PyStackRef_True ((_PyStackRef){.bits = ((uintptr_t)&_Py_TrueStruct) | Py_TAG_DEFERRED })
|
|
#else
|
|
# define PyStackRef_True ((_PyStackRef){.bits = ((uintptr_t)&_Py_TrueStruct) })
|
|
#endif
|
|
|
|
#ifdef Py_GIL_DISABLED
|
|
# define PyStackRef_False ((_PyStackRef){.bits = ((uintptr_t)&_Py_FalseStruct) | Py_TAG_DEFERRED })
|
|
#else
|
|
# define PyStackRef_False ((_PyStackRef){.bits = ((uintptr_t)&_Py_FalseStruct) })
|
|
#endif
|
|
|
|
#ifdef Py_GIL_DISABLED
|
|
# define PyStackRef_None ((_PyStackRef){.bits = ((uintptr_t)&_Py_NoneStruct) | Py_TAG_DEFERRED })
|
|
#else
|
|
# define PyStackRef_None ((_PyStackRef){.bits = ((uintptr_t)&_Py_NoneStruct) })
|
|
#endif
|
|
|
|
|
|
static inline int
|
|
PyStackRef_Is(_PyStackRef a, _PyStackRef b) {
|
|
return a.bits == b.bits;
|
|
}
|
|
|
|
static inline int
|
|
PyStackRef_IsDeferred(_PyStackRef ref)
|
|
{
|
|
return ((ref.bits & Py_TAG_BITS) == Py_TAG_DEFERRED);
|
|
}
|
|
|
|
// Gets a PyObject * from a _PyStackRef
|
|
static inline PyObject *
|
|
PyStackRef_AsPyObjectBorrow(_PyStackRef stackref)
|
|
{
|
|
#ifdef Py_GIL_DISABLED
|
|
PyObject *cleared = ((PyObject *)((stackref).bits & (~Py_TAG_BITS)));
|
|
return cleared;
|
|
#else
|
|
return ((PyObject *)(stackref).bits);
|
|
#endif
|
|
}
|
|
|
|
// Converts a PyStackRef back to a PyObject *, stealing the
|
|
// PyStackRef.
|
|
static inline PyObject *
|
|
PyStackRef_AsPyObjectSteal(_PyStackRef stackref)
|
|
{
|
|
#ifdef Py_GIL_DISABLED
|
|
if (!PyStackRef_IsNull(stackref) && PyStackRef_IsDeferred(stackref)) {
|
|
return Py_NewRef(PyStackRef_AsPyObjectBorrow(stackref));
|
|
}
|
|
return PyStackRef_AsPyObjectBorrow(stackref);
|
|
#else
|
|
return PyStackRef_AsPyObjectBorrow(stackref);
|
|
#endif
|
|
}
|
|
|
|
// Converts a PyStackRef back to a PyObject *, converting the
|
|
// stackref to a new reference.
|
|
static inline PyObject *
|
|
PyStackRef_AsPyObjectNew(_PyStackRef stackref)
|
|
{
|
|
return Py_NewRef(PyStackRef_AsPyObjectBorrow(stackref));
|
|
}
|
|
|
|
static inline PyTypeObject *
|
|
PyStackRef_TYPE(_PyStackRef stackref)
|
|
{
|
|
return Py_TYPE(PyStackRef_AsPyObjectBorrow(stackref));
|
|
}
|
|
|
|
// Converts a PyObject * to a PyStackRef, stealing the reference
|
|
static inline _PyStackRef
|
|
_PyStackRef_FromPyObjectSteal(PyObject *obj)
|
|
{
|
|
#ifdef Py_GIL_DISABLED
|
|
// Make sure we don't take an already tagged value.
|
|
assert(((uintptr_t)obj & Py_TAG_BITS) == 0);
|
|
int tag = (obj == NULL || _Py_IsImmortal(obj)) ? (Py_TAG_DEFERRED) : Py_TAG_PTR;
|
|
return ((_PyStackRef){.bits = ((uintptr_t)(obj)) | tag});
|
|
#else
|
|
return ((_PyStackRef){.bits = ((uintptr_t)(obj))});
|
|
#endif
|
|
}
|
|
|
|
#define PyStackRef_FromPyObjectSteal(obj) _PyStackRef_FromPyObjectSteal(_PyObject_CAST(obj))
|
|
|
|
|
|
// Converts a PyObject * to a PyStackRef, with a new reference
|
|
static inline _PyStackRef
|
|
PyStackRef_FromPyObjectNew(PyObject *obj)
|
|
{
|
|
#ifdef Py_GIL_DISABLED
|
|
// Make sure we don't take an already tagged value.
|
|
assert(((uintptr_t)obj & Py_TAG_BITS) == 0);
|
|
assert(obj != NULL);
|
|
// TODO (gh-117139): Add deferred objects later.
|
|
if (_Py_IsImmortal(obj)) {
|
|
return (_PyStackRef){ .bits = (uintptr_t)obj | Py_TAG_DEFERRED };
|
|
}
|
|
else {
|
|
return (_PyStackRef){ .bits = (uintptr_t)(Py_NewRef(obj)) | Py_TAG_PTR };
|
|
}
|
|
#else
|
|
return ((_PyStackRef){ .bits = (uintptr_t)(Py_NewRef(obj)) });
|
|
#endif
|
|
}
|
|
|
|
#define PyStackRef_FromPyObjectNew(obj) PyStackRef_FromPyObjectNew(_PyObject_CAST(obj))
|
|
|
|
// Same as PyStackRef_FromPyObjectNew but only for immortal objects.
|
|
static inline _PyStackRef
|
|
PyStackRef_FromPyObjectImmortal(PyObject *obj)
|
|
{
|
|
#ifdef Py_GIL_DISABLED
|
|
// Make sure we don't take an already tagged value.
|
|
assert(((uintptr_t)obj & Py_TAG_BITS) == 0);
|
|
assert(obj != NULL);
|
|
assert(_Py_IsImmortal(obj));
|
|
return (_PyStackRef){ .bits = (uintptr_t)obj | Py_TAG_DEFERRED };
|
|
#else
|
|
assert(_Py_IsImmortal(obj));
|
|
return ((_PyStackRef){ .bits = (uintptr_t)(obj) });
|
|
#endif
|
|
}
|
|
|
|
#define PyStackRef_FromPyObjectImmortal(obj) PyStackRef_FromPyObjectImmortal(_PyObject_CAST(obj))
|
|
|
|
|
|
#define PyStackRef_CLEAR(op) \
|
|
do { \
|
|
_PyStackRef *_tmp_op_ptr = &(op); \
|
|
_PyStackRef _tmp_old_op = (*_tmp_op_ptr); \
|
|
if (!PyStackRef_IsNull(_tmp_old_op)) { \
|
|
*_tmp_op_ptr = PyStackRef_NULL; \
|
|
PyStackRef_CLOSE(_tmp_old_op); \
|
|
} \
|
|
} while (0)
|
|
|
|
static inline void
|
|
PyStackRef_CLOSE(_PyStackRef stackref)
|
|
{
|
|
#ifdef Py_GIL_DISABLED
|
|
if (PyStackRef_IsDeferred(stackref)) {
|
|
// No assert for being immortal or deferred here.
|
|
// The GC unsets deferred objects right before clearing.
|
|
return;
|
|
}
|
|
Py_DECREF(PyStackRef_AsPyObjectBorrow(stackref));
|
|
#else
|
|
Py_DECREF(PyStackRef_AsPyObjectBorrow(stackref));
|
|
#endif
|
|
}
|
|
|
|
#define PyStackRef_XCLOSE(stackref) \
|
|
do { \
|
|
_PyStackRef _tmp = (stackref); \
|
|
if (!PyStackRef_IsNull(_tmp)) { \
|
|
PyStackRef_CLOSE(_tmp); \
|
|
} \
|
|
} while (0);
|
|
|
|
|
|
static inline _PyStackRef
|
|
PyStackRef_DUP(_PyStackRef stackref)
|
|
{
|
|
#ifdef Py_GIL_DISABLED
|
|
if (PyStackRef_IsDeferred(stackref)) {
|
|
assert(PyStackRef_IsNull(stackref) ||
|
|
_Py_IsImmortal(PyStackRef_AsPyObjectBorrow(stackref)));
|
|
return stackref;
|
|
}
|
|
Py_INCREF(PyStackRef_AsPyObjectBorrow(stackref));
|
|
return stackref;
|
|
#else
|
|
Py_INCREF(PyStackRef_AsPyObjectBorrow(stackref));
|
|
return stackref;
|
|
#endif
|
|
}
|
|
|
|
static inline _PyStackRef
|
|
PyStackRef_XDUP(_PyStackRef stackref)
|
|
{
|
|
if (!PyStackRef_IsNull(stackref)) {
|
|
return PyStackRef_DUP(stackref);
|
|
}
|
|
return stackref;
|
|
}
|
|
|
|
|
|
static inline void
|
|
_PyObjectStack_FromStackRefStack(PyObject **dst, const _PyStackRef *src, size_t length)
|
|
{
|
|
for (size_t i = 0; i < length; i++) {
|
|
dst[i] = PyStackRef_AsPyObjectBorrow(src[i]);
|
|
}
|
|
}
|
|
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
#endif /* !Py_INTERNAL_STACKREF_H */
|