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

_Py_Dealloc is an escaping call #124483

Open
brandtbucher opened this issue Sep 25, 2024 · 4 comments
Open

_Py_Dealloc is an escaping call #124483

brandtbucher opened this issue Sep 25, 2024 · 4 comments
Assignees
Labels
3.13 bugs and security fixes 3.14 new features, bugs and security fixes interpreter-core (Objects, Python, Grammar, and Parser dirs) type-crash A hard crash of the interpreter, possibly with a core dump

Comments

@brandtbucher
Copy link
Member

brandtbucher commented Sep 25, 2024

Crash report

We like to pretend that _Py_Dealloc (and, by extension, Py_DECREF, Py_XDECREF, Py_CLEAR, PyStackRef_CLOSE, PyStackRef_CLEAR, etc...) isn't an escaping call in tier two. However, it's perfectly capable of invalidating the world. For example, the following code crashes when run under the JIT:

import sys

class Crashy:
    def __del__(self):
        ns = sys._getframe(1).f_locals
        if ns["i"] == 999:
            ns["i"] = None

def crashy():
    for i in range(1000):
        n = Crashy()
        i + i  # Remove guards for i.
        n = None  # Change i.
        i + i  # Crash!

crashy()

Longer term, we're hoping that removing refcounting operations in tier two will help us out here. Fortunately, the _SET_IP and _CHECK_VALIDITY instructions necessary to fix this issue aren't expensive enough to show up on benchmarks when added back (for now).

Linked PRs

@brandtbucher brandtbucher added interpreter-core (Objects, Python, Grammar, and Parser dirs) type-crash A hard crash of the interpreter, possibly with a core dump 3.14 new features, bugs and security fixes labels Sep 25, 2024
@brandtbucher brandtbucher self-assigned this Sep 25, 2024
@brandtbucher brandtbucher added the 3.13 bugs and security fixes label Sep 26, 2024
@brandtbucher
Copy link
Member Author

Doing this correctly is actually pretty tricky:

  • PyStackRef_CLOSE should be an escaping call, but that means that refs need to be closed in the order they occur on the stack, and all input refs must be closed before output refs are created. That's actually quite awkward in practice, since it's common to create outputs before inputs are destroyed.
  • Things that consume references, like _PyTuple_FromStackRefSteal, become escaping only on error. That makes it basically impossible to adjust the stack pointer correctly.

After talking it through with @markshannon, one possible approach to making this work could be:

  • Rather than adjusting the stack level when references are consumed, overwrite their stack slot with PyStackRef_NULL. That way it doesn't matter what order things are consumed in, and the stack level doesn't need to be adjusted more than once per instruction.
  • Although we discourage it in public APIs, _PyTuple_FromStackRefSteal (and functions like it) should only steal references on success. That way, they're non-escaping.

That should give us a good path forward here. Fewer stack adjustments is also very nice for top-of-stack caching.

@markshannon
Copy link
Member

Note: the above example doesn't crash anymore as we've increased the thresholds. Increasing 999 to 9999 produces a crash.

@markshannon
Copy link
Member

Fixing the crash is actually quite straightforward.

All the stuff around keeping the stack consistent is potentially important for the GC, but we don't rely on it at the moment.

@brandtbucher
Copy link
Member Author

Leaving this open because of #128678 (comment).

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
3.13 bugs and security fixes 3.14 new features, bugs and security fixes interpreter-core (Objects, Python, Grammar, and Parser dirs) type-crash A hard crash of the interpreter, possibly with a core dump
Projects
None yet
Development

No branches or pull requests

2 participants