-
-
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
GH-129817: thread safety for tp_flags #130983
Conversation
Use separate structure member in PyHeapTypeObject, `ht_flags`, to store a copy of the type flags. That allows safe toggling of some of the flags after the type has been exposed.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This seems related to the following issue in that some of the operations that change tp_flags
also change type slots:
Type slots are not thread-safe in free-threaded builds (#127266)
Overall, I'm not really sure about this:
- I think these sorts of changes are more likely to introduce problems than fix any problems users will actually encounter
- The abc related flags seem like things that are going to be set during or immediately after type creation. I'm not convinced we should spend effort making that thread-safe
// Non-heap types are immutable and so these flags can only be toggled | ||
// after creation on heap types. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm a bit worried about this assumption.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Do you worry that it's not currently true or that it might be not true in a future version of Python? The fact that non-heap types are immutable seems a pretty important assumption currently. In PyType_Ready()
:
/* Historically, all static types were immutable. See bpo-43908 */
if (!(type->tp_flags & Py_TPFLAGS_HEAPTYPE)) {
type_add_flags(type, Py_TPFLAGS_IMMUTABLETYPE);
/* Static types must be immortal */
_Py_SetImmortalUntracked((PyObject *)type);
}
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm worried that it's not currently true:
- I don't really trust that code comment. Static types and C API extensions predate
Py_TPFLAGS_IMMUTABLETYPE
. I think you can support attribute modification on types if you construct your static type in the right way. - Immutable in that context is talking about attribute modification, not changes to
tp_flags
. I don't think it covers things likecollections.abc.Iterator.register(my_type)
- C extensions can modify type slots and tp_flags even if you can't change attributes from Python code.
Yeah, I think it's related. Doing a stop-the-world pause seems like the only reasonable fix for that. Maybe the same thing could be done for these couple of tp_flags that can be toggled after initial type creation?
I agree that it's pretty unlikely for users to actually encounter problems with this. As you say, the ABC Aside from the flags to support ABC, the only other one is We could fix this by rather than clearing the
Yeah, I think that's the case. In restrospect, we should have did the Another idea is to make |
I think these are reasonable ideas. I'm still not sure they are worth pursuing right now. I'd be okay with deferring work on the potential issues with |
Closing this for now, we can re-visit for 3.15. |
Note: this PR needs a bit more polish before it becomes non-draft. I added Sam and Matt to the reviewers in case they have some initial feedback on this approach.
Use a separate structure member in
PyHeapTypeObject
,ht_flags
, to store a copy of the type flags. That allows safe toggling of some of the flags after the type has been initialized and potentially exposed to other threads.We would prefer to not use an atomic load whenever
tp_flags
is read. That causes a lot of code churn (see gh-130892) and potentially some performance hit on platforms with weak memory ordering. Instead, we would like to use a normal load and only set the flags before the type has been exposed to other threads. That's mostly the case except for the flags listed below.The following type flags cause issues in that they may be toggled after the type is initially created and exposed:
Py_TPFLAGS_SEQUENCE
andPy_TPFLAGS_MAPPING
. Set by the_abc_register
function. It seems there is no way to toggle these flags off once set.Py_TPFLAGS_IS_ABSTRACT
. Toggled by assigning a non-empty set to__abstractmethods__
Py_TPFLAGS_HAVE_VECTORCALL
. Toggled off (no way to turn back on) by assigning__call__
to the type.This PR does the following (only to the free-threaded build, the default build continues to work the same):
PyHeapTypeObject
,ht_flags
. This member only exists in the free-threaded build.tp_flags
value intoht_flags
whentype_ready()
completes.ht_flags
. Use atomic operations to read and write it._PyType_HasFeatureSafe()
to test the above flags. That function uses an atomic load.This approach causes a bit of extra complication since we have to check
Py_TPFLAGS_HEAPTYPE
first to know where to look at the flags. For non-heap types, they are intp_flags
always. This could be avoided by adding an additional member toPyTypeObject
. However, I think thePy_TPFLAGS_HEAPTYPE
check should be cheap enough and putting it inPyHeapTypeObject
ensures that extension types don't get confused by it.Note that since all non-heap types are immutable, it is not possible to toggle type flags for them (at least, CPython itself doesn't do so). Also note that this change can break extensions that manipulate
tp_flags
directly or directly tests them, rather than using functions.tp_flags
with subinterpreters and static types #129817