-
-
Notifications
You must be signed in to change notification settings - Fork 31.5k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Use tagged pointers on the stack in the default build. #127705
Comments
Stackrefs on the heap.We use The free-threaded build allows any |
One challenge here is that objects can become immortal dynamically, such as by reference count overflow. That makes relying on bits in the |
That isn't a problem. The implication |
The unsafe assumption is:
static inline void
PyStackRef_CLOSE(_PyStackRef ref)
{
assert(!PyStackRef_IsNull(ref));
if (PyStackRef_IsUncountedMortal(ref)) { // (ref.bits & Py_TAG_BITS) == 0
Py_DECREF_MORTAL(BITS_TO_PTR(ref));
}
} static inline void Py_DECREF_MORTAL(const char *filename, int lineno, PyObject *op)
{
if (op->ob_refcnt <= 0) {
_Py_NegativeRefcount(filename, lineno, op);
}
_Py_DECREF_STAT_INC();
assert(!_Py_IsStaticImmortal(op));
_Py_DECREF_DecRefTotal();
if (--op->ob_refcnt == 0) { // not safe !!!
_Py_Dealloc(op);
}
} |
* Add location information when accessing already closed stackref * Add #def option to track closed stackrefs to provide precise information for use after free and double frees.
You say that The problem is when objects become immortal by overflowing their refcount. These objects have never been declared immortal, just happen to have a reference count that looks like they are immortal. The problem is not the decref alone, but the decref combined with the missing increfs due to an object becoming temporarily immortal. |
Check for immortality in refcount accounting
bigmem buildbots are still failing:
|
I'm looking into it. |
The additional cost of testing the object for immortality in the few cases where we need to distinguish between a reference to an immortal object and a reference with an embedded reference count turns out to be cheap. The tagging scheme:
|
Now, only the tracerefs build is still failing. |
…1198) This is missing `_PyReftracerTrack` calls, see pythongh-131238. Merging as-is for the 3.14.0a6 release.
… _PyReftracerTrack is called
The change commit fd545d7 broke Linux TraceRefs buildbot: https://buildbot.python.org/#/builders/484/builds/6705
|
…ake sure _PyReftracerTrack is called" This broke Py_TRACE_REFS builds. This reverts commit fd545d7.
Fixed by 684a759. |
Currently all references to objects in frameobjects use
_PyStackRef
instead ofPyObject *
.This is necessary for the free-threaded build to support deferred references.
For the default build
_PyStackRef
is just an alias forPyObject *
.We should change
_PyStackRef
to use proper tagged pointers in the default build for two important reasons:My initial implementation is 0.8% slower, although I'd like to get that closer to 0 before merging anything. There is some speedup in the GC due to streamlined immortality checks, and some slowdown due to increased overhead of turning new
PyObject *
references into_PyStackRef
s.This small slowdown will allow us a large speedup (maybe more than 5%) as we can do the following:
LOAD_
instructions in the interpreter.The tagging scheme:
This tagging scheme is chosen as it provides the best performance for the most common operations:
ptr & 1
NULL
is treated as immortal and tagged, this is the same as PyStackRef_CLOSE.Maintaining the invariant that tag
11
is used for all immortal objects is a bit expensive, but can be mitigated by pushing the conversion fromPyObject *
to_PyStackRef
down to a point where it is known whether an object is newly created or not.For newly created objects
PyStackRef_FromPyObjectStealMortal
can be used which performs no immortality check.00
, or01
. This is OK as immortal refcounts have a huge margin of error and the number of possible references to one of these immortal objects is very small.Linked PRs
_PyStackRef
s in the default build. #127875_PyStackRef
s inspired by HPy debug mode #128121The text was updated successfully, but these errors were encountered: