Skip to content
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-130851: Only intern constants of types generated by the compiler #130901

Merged
merged 2 commits into from
Mar 7, 2025

Conversation

colesbury
Copy link
Contributor

@colesbury colesbury commented Mar 5, 2025

The free threading build interns and immortalizes most constants generated by the bytecode compiler. However, users can construct their own code objects with arbitrary constants. Don't intern or immortalize these objects if they are not a type that we know how to handle.

This fixes a refleak failure in the recently added test_code.test_unusual_constants test.

It also fixes a potential crash that could happen when we try to destory a immortalized object on interpreter shutdown.

The free threading build interns and immortalizes most constants
generated by the bytecode compiler. However, users can construct their
own code objects with arbitrary constants. Don't intern or immortalize
these objects if they are not a type that we know how to handle.

This fixes a refleak failure in the recently added
`test_code.test_unusual_constants` test.

It also fixes a potential crash that could happen when we try to
destory a immortalized object on interpreter shutdown.
@colesbury colesbury added skip news 🔨 test-with-refleak-buildbots Test PR w/ refleak buildbots; report in status section labels Mar 5, 2025
@bedevere-bot
Copy link

🤖 New build scheduled with the buildbot fleet by @colesbury for commit 5c3dd1c 🤖

Results will be shown at:

https://buildbot.python.org/all/#/grid?branch=refs%2Fpull%2F130901%2Fmerge

If you want to schedule another build, you need to add the 🔨 test-with-refleak-buildbots label again.

@bedevere-bot bedevere-bot removed the 🔨 test-with-refleak-buildbots Test PR w/ refleak buildbots; report in status section label Mar 5, 2025
@colesbury colesbury changed the title gh-130851: Only intern constants of types generated by compiler gh-130851: Only intern constants of types generated by the compiler Mar 5, 2025
@colesbury colesbury marked this pull request as ready for review March 6, 2025 22:08
@colesbury colesbury requested a review from markshannon as a code owner March 6, 2025 22:08
@colesbury colesbury requested a review from Yhg1s March 6, 2025 22:08
@colesbury
Copy link
Contributor Author

@Yhg1s - The first attempt (#130853) had a few problems that this addresses:

  1. The test_unusual_constants test triggered the refleak checker due to the immortalization
  2. Interning and immortalizing some variables could lead to crashes on exit in:

cpython/Objects/codeobject.c

Lines 2730 to 2733 in a025f27

destroy_key(void *key)
{
_Py_ClearImmortal(key);
}

This changes the behavior so that we now only intern and immortalize code constants that are instances of types that the bytecode compiler generates.

@colesbury colesbury added the needs backport to 3.13 bugs and security fixes label Mar 6, 2025
@colesbury
Copy link
Contributor Author

The slice behavior will need a bit of care when backporting to 3.13, because the bytecode compiler only started generating slices constants in 3.14:

#125064

@@ -241,8 +278,9 @@ intern_constants(PyObject *tuple, int *modified)

// Intern non-string constants in the free-threaded build
_PyThreadStateImpl *tstate = (_PyThreadStateImpl *)_PyThreadState_GET();
if (!_Py_IsImmortal(v) && !PyCode_Check(v) &&
!PyUnicode_CheckExact(v) && !tstate->suppress_co_const_immortalization)
if (!_Py_IsImmortal(v) && !PyUnicode_CheckExact(v) &&
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is the explicit check for unicode necessary, given that should_immortalize_constant also checks that? (For performance, perhaps?)

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The checks were in both places because I wrote should_immortalize_constant recursively for tuples, sets, and slices.

I've changed should_immortalize_constant so that it's no longer recursive and instead interns and immortalizes tuples, set, and slices if we've already immortalized their elements. That should still end up interning and immortalizing the same objects, but avoids the seemingly redundant checks.

@colesbury colesbury merged commit 12db452 into python:main Mar 7, 2025
41 checks passed
@miss-islington-app
Copy link

Thanks @colesbury for the PR 🌮🎉.. I'm working now to backport this PR to: 3.13.
🐍🍒⛏🤖

@colesbury colesbury deleted the gh-130851-should-intern branch March 7, 2025 15:34
@miss-islington-app
Copy link

Sorry, @colesbury, I could not cleanly backport this to 3.13 due to a conflict.
Please backport using cherry_picker on command line.

cherry_picker 12db45211d411583cbe272c7ba6811a811b721ca 3.13

colesbury added a commit to colesbury/cpython that referenced this pull request Mar 7, 2025
…he compiler (pythonGH-130901)

The free-threading build interns and immortalizes most constants
generated by the bytecode compiler. However, users can construct their
own code objects with arbitrary constants. We should not intern or
immortalize these objects if they are not of a type that we know how to
handle.

This change fixes a reference leak failure in the recently added
`test_code.test_unusual_constants` test. It also addresses a potential
crash that could occur when attempting to destroy an immortalized
object during interpreter shutdown.
(cherry picked from commit 12db452)

Co-authored-by: Sam Gross <[email protected]>
@bedevere-app
Copy link

bedevere-app bot commented Mar 7, 2025

GH-130953 is a backport of this pull request to the 3.13 branch.

@bedevere-app bedevere-app bot removed the needs backport to 3.13 bugs and security fixes label Mar 7, 2025
colesbury added a commit that referenced this pull request Mar 8, 2025
…piler (GH-130901) (#130953)

The free-threading build interns and immortalizes most constants
generated by the bytecode compiler. However, users can construct their
own code objects with arbitrary constants. We should not intern or
immortalize these objects if they are not of a type that we know how to
handle.

This change fixes a reference leak failure in the recently added
`test_code.test_unusual_constants` test. It also addresses a potential
crash that could occur when attempting to destroy an immortalized
object during interpreter shutdown.
(cherry picked from commit 12db452)
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants