| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | #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
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | #include "pycore_object_deferred.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | #include <stddef.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | /*
 | 
					
						
							|  |  |  |   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! | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | typedef union { | 
					
						
							|  |  |  |     uintptr_t bits; | 
					
						
							|  |  |  | } _PyStackRef; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define Py_TAG_DEFERRED (1)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | #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); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | // Gets a PyObject * from a _PyStackRef
 | 
					
						
							|  |  |  | static inline PyObject * | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | PyStackRef_AsPyObjectBorrow(_PyStackRef stackref) | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | #ifdef Py_GIL_DISABLED
 | 
					
						
							|  |  |  |     PyObject *cleared = ((PyObject *)((stackref).bits & (~Py_TAG_BITS))); | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  |     return cleared; | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | #else
 | 
					
						
							|  |  |  |     return ((PyObject *)(stackref).bits); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | // 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); | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  |     return PyStackRef_AsPyObjectBorrow(stackref); | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // 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)); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | // Converts a PyObject * to a PyStackRef, stealing the reference
 | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | static inline _PyStackRef | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | _PyStackRef_FromPyObjectSteal(PyObject *obj) | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | #ifdef Py_GIL_DISABLED
 | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  |     // Make sure we don't take an already tagged value.
 | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  |     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}); | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  |     return ((_PyStackRef){.bits = ((uintptr_t)(obj))}); | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define PyStackRef_FromPyObjectSteal(obj) _PyStackRef_FromPyObjectSteal(_PyObject_CAST(obj))
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | // Converts a PyObject * to a PyStackRef, with a new reference
 | 
					
						
							|  |  |  | static inline _PyStackRef | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | PyStackRef_FromPyObjectNew(PyObject *obj) | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | #ifdef Py_GIL_DISABLED
 | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  |     // Make sure we don't take an already tagged value.
 | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  |     assert(((uintptr_t)obj & Py_TAG_BITS) == 0); | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  |     assert(obj != NULL); | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  |     // TODO (gh-117139): Add deferred objects later.
 | 
					
						
							|  |  |  |     if (_Py_IsImmortal(obj)) { | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  |         return (_PyStackRef){ .bits = (uintptr_t)obj | Py_TAG_DEFERRED }; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  |         return (_PyStackRef){ .bits = (uintptr_t)(Py_NewRef(obj)) | Py_TAG_PTR }; | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  |     return ((_PyStackRef){ .bits = (uintptr_t)(Py_NewRef(obj)) }); | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | #define PyStackRef_FromPyObjectNew(obj) PyStackRef_FromPyObjectNew(_PyObject_CAST(obj))
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Same as PyStackRef_FromPyObjectNew but only for immortal objects.
 | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | static inline _PyStackRef | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | PyStackRef_FromPyObjectImmortal(PyObject *obj) | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | #ifdef Py_GIL_DISABLED
 | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  |     // Make sure we don't take an already tagged value.
 | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  |     assert(((uintptr_t)obj & Py_TAG_BITS) == 0); | 
					
						
							|  |  |  |     assert(obj != NULL); | 
					
						
							|  |  |  |     assert(_Py_IsImmortal(obj)); | 
					
						
							|  |  |  |     return (_PyStackRef){ .bits = (uintptr_t)obj | Py_TAG_DEFERRED }; | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  |     assert(_Py_IsImmortal(obj)); | 
					
						
							|  |  |  |     return ((_PyStackRef){ .bits = (uintptr_t)(obj) }); | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | #define PyStackRef_FromPyObjectImmortal(obj) PyStackRef_FromPyObjectImmortal(_PyObject_CAST(obj))
 | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define PyStackRef_CLEAR(op) \
 | 
					
						
							|  |  |  |     do { \ | 
					
						
							|  |  |  |         _PyStackRef *_tmp_op_ptr = &(op); \ | 
					
						
							|  |  |  |         _PyStackRef _tmp_old_op = (*_tmp_op_ptr); \ | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  |         if (!PyStackRef_IsNull(_tmp_old_op)) { \ | 
					
						
							|  |  |  |             *_tmp_op_ptr = PyStackRef_NULL; \ | 
					
						
							|  |  |  |             PyStackRef_CLOSE(_tmp_old_op); \ | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  |         } \ | 
					
						
							|  |  |  |     } while (0) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline void | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | PyStackRef_CLOSE(_PyStackRef stackref) | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | #ifdef Py_GIL_DISABLED
 | 
					
						
							|  |  |  |     if (PyStackRef_IsDeferred(stackref)) { | 
					
						
							|  |  |  |         // No assert for being immortal or deferred here.
 | 
					
						
							|  |  |  |         // The GC unsets deferred objects right before clearing.
 | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  |     Py_DECREF(PyStackRef_AsPyObjectBorrow(stackref)); | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  |     Py_DECREF(PyStackRef_AsPyObjectBorrow(stackref)); | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | #define PyStackRef_XCLOSE(stackref) \
 | 
					
						
							|  |  |  |     do {                            \ | 
					
						
							|  |  |  |         _PyStackRef _tmp = (stackref); \ | 
					
						
							|  |  |  |         if (!PyStackRef_IsNull(_tmp)) { \ | 
					
						
							|  |  |  |             PyStackRef_CLOSE(_tmp); \ | 
					
						
							|  |  |  |         } \ | 
					
						
							|  |  |  |     } while (0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline _PyStackRef | 
					
						
							|  |  |  | PyStackRef_DUP(_PyStackRef stackref) | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | #ifdef Py_GIL_DISABLED
 | 
					
						
							|  |  |  |     if (PyStackRef_IsDeferred(stackref)) { | 
					
						
							|  |  |  |         assert(PyStackRef_IsNull(stackref) || | 
					
						
							|  |  |  |             _Py_IsImmortal(PyStackRef_AsPyObjectBorrow(stackref))); | 
					
						
							|  |  |  |         return stackref; | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  |     Py_INCREF(PyStackRef_AsPyObjectBorrow(stackref)); | 
					
						
							|  |  |  |     return stackref; | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  |     Py_INCREF(PyStackRef_AsPyObjectBorrow(stackref)); | 
					
						
							|  |  |  |     return stackref; | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline _PyStackRef | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | PyStackRef_XDUP(_PyStackRef stackref) | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  |     if (!PyStackRef_IsNull(stackref)) { | 
					
						
							|  |  |  |         return PyStackRef_DUP(stackref); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return stackref; | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | static inline void | 
					
						
							|  |  |  | _PyObjectStack_FromStackRefStack(PyObject **dst, const _PyStackRef *src, size_t length) | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  |     for (size_t i = 0; i < length; i++) { | 
					
						
							|  |  |  |         dst[i] = PyStackRef_AsPyObjectBorrow(src[i]); | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-01 04:46:13 +08:00
										 |  |  | #ifdef __cplusplus
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | #endif /* !Py_INTERNAL_STACKREF_H */
 |