-
-
Notifications
You must be signed in to change notification settings - Fork 30.4k
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
Function version numbers do not obey necessary invariants #117086
Comments
When allocating a shared version number we need to check that not only is the code object the same, but that the globals, builtins and (number of) defaults is the same. Otherwise, the version number could be invalid. We can ensure that everything is correct as follows:
However, we cannot know the version number of the globals when the code object is created, so we will need to initialize it lazily. Here's a scheme for doing that:
|
I'm not sure that we need to worry about the number of defaults. Those can only change by explicit assignment to |
The defaults can be changed via |
Brandt's example modified the function object by assigning a tuple of different length to |
The example I was thinking of is this which is just an assert. If we want to optimize on the exact defaults, not just the number, we will need to add a guard in |
Bug report
Bug description:
It is a necessary invariant of version numbers that an equal version number means that the objects are equivalent. That is, all operations on the two objects will give the same result.
This works for class and dict version numbers as no version number is shared between different objects and any change to the object results in a new version number.
However, we share version numbers for functions and that causes problems. #117051
We have good reasons for sharing version numbers. It allows much better optimization of closures. #98525
Want we want is a scheme that allows us to share version numbers, but preserves the invariant that the same function number means a function that is functionally identical.
Assuming that the function version gets modified whenever the function version is modified, if we want to share versions we only need to ensure the invariant when creating a function.
We do not need to ensure that all functions that are functionally identical share a version, just enough to optimize most common cases.
A function consists of the following fields that we care about when optimizing:
It is
MAKE_FUNCTION
that is responsible for setting the version number., which we get only from the code object, meaning that if the globals, builtins or defaults differ from any other function with the same code object, the version number is invalid.CPython versions tested on:
3.13, CPython main branch
Operating systems tested on:
No response
Linked PRs
The text was updated successfully, but these errors were encountered: