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

Reduce gdb deadlocks #105

Open
aantn opened this issue Oct 11, 2021 · 7 comments
Open

Reduce gdb deadlocks #105

aantn opened this issue Oct 11, 2021 · 7 comments

Comments

@aantn
Copy link

aantn commented Oct 11, 2021

When using python-hunter's remote feature, a warning is printed that the process may deadlock. (See here.)

I strongly suspect this can be fixed by setting the gdb flag scheduler-locking off. IDEs like VSCode reliably inject code into Python processes to support debugging existing processes. VSCode's implementation explicitly mentions the importance of setting scheduler-locking off to prevent deadlocks

I unfortunately don't have the time to open a PR for python-hunter at the moment, but I do plan to maintain a fork of pyrasite with this and other fixes. I will update here if I find additional bugs that can cause deadlocks.

@aantn
Copy link
Author

aantn commented Oct 11, 2021

To clarify, this is a different deadlock than the one pydevd deals with as pydevd code uses PyRun_SimpleString whereas python-hunter uses Py_AddPendingCall. But you can see in the Python source code that Py_AddPendingCall acquires a mutex so not using scheduler-locking off could very conceivably lead to a deadlock here.

@ionelmc
Copy link
Owner

ionelmc commented Oct 12, 2021

Well technically it could deadlock - if you manage to attach hunter via gdb at the exact time where a signal would be delivered to the process (signal delivery would use the Py_AddPendingCall routines). It's still better than using PyRun_SimpleString directly tho (that can deadlock way more easier - basically if you run it while something holds the GIL you're in for very unpredictable behavior - including deadlocks). That's why I used Py_AddPendingCall - it's the correct way with regard to the GIL.

I disagree with your assessment that VSCode reliably injects code into Python processes, or any debugger that relies on PyRun_SimpleString directly.

set scheduler-locking off basically allows other threads to run while gdb runs whatever code is injected - I guess we could add that for good measure but I wouldn't delete the warning just because there's one less chance for things to go sideways :-)

Make a PR with the change and lets see how it works.

@aantn
Copy link
Author

aantn commented Oct 12, 2021

Regarding PyRun_SimpleString, assuming you call PyGILState_Ensure first (all the debuggers do) and other threads are not stopped at that point in time, why is a deadlock possible?

@ionelmc
Copy link
Owner

ionelmc commented Oct 13, 2021

How would the GIL get released for a single thread process if you pause said process while the GIL is held somewhere?

@aantn
Copy link
Author

aantn commented Oct 13, 2021

As far as I understand, PyGILState_Ensure is re-entrant so you can successfully call it from the thread that gdb stopped even if that thread already holds the GIL. If another thread holds the GIL, it's OK because scheduler-locking off will allow that thread to keep on running until it releases the GIL.

@ionelmc
Copy link
Owner

ionelmc commented Oct 13, 2021

To be honest I don't know the cpython internals well enough to say it's right or wrong. From what I've tried the Py_AddPendingCall worked really well while PyRun_SimpleString didn't. Maybe all I missed was the scheduler-locking off - who knows. But we can integration test which technique is really the best, there are already some integration tests on the remote with gdb feature - they could be extended to include thread and all sorts of crazy stuff.

@aantn
Copy link
Author

aantn commented Oct 13, 2021

Cool, I'll give the tests a look. I've been meaning to play around with python-hunter in depth for a long time. I hope to have the justification to do so at my day job soon :)

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

2 participants