mirror of https://github.com/python/cpython.git
57 lines
1.9 KiB
C
57 lines
1.9 KiB
C
|
#ifndef Py_INTERNAL_OBJECT_H
|
||
|
#define Py_INTERNAL_OBJECT_H
|
||
|
#ifdef __cplusplus
|
||
|
extern "C" {
|
||
|
#endif
|
||
|
|
||
|
#if !defined(Py_BUILD_CORE) && !defined(Py_BUILD_CORE_BUILTIN)
|
||
|
# error "this header requires Py_BUILD_CORE or Py_BUILD_CORE_BUILTIN defined"
|
||
|
#endif
|
||
|
|
||
|
/* Tell the GC to track this object.
|
||
|
*
|
||
|
* NB: While the object is tracked by the collector, it must be safe to call the
|
||
|
* ob_traverse method.
|
||
|
*
|
||
|
* Internal note: _PyRuntime.gc.generation0->_gc_prev doesn't have any bit flags
|
||
|
* because it's not object header. So we don't use _PyGCHead_PREV() and
|
||
|
* _PyGCHead_SET_PREV() for it to avoid unnecessary bitwise operations.
|
||
|
*
|
||
|
* The PyObject_GC_Track() function is the public version of this macro.
|
||
|
*/
|
||
|
#define _PyObject_GC_TRACK(o) do { \
|
||
|
PyGC_Head *g = _Py_AS_GC(o); \
|
||
|
if (g->_gc_next != 0) { \
|
||
|
Py_FatalError("GC object already tracked"); \
|
||
|
} \
|
||
|
assert((g->_gc_prev & _PyGC_PREV_MASK_COLLECTING) == 0); \
|
||
|
PyGC_Head *last = (PyGC_Head*)(_PyRuntime.gc.generation0->_gc_prev); \
|
||
|
_PyGCHead_SET_NEXT(last, g); \
|
||
|
_PyGCHead_SET_PREV(g, last); \
|
||
|
_PyGCHead_SET_NEXT(g, _PyRuntime.gc.generation0); \
|
||
|
_PyRuntime.gc.generation0->_gc_prev = (uintptr_t)g; \
|
||
|
} while (0);
|
||
|
|
||
|
/* Tell the GC to stop tracking this object.
|
||
|
*
|
||
|
* Internal note: This may be called while GC. So _PyGC_PREV_MASK_COLLECTING must
|
||
|
* be cleared. But _PyGC_PREV_MASK_FINALIZED bit is kept.
|
||
|
*
|
||
|
* The PyObject_GC_UnTrack() function is the public version of this macro.
|
||
|
*/
|
||
|
#define _PyObject_GC_UNTRACK(o) do { \
|
||
|
PyGC_Head *g = _Py_AS_GC(o); \
|
||
|
PyGC_Head *prev = _PyGCHead_PREV(g); \
|
||
|
PyGC_Head *next = _PyGCHead_NEXT(g); \
|
||
|
assert(next != NULL); \
|
||
|
_PyGCHead_SET_NEXT(prev, next); \
|
||
|
_PyGCHead_SET_PREV(next, prev); \
|
||
|
g->_gc_next = 0; \
|
||
|
g->_gc_prev &= _PyGC_PREV_MASK_FINALIZED; \
|
||
|
} while (0);
|
||
|
|
||
|
#ifdef __cplusplus
|
||
|
}
|
||
|
#endif
|
||
|
#endif /* !Py_INTERNAL_OBJECT_H */
|