mirror of
				https://github.com/python/cpython.git
				synced 2025-10-26 03:04:41 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			656 lines
		
	
	
	
		
			19 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			656 lines
		
	
	
	
		
			19 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #ifndef Py_INTERNAL_STACKREF_H
 | |
| #define Py_INTERNAL_STACKREF_H
 | |
| #ifdef __cplusplus
 | |
| extern "C" {
 | |
| #endif
 | |
| 
 | |
| // Define this to get precise tracking of stackrefs.
 | |
| // #define Py_STACKREF_DEBUG 1
 | |
| 
 | |
| // Define this to get precise tracking of closed stackrefs.
 | |
| // This will use unbounded memory, as it can only grow.
 | |
| // Use this to track double closes in short-lived programs
 | |
| // #define Py_STACKREF_CLOSE_DEBUG 1
 | |
| 
 | |
| #ifndef Py_BUILD_CORE
 | |
| #  error "this header requires Py_BUILD_CORE define"
 | |
| #endif
 | |
| 
 | |
| #include "pycore_object_deferred.h"
 | |
| 
 | |
| #include <stddef.h>
 | |
| #include <stdbool.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!
 | |
| */
 | |
| 
 | |
| 
 | |
| #if !defined(Py_GIL_DISABLED) && defined(Py_STACKREF_DEBUG)
 | |
| 
 | |
| typedef union _PyStackRef {
 | |
|     uint64_t index;
 | |
| } _PyStackRef;
 | |
| 
 | |
| #define Py_TAG_BITS 0
 | |
| 
 | |
| PyAPI_FUNC(PyObject *) _Py_stackref_get_object(_PyStackRef ref);
 | |
| PyAPI_FUNC(PyObject *) _Py_stackref_close(_PyStackRef ref, const char *filename, int linenumber);
 | |
| PyAPI_FUNC(_PyStackRef) _Py_stackref_create(PyObject *obj, const char *filename, int linenumber);
 | |
| PyAPI_FUNC(void) _Py_stackref_record_borrow(_PyStackRef ref, const char *filename, int linenumber);
 | |
| extern void _Py_stackref_associate(PyInterpreterState *interp, PyObject *obj, _PyStackRef ref);
 | |
| 
 | |
| static const _PyStackRef PyStackRef_NULL = { .index = 0 };
 | |
| 
 | |
| #define PyStackRef_None ((_PyStackRef){ .index = 1 } )
 | |
| #define PyStackRef_False ((_PyStackRef){ .index = 2 })
 | |
| #define PyStackRef_True ((_PyStackRef){ .index = 3 })
 | |
| 
 | |
| #define LAST_PREDEFINED_STACKREF_INDEX 3
 | |
| 
 | |
| static inline int
 | |
| PyStackRef_IsNull(_PyStackRef ref)
 | |
| {
 | |
|     return ref.index == 0;
 | |
| }
 | |
| 
 | |
| static inline int
 | |
| PyStackRef_IsTrue(_PyStackRef ref)
 | |
| {
 | |
|     return _Py_stackref_get_object(ref) == Py_True;
 | |
| }
 | |
| 
 | |
| static inline int
 | |
| PyStackRef_IsFalse(_PyStackRef ref)
 | |
| {
 | |
|     return _Py_stackref_get_object(ref) == Py_False;
 | |
| }
 | |
| 
 | |
| static inline int
 | |
| PyStackRef_IsNone(_PyStackRef ref)
 | |
| {
 | |
|     return _Py_stackref_get_object(ref) == Py_None;
 | |
| }
 | |
| 
 | |
| static inline PyObject *
 | |
| _PyStackRef_AsPyObjectBorrow(_PyStackRef ref, const char *filename, int linenumber)
 | |
| {
 | |
|     _Py_stackref_record_borrow(ref, filename, linenumber);
 | |
|     return _Py_stackref_get_object(ref);
 | |
| }
 | |
| 
 | |
| #define PyStackRef_AsPyObjectBorrow(REF) _PyStackRef_AsPyObjectBorrow((REF), __FILE__, __LINE__)
 | |
| 
 | |
| static inline PyObject *
 | |
| _PyStackRef_AsPyObjectSteal(_PyStackRef ref, const char *filename, int linenumber)
 | |
| {
 | |
|     return _Py_stackref_close(ref, filename, linenumber);
 | |
| }
 | |
| #define PyStackRef_AsPyObjectSteal(REF) _PyStackRef_AsPyObjectSteal((REF), __FILE__, __LINE__)
 | |
| 
 | |
| static inline _PyStackRef
 | |
| _PyStackRef_FromPyObjectNew(PyObject *obj, const char *filename, int linenumber)
 | |
| {
 | |
|     Py_INCREF(obj);
 | |
|     return _Py_stackref_create(obj, filename, linenumber);
 | |
| }
 | |
| #define PyStackRef_FromPyObjectNew(obj) _PyStackRef_FromPyObjectNew(_PyObject_CAST(obj), __FILE__, __LINE__)
 | |
| 
 | |
| static inline _PyStackRef
 | |
| _PyStackRef_FromPyObjectSteal(PyObject *obj, const char *filename, int linenumber)
 | |
| {
 | |
|     return _Py_stackref_create(obj, filename, linenumber);
 | |
| }
 | |
| #define PyStackRef_FromPyObjectSteal(obj) _PyStackRef_FromPyObjectSteal(_PyObject_CAST(obj), __FILE__, __LINE__)
 | |
| 
 | |
| static inline _PyStackRef
 | |
| _PyStackRef_FromPyObjectImmortal(PyObject *obj, const char *filename, int linenumber)
 | |
| {
 | |
|     assert(_Py_IsImmortal(obj));
 | |
|     return _Py_stackref_create(obj, filename, linenumber);
 | |
| }
 | |
| #define PyStackRef_FromPyObjectImmortal(obj) _PyStackRef_FromPyObjectImmortal(_PyObject_CAST(obj), __FILE__, __LINE__)
 | |
| 
 | |
| static inline void
 | |
| _PyStackRef_CLOSE(_PyStackRef ref, const char *filename, int linenumber)
 | |
| {
 | |
|     PyObject *obj = _Py_stackref_close(ref, filename, linenumber);
 | |
|     Py_DECREF(obj);
 | |
| }
 | |
| #define PyStackRef_CLOSE(REF) _PyStackRef_CLOSE((REF), __FILE__, __LINE__)
 | |
| 
 | |
| static inline void
 | |
| PyStackRef_XCLOSE(_PyStackRef ref)
 | |
| {
 | |
|     if (PyStackRef_IsNull(ref)) {
 | |
|         return;
 | |
|     }
 | |
|     PyObject *obj = _Py_stackref_close(ref);
 | |
|     Py_DECREF(obj);
 | |
| }
 | |
| 
 | |
| static inline _PyStackRef
 | |
| _PyStackRef_DUP(_PyStackRef ref, const char *filename, int linenumber)
 | |
| {
 | |
|     PyObject *obj = _Py_stackref_get_object(ref);
 | |
|     Py_INCREF(obj);
 | |
|     return _Py_stackref_create(obj, filename, linenumber);
 | |
| }
 | |
| #define PyStackRef_DUP(REF) _PyStackRef_DUP(REF, __FILE__, __LINE__)
 | |
| 
 | |
| extern void PyStackRef_CLOSE_SPECIALIZED(_PyStackRef ref, destructor destruct);
 | |
| 
 | |
| static inline _PyStackRef
 | |
| PyStackRef_MakeHeapSafe(_PyStackRef ref)
 | |
| {
 | |
|     return ref;
 | |
| }
 | |
| 
 | |
| #define PyStackRef_CLEAR(REF) \
 | |
|     do { \
 | |
|         _PyStackRef *_tmp_op_ptr = &(REF); \
 | |
|         _PyStackRef _tmp_old_op = (*_tmp_op_ptr); \
 | |
|         *_tmp_op_ptr = PyStackRef_NULL; \
 | |
|         PyStackRef_XCLOSE(_tmp_old_op); \
 | |
|     } while (0)
 | |
| 
 | |
| static inline _PyStackRef
 | |
| _PyStackRef_FromPyObjectStealMortal(PyObject *obj, const char *filename, int linenumber)
 | |
| {
 | |
|     assert(!_Py_IsImmortal(obj));
 | |
|     return _Py_stackref_create(obj, filename, linenumber);
 | |
| }
 | |
| #define PyStackRef_FromPyObjectStealMortal(obj) _PyStackRef_FromPyObjectStealMortal(_PyObject_CAST(obj), __FILE__, __LINE__)
 | |
| 
 | |
| static inline bool
 | |
| PyStackRef_IsHeapSafe(_PyStackRef ref)
 | |
| {
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| 
 | |
| #else
 | |
| 
 | |
| typedef union _PyStackRef {
 | |
|     uintptr_t bits;
 | |
| } _PyStackRef;
 | |
| 
 | |
| 
 | |
| #ifdef Py_GIL_DISABLED
 | |
| 
 | |
| #define Py_TAG_DEFERRED (1)
 | |
| 
 | |
| #define Py_TAG_PTR      ((uintptr_t)0)
 | |
| #define Py_TAG_BITS     ((uintptr_t)1)
 | |
| 
 | |
| 
 | |
| static const _PyStackRef PyStackRef_NULL = { .bits = Py_TAG_DEFERRED};
 | |
| #define PyStackRef_IsNull(stackref) ((stackref).bits == PyStackRef_NULL.bits)
 | |
| #define PyStackRef_True ((_PyStackRef){.bits = ((uintptr_t)&_Py_TrueStruct) | Py_TAG_DEFERRED })
 | |
| #define PyStackRef_False ((_PyStackRef){.bits = ((uintptr_t)&_Py_FalseStruct) | Py_TAG_DEFERRED })
 | |
| #define PyStackRef_None ((_PyStackRef){.bits = ((uintptr_t)&_Py_NoneStruct) | Py_TAG_DEFERRED })
 | |
| 
 | |
| // Checks that mask out the deferred bit in the free threading build.
 | |
| #define PyStackRef_IsNone(ref) (PyStackRef_AsPyObjectBorrow(ref) == Py_None)
 | |
| #define PyStackRef_IsTrue(ref) (PyStackRef_AsPyObjectBorrow(ref) == Py_True)
 | |
| #define PyStackRef_IsFalse(ref) (PyStackRef_AsPyObjectBorrow(ref) == Py_False)
 | |
| 
 | |
| static inline PyObject *
 | |
| PyStackRef_AsPyObjectBorrow(_PyStackRef stackref)
 | |
| {
 | |
|     PyObject *cleared = ((PyObject *)((stackref).bits & (~Py_TAG_BITS)));
 | |
|     return cleared;
 | |
| }
 | |
| 
 | |
| #define PyStackRef_IsDeferred(ref) (((ref).bits & Py_TAG_BITS) == Py_TAG_DEFERRED)
 | |
| 
 | |
| static inline PyObject *
 | |
| PyStackRef_NotDeferred_AsPyObject(_PyStackRef stackref)
 | |
| {
 | |
|     assert(!PyStackRef_IsDeferred(stackref));
 | |
|     return (PyObject *)stackref.bits;
 | |
| }
 | |
| 
 | |
| static inline PyObject *
 | |
| PyStackRef_AsPyObjectSteal(_PyStackRef stackref)
 | |
| {
 | |
|     assert(!PyStackRef_IsNull(stackref));
 | |
|     if (PyStackRef_IsDeferred(stackref)) {
 | |
|         return Py_NewRef(PyStackRef_AsPyObjectBorrow(stackref));
 | |
|     }
 | |
|     return PyStackRef_AsPyObjectBorrow(stackref);
 | |
| }
 | |
| 
 | |
| static inline _PyStackRef
 | |
| _PyStackRef_FromPyObjectSteal(PyObject *obj)
 | |
| {
 | |
|     assert(obj != NULL);
 | |
|     // Make sure we don't take an already tagged value.
 | |
|     assert(((uintptr_t)obj & Py_TAG_BITS) == 0);
 | |
|     return (_PyStackRef){ .bits = (uintptr_t)obj };
 | |
| }
 | |
| #   define PyStackRef_FromPyObjectSteal(obj) _PyStackRef_FromPyObjectSteal(_PyObject_CAST(obj))
 | |
| 
 | |
| 
 | |
| static inline _PyStackRef
 | |
| PyStackRef_FromPyObjectStealMortal(PyObject *obj)
 | |
| {
 | |
|     assert(obj != NULL);
 | |
|     assert(!_Py_IsImmortal(obj));
 | |
|     // Make sure we don't take an already tagged value.
 | |
|     assert(((uintptr_t)obj & Py_TAG_BITS) == 0);
 | |
|     return (_PyStackRef){ .bits = (uintptr_t)obj };
 | |
| }
 | |
| 
 | |
| static inline _PyStackRef
 | |
| PyStackRef_FromPyObjectNew(PyObject *obj)
 | |
| {
 | |
|     // Make sure we don't take an already tagged value.
 | |
|     assert(((uintptr_t)obj & Py_TAG_BITS) == 0);
 | |
|     assert(obj != NULL);
 | |
|     if (_PyObject_HasDeferredRefcount(obj)) {
 | |
|         return (_PyStackRef){ .bits = (uintptr_t)obj | Py_TAG_DEFERRED };
 | |
|     }
 | |
|     else {
 | |
|         return (_PyStackRef){ .bits = (uintptr_t)(Py_NewRef(obj)) | Py_TAG_PTR };
 | |
|     }
 | |
| }
 | |
| #define PyStackRef_FromPyObjectNew(obj) PyStackRef_FromPyObjectNew(_PyObject_CAST(obj))
 | |
| 
 | |
| static inline _PyStackRef
 | |
| PyStackRef_FromPyObjectImmortal(PyObject *obj)
 | |
| {
 | |
|     // 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 };
 | |
| }
 | |
| #define PyStackRef_FromPyObjectImmortal(obj) PyStackRef_FromPyObjectImmortal(_PyObject_CAST(obj))
 | |
| 
 | |
| #define PyStackRef_CLOSE(REF)                                        \
 | |
|         do {                                                            \
 | |
|             _PyStackRef _close_tmp = (REF);                             \
 | |
|             assert(!PyStackRef_IsNull(_close_tmp));                     \
 | |
|             if (!PyStackRef_IsDeferred(_close_tmp)) {                   \
 | |
|                 Py_DECREF(PyStackRef_AsPyObjectBorrow(_close_tmp));     \
 | |
|             }                                                           \
 | |
|         } while (0)
 | |
| 
 | |
| static inline void
 | |
| PyStackRef_CLOSE_SPECIALIZED(_PyStackRef ref, destructor destruct)
 | |
| {
 | |
|     (void)destruct;
 | |
|     PyStackRef_CLOSE(ref);
 | |
| }
 | |
| 
 | |
| static inline _PyStackRef
 | |
| PyStackRef_DUP(_PyStackRef stackref)
 | |
| {
 | |
|     assert(!PyStackRef_IsNull(stackref));
 | |
|     if (PyStackRef_IsDeferred(stackref)) {
 | |
|         assert(_Py_IsImmortal(PyStackRef_AsPyObjectBorrow(stackref)) ||
 | |
|                _PyObject_HasDeferredRefcount(PyStackRef_AsPyObjectBorrow(stackref))
 | |
|         );
 | |
|         return stackref;
 | |
|     }
 | |
|     Py_INCREF(PyStackRef_AsPyObjectBorrow(stackref));
 | |
|     return stackref;
 | |
| }
 | |
| 
 | |
| static inline bool
 | |
| PyStackRef_IsHeapSafe(_PyStackRef ref)
 | |
| {
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| static inline _PyStackRef
 | |
| PyStackRef_MakeHeapSafe(_PyStackRef ref)
 | |
| {
 | |
|     return ref;
 | |
| }
 | |
| 
 | |
| // Convert a possibly deferred reference to a strong reference.
 | |
| static inline _PyStackRef
 | |
| PyStackRef_AsStrongReference(_PyStackRef stackref)
 | |
| {
 | |
|     return PyStackRef_FromPyObjectSteal(PyStackRef_AsPyObjectSteal(stackref));
 | |
| }
 | |
| 
 | |
| #define PyStackRef_XCLOSE(stackref) \
 | |
|     do {                            \
 | |
|         _PyStackRef _tmp = (stackref); \
 | |
|         if (!PyStackRef_IsNull(_tmp)) { \
 | |
|             PyStackRef_CLOSE(_tmp); \
 | |
|         } \
 | |
|     } while (0);
 | |
| 
 | |
| #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)
 | |
| 
 | |
| #define PyStackRef_FromPyObjectNewMortal PyStackRef_FromPyObjectNew
 | |
| 
 | |
| #else // Py_GIL_DISABLED
 | |
| 
 | |
| // With GIL
 | |
| 
 | |
| /* References to immortal objects always have their tag bit set to Py_TAG_REFCNT
 | |
|  * as they can (must) have their reclamation deferred */
 | |
| 
 | |
| #define Py_TAG_BITS 1
 | |
| #define Py_TAG_REFCNT 1
 | |
| #if _Py_IMMORTAL_FLAGS != Py_TAG_REFCNT
 | |
| #  error "_Py_IMMORTAL_FLAGS != Py_TAG_REFCNT"
 | |
| #endif
 | |
| 
 | |
| #define BITS_TO_PTR(REF) ((PyObject *)((REF).bits))
 | |
| #define BITS_TO_PTR_MASKED(REF) ((PyObject *)(((REF).bits) & (~Py_TAG_BITS)))
 | |
| 
 | |
| #define PyStackRef_NULL_BITS Py_TAG_REFCNT
 | |
| static const _PyStackRef PyStackRef_NULL = { .bits = PyStackRef_NULL_BITS };
 | |
| 
 | |
| #define PyStackRef_IsNull(ref) ((ref).bits == PyStackRef_NULL_BITS)
 | |
| #define PyStackRef_True ((_PyStackRef){.bits = ((uintptr_t)&_Py_TrueStruct) | Py_TAG_REFCNT })
 | |
| #define PyStackRef_False ((_PyStackRef){.bits = ((uintptr_t)&_Py_FalseStruct) | Py_TAG_REFCNT })
 | |
| #define PyStackRef_None ((_PyStackRef){.bits = ((uintptr_t)&_Py_NoneStruct) | Py_TAG_REFCNT })
 | |
| 
 | |
| #define PyStackRef_IsTrue(REF) ((REF).bits == (((uintptr_t)&_Py_TrueStruct) | Py_TAG_REFCNT))
 | |
| #define PyStackRef_IsFalse(REF) ((REF).bits == (((uintptr_t)&_Py_FalseStruct) | Py_TAG_REFCNT))
 | |
| #define PyStackRef_IsNone(REF) ((REF).bits == (((uintptr_t)&_Py_NoneStruct) | Py_TAG_REFCNT))
 | |
| 
 | |
| #ifdef Py_DEBUG
 | |
| 
 | |
| static inline void PyStackRef_CheckValid(_PyStackRef ref) {
 | |
|     assert(ref.bits != 0);
 | |
|     int tag = ref.bits & Py_TAG_BITS;
 | |
|     PyObject *obj = BITS_TO_PTR_MASKED(ref);
 | |
|     switch (tag) {
 | |
|         case 0:
 | |
|             /* Can be immortal if object was made immortal after reference came into existence */
 | |
|             assert(!_Py_IsStaticImmortal(obj));
 | |
|             break;
 | |
|         case Py_TAG_REFCNT:
 | |
|             assert(obj == NULL || _Py_IsImmortal(obj));
 | |
|             break;
 | |
|         default:
 | |
|             assert(0);
 | |
|     }
 | |
| }
 | |
| 
 | |
| #else
 | |
| 
 | |
| #define PyStackRef_CheckValid(REF) ((void)0)
 | |
| 
 | |
| #endif
 | |
| 
 | |
| #ifdef _WIN32
 | |
| #define PyStackRef_RefcountOnObject(REF) (((REF).bits & Py_TAG_BITS) == 0)
 | |
| #define PyStackRef_AsPyObjectBorrow BITS_TO_PTR_MASKED
 | |
| #else
 | |
| /* Does this ref not have an embedded refcount and thus not refer to a declared immmortal object? */
 | |
| static inline int
 | |
| PyStackRef_RefcountOnObject(_PyStackRef ref)
 | |
| {
 | |
|     return (ref.bits & Py_TAG_BITS) == 0;
 | |
| }
 | |
| 
 | |
| static inline PyObject *
 | |
| PyStackRef_AsPyObjectBorrow(_PyStackRef ref)
 | |
| {
 | |
|     return BITS_TO_PTR_MASKED(ref);
 | |
| }
 | |
| #endif
 | |
| 
 | |
| static inline PyObject *
 | |
| PyStackRef_AsPyObjectSteal(_PyStackRef ref)
 | |
| {
 | |
|     if (PyStackRef_RefcountOnObject(ref)) {
 | |
|         return BITS_TO_PTR(ref);
 | |
|     }
 | |
|     else {
 | |
|         return Py_NewRef(BITS_TO_PTR_MASKED(ref));
 | |
|     }
 | |
| }
 | |
| 
 | |
| static inline _PyStackRef
 | |
| PyStackRef_FromPyObjectSteal(PyObject *obj)
 | |
| {
 | |
|     assert(obj != NULL);
 | |
| #if SIZEOF_VOID_P > 4
 | |
|     unsigned int tag = obj->ob_flags & Py_TAG_BITS;
 | |
| #else
 | |
|     unsigned int tag = _Py_IsImmortal(obj) ? Py_TAG_REFCNT : 0;
 | |
| #endif
 | |
|     _PyStackRef ref = ((_PyStackRef){.bits = ((uintptr_t)(obj)) | tag});
 | |
|     PyStackRef_CheckValid(ref);
 | |
|     return ref;
 | |
| }
 | |
| 
 | |
| static inline _PyStackRef
 | |
| PyStackRef_FromPyObjectStealMortal(PyObject *obj)
 | |
| {
 | |
|     assert(obj != NULL);
 | |
|     assert(!_Py_IsImmortal(obj));
 | |
|     _PyStackRef ref = ((_PyStackRef){.bits = ((uintptr_t)(obj)) });
 | |
|     PyStackRef_CheckValid(ref);
 | |
|     return ref;
 | |
| }
 | |
| 
 | |
| // Check if a stackref is exactly the same as another stackref, including the
 | |
| // the deferred bit. This can only be used safely if you know that the deferred
 | |
| // bits of `a` and `b` match.
 | |
| #define PyStackRef_IsExactly(a, b) \
 | |
|     (assert(((a).bits & Py_TAG_BITS) == ((b).bits & Py_TAG_BITS)), (a).bits == (b).bits)
 | |
| 
 | |
| static inline _PyStackRef
 | |
| _PyStackRef_FromPyObjectNew(PyObject *obj)
 | |
| {
 | |
|     assert(obj != NULL);
 | |
|     if (_Py_IsImmortal(obj)) {
 | |
|         return (_PyStackRef){ .bits = ((uintptr_t)obj) | Py_TAG_REFCNT};
 | |
|     }
 | |
|     Py_INCREF_MORTAL(obj);
 | |
|     _PyStackRef ref = (_PyStackRef){ .bits = (uintptr_t)obj };
 | |
|     PyStackRef_CheckValid(ref);
 | |
|     return ref;
 | |
| }
 | |
| #define PyStackRef_FromPyObjectNew(obj) _PyStackRef_FromPyObjectNew(_PyObject_CAST(obj))
 | |
| 
 | |
| static inline _PyStackRef
 | |
| _PyStackRef_FromPyObjectNewMortal(PyObject *obj)
 | |
| {
 | |
|     assert(obj != NULL);
 | |
|     Py_INCREF_MORTAL(obj);
 | |
|     _PyStackRef ref = (_PyStackRef){ .bits = (uintptr_t)obj };
 | |
|     PyStackRef_CheckValid(ref);
 | |
|     return ref;
 | |
| }
 | |
| #define PyStackRef_FromPyObjectNewMortal(obj) _PyStackRef_FromPyObjectNewMortal(_PyObject_CAST(obj))
 | |
| 
 | |
| /* Create a new reference from an object with an embedded reference count */
 | |
| static inline _PyStackRef
 | |
| PyStackRef_FromPyObjectImmortal(PyObject *obj)
 | |
| {
 | |
|     assert(_Py_IsImmortal(obj));
 | |
|     return (_PyStackRef){ .bits = (uintptr_t)obj | Py_TAG_REFCNT};
 | |
| }
 | |
| 
 | |
| /* WARNING: This macro evaluates its argument more than once */
 | |
| #ifdef _WIN32
 | |
| #define PyStackRef_DUP(REF) \
 | |
|     (PyStackRef_RefcountOnObject(REF) ? (Py_INCREF_MORTAL(BITS_TO_PTR(REF)), (REF)) : (REF))
 | |
| #else
 | |
| static inline _PyStackRef
 | |
| PyStackRef_DUP(_PyStackRef ref)
 | |
| {
 | |
|     assert(!PyStackRef_IsNull(ref));
 | |
|     if (PyStackRef_RefcountOnObject(ref)) {
 | |
|         Py_INCREF_MORTAL(BITS_TO_PTR(ref));
 | |
|     }
 | |
|     return ref;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| static inline bool
 | |
| PyStackRef_IsHeapSafe(_PyStackRef ref)
 | |
| {
 | |
|     return (ref.bits & Py_TAG_BITS) == 0 || ref.bits == PyStackRef_NULL_BITS ||  _Py_IsImmortal(BITS_TO_PTR_MASKED(ref));
 | |
| }
 | |
| 
 | |
| static inline _PyStackRef
 | |
| PyStackRef_MakeHeapSafe(_PyStackRef ref)
 | |
| {
 | |
|     if (PyStackRef_IsHeapSafe(ref)) {
 | |
|         return ref;
 | |
|     }
 | |
|     PyObject *obj = BITS_TO_PTR_MASKED(ref);
 | |
|     Py_INCREF(obj);
 | |
|     ref.bits = (uintptr_t)obj;
 | |
|     PyStackRef_CheckValid(ref);
 | |
|     return ref;
 | |
| }
 | |
| 
 | |
| #ifdef _WIN32
 | |
| #define PyStackRef_CLOSE(REF) \
 | |
| do { \
 | |
|     _PyStackRef _temp = (REF); \
 | |
|     if (PyStackRef_RefcountOnObject(_temp)) Py_DECREF_MORTAL(BITS_TO_PTR(_temp)); \
 | |
| } while (0)
 | |
| #else
 | |
| static inline void
 | |
| PyStackRef_CLOSE(_PyStackRef ref)
 | |
| {
 | |
|     assert(!PyStackRef_IsNull(ref));
 | |
|     if (PyStackRef_RefcountOnObject(ref)) {
 | |
|         Py_DECREF_MORTAL(BITS_TO_PTR(ref));
 | |
|     }
 | |
| }
 | |
| #endif
 | |
| 
 | |
| static inline void
 | |
| PyStackRef_CLOSE_SPECIALIZED(_PyStackRef ref, destructor destruct)
 | |
| {
 | |
|     assert(!PyStackRef_IsNull(ref));
 | |
|     if (PyStackRef_RefcountOnObject(ref)) {
 | |
|         Py_DECREF_MORTAL_SPECIALIZED(BITS_TO_PTR(ref), destruct);
 | |
|     }
 | |
| }
 | |
| 
 | |
| #ifdef _WIN32
 | |
| #define PyStackRef_XCLOSE PyStackRef_CLOSE
 | |
| #else
 | |
| static inline void
 | |
| PyStackRef_XCLOSE(_PyStackRef ref)
 | |
| {
 | |
|     assert(ref.bits != 0);
 | |
|     if (PyStackRef_RefcountOnObject(ref)) {
 | |
|         assert(!PyStackRef_IsNull(ref));
 | |
|         Py_DECREF_MORTAL(BITS_TO_PTR(ref));
 | |
|     }
 | |
| }
 | |
| #endif
 | |
| 
 | |
| #define PyStackRef_CLEAR(REF) \
 | |
|     do { \
 | |
|         _PyStackRef *_tmp_op_ptr = &(REF); \
 | |
|         _PyStackRef _tmp_old_op = (*_tmp_op_ptr); \
 | |
|         *_tmp_op_ptr = PyStackRef_NULL; \
 | |
|         PyStackRef_XCLOSE(_tmp_old_op); \
 | |
|     } while (0)
 | |
| 
 | |
| 
 | |
| #endif // Py_GIL_DISABLED
 | |
| 
 | |
| // Note: this is a macro because MSVC (Windows) has trouble inlining it.
 | |
| 
 | |
| #define PyStackRef_Is(a, b) (((a).bits & (~Py_TAG_REFCNT)) == ((b).bits & (~Py_TAG_REFCNT)))
 | |
| 
 | |
| #endif // !defined(Py_GIL_DISABLED) && defined(Py_STACKREF_DEBUG)
 | |
| 
 | |
| #define PyStackRef_TYPE(stackref) Py_TYPE(PyStackRef_AsPyObjectBorrow(stackref))
 | |
| 
 | |
| // Converts a PyStackRef back to a PyObject *, converting the
 | |
| // stackref to a new reference.
 | |
| #define PyStackRef_AsPyObjectNew(stackref) Py_NewRef(PyStackRef_AsPyObjectBorrow(stackref))
 | |
| 
 | |
| // StackRef type checks
 | |
| 
 | |
| static inline bool
 | |
| PyStackRef_GenCheck(_PyStackRef stackref)
 | |
| {
 | |
|     return PyGen_Check(PyStackRef_AsPyObjectBorrow(stackref));
 | |
| }
 | |
| 
 | |
| static inline bool
 | |
| PyStackRef_BoolCheck(_PyStackRef stackref)
 | |
| {
 | |
|     return PyBool_Check(PyStackRef_AsPyObjectBorrow(stackref));
 | |
| }
 | |
| 
 | |
| static inline bool
 | |
| PyStackRef_LongCheck(_PyStackRef stackref)
 | |
| {
 | |
|     return PyLong_Check(PyStackRef_AsPyObjectBorrow(stackref));
 | |
| }
 | |
| 
 | |
| static inline bool
 | |
| PyStackRef_ExceptionInstanceCheck(_PyStackRef stackref)
 | |
| {
 | |
|     return PyExceptionInstance_Check(PyStackRef_AsPyObjectBorrow(stackref));
 | |
| }
 | |
| 
 | |
| static inline bool
 | |
| PyStackRef_CodeCheck(_PyStackRef stackref)
 | |
| {
 | |
|     return PyCode_Check(PyStackRef_AsPyObjectBorrow(stackref));
 | |
| }
 | |
| 
 | |
| static inline bool
 | |
| PyStackRef_FunctionCheck(_PyStackRef stackref)
 | |
| {
 | |
|     return PyFunction_Check(PyStackRef_AsPyObjectBorrow(stackref));
 | |
| }
 | |
| 
 | |
| #ifdef __cplusplus
 | |
| }
 | |
| #endif
 | |
| #endif /* !Py_INTERNAL_STACKREF_H */
 | 
