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

Add option to use the PyGILState API in gil_scoped_acquire (#1276) #1286

Open
wants to merge 1 commit into
base: master
Choose a base branch
from

Conversation

KyleFromKitware
Copy link

No description provided.

@KyleFromKitware KyleFromKitware changed the title Add option to use the PyGILState API in gil_scoped_acquire (#1276) WIP: Add option to use the PyGILState API in gil_scoped_acquire (#1276) Feb 16, 2018
@KyleFromKitware KyleFromKitware changed the title WIP: Add option to use the PyGILState API in gil_scoped_acquire (#1276) Add option to use the PyGILState API in gil_scoped_acquire (#1276) Feb 16, 2018
@jagerman
Copy link
Member

This patch seems rather complicated (going via the global option). It also has the problem that the global setting could get you into trouble: for example, { gil_scoped_acquire gil; f(); } where f() invokes some Python code that loads another pybind module that changes the global option. Or, similarly, the order in which you load modules ends up dictating whether a feature in one module (relying on pybind's advanced gil abilities) works or not. Load A, or B followed by A and A works; load A followed by B and A stops working.

I think a simpler approach would be to entirely isolate the basic Python API approach into its own RAII class. Call them, perhaps, basic_gil_acquire and basic_gil_release, which use the vanilla API.

@KyleFromKitware
Copy link
Author

One of the things I found during testing of this patch is that each shared library or executable gets its own copy of the global options singleton, because it's declared in a header and not exported - different modules have no way of knowing that the others are using pybind11. That said, perhaps you're right that adding new classes would be the better approach, though this would also involve making new PYBIND11_OVERLOAD() macros that use the new classes too.

@jagerman
Copy link
Member

Oh, right, I forgot about the macros. The number of different macros there is already getting unwieldy; doubling them up doesn't sound nice at all. (I'd like to somehow replace all those macros, but I don't yet have an idea of how to go about eliminating them--mainly because, at least right now, the macro needs to be able to return from the calling code).

That said, there's another advantage of having a different class: it ought to be possible to detect a deadlock when mixing implementations (like the one you ran into originally), at least in some cases, and raise an exception instead.

@KyleFromKitware
Copy link
Author

What about just leaving it up to the caller to do the actual return? Example:

void my_class::some_func(int arg) {
  return pybind11::overload(this, "some_func", arg);
}

That's just an example, it's nowhere near complete, and it probably wouldn't actually look like that. It seems to me the issue is more with templated functions being more unwieldy than macros.

@KyleFromKitware
Copy link
Author

We found another case for making a global configuration option: error_already_set. It calls gil_scoped_acquire in the destructor. We were able to redefine our own versions of the PYBIND11_OVERLOAD() macros to use the basic PyGILState_* API, but error_already_set is used extensively throughout pybind, so making a new version of it isn't feasible without simply forking pybind altogether.

@KyleFromKitware
Copy link
Author

Here is a simple example that illustrates the problem with error_already_set:

#include <pybind11/pybind11.h>
#include <pybind11/embed.h>

#include <iostream>

class basic_gil_scoped_acquire
{
public:
  basic_gil_scoped_acquire()
  {
    state = PyGILState_Ensure();
  }

  ~basic_gil_scoped_acquire()
  {
    PyGILState_Release(state);
  }

private:
  PyGILState_STATE state;
};

class basic_gil_scoped_release
{
public:
  basic_gil_scoped_release()
  {
    state = PyEval_SaveThread();
  }

  ~basic_gil_scoped_release()
  {
    PyEval_RestoreThread(state);
  }

private:
  PyThreadState *state;
};

static const char pycode[] =
"raise Exception('This is a test')";

int main(int argc, char **argv)
{
  pybind11::scoped_interpreter interp;

  basic_gil_scoped_release release;

  try
  {
    basic_gil_scoped_acquire acquire;
    pybind11::exec(pycode);
  }
  catch (pybind11::error_already_set &e)
  {
    std::cout << e.what() << std::endl;
  }

  return 0;
}

This will segfault when the error_already_set is destructed because of the above issue.

@scdub
Copy link

scdub commented Feb 4, 2022

@KyleFromKitware is there anything that I can do to help move this PR forward? In embedded Python deployments this continues to prevent working with libraries that use the gil_scoped functions.

@rwgk
Copy link
Collaborator

rwgk commented Feb 4, 2022

Just to log: I looked, and I have the same worries as @jagerman (Feb 17, 2018).

Threading is a can of worms in general. Mixing in the pitfalls of pybind11::options is likely to make it worse.

Another thought, which may be naive: could the proposed basic_gil_acquire become the default (under the old name). And the old implementation moved to a separate RAII type?

It doesn't solve the issue with the OVERRIDE macros.

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

Successfully merging this pull request may close these issues.

None yet

4 participants