112 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			C++
		
	
	
		
		
			
		
	
	
			112 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			C++
		
	
	
| 
								 | 
							
								/*
							 | 
						||
| 
								 | 
							
								    tests/cross_module_gil_utils.cpp -- tools for acquiring GIL from a different module
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    Copyright (c) 2019 Google LLC
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    All rights reserved. Use of this source code is governed by a
							 | 
						||
| 
								 | 
							
								    BSD-style license that can be found in the LICENSE file.
							 | 
						||
| 
								 | 
							
								*/
							 | 
						||
| 
								 | 
							
								#if defined(PYBIND11_INTERNALS_VERSION)
							 | 
						||
| 
								 | 
							
								#    undef PYBIND11_INTERNALS_VERSION
							 | 
						||
| 
								 | 
							
								#endif
							 | 
						||
| 
								 | 
							
								#define PYBIND11_INTERNALS_VERSION 21814642 // Ensure this module has its own `internals` instance.
							 | 
						||
| 
								 | 
							
								#include <pybind11/pybind11.h>
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include <cstdint>
							 | 
						||
| 
								 | 
							
								#include <string>
							 | 
						||
| 
								 | 
							
								#include <thread>
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// This file mimics a DSO that makes pybind11 calls but does not define a
							 | 
						||
| 
								 | 
							
								// PYBIND11_MODULE. The purpose is to test that such a DSO can create a
							 | 
						||
| 
								 | 
							
								// py::gil_scoped_acquire when the running thread is in a GIL-released state.
							 | 
						||
| 
								 | 
							
								//
							 | 
						||
| 
								 | 
							
								// Note that we define a Python module here for convenience, but in general
							 | 
						||
| 
								 | 
							
								// this need not be the case. The typical scenario would be a DSO that implements
							 | 
						||
| 
								 | 
							
								// shared logic used internally by multiple pybind11 modules.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								namespace {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								namespace py = pybind11;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								void gil_acquire() { py::gil_scoped_acquire gil; }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								std::string gil_multi_acquire_release(unsigned bits) {
							 | 
						||
| 
								 | 
							
								    if ((bits & 0x1u) != 0u) {
							 | 
						||
| 
								 | 
							
								        py::gil_scoped_acquire gil;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    if ((bits & 0x2u) != 0u) {
							 | 
						||
| 
								 | 
							
								        py::gil_scoped_release gil;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    if ((bits & 0x4u) != 0u) {
							 | 
						||
| 
								 | 
							
								        py::gil_scoped_acquire gil;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    if ((bits & 0x8u) != 0u) {
							 | 
						||
| 
								 | 
							
								        py::gil_scoped_release gil;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    return PYBIND11_INTERNALS_ID;
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								struct CustomAutoGIL {
							 | 
						||
| 
								 | 
							
								    CustomAutoGIL() : gstate(PyGILState_Ensure()) {}
							 | 
						||
| 
								 | 
							
								    ~CustomAutoGIL() { PyGILState_Release(gstate); }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    PyGILState_STATE gstate;
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								struct CustomAutoNoGIL {
							 | 
						||
| 
								 | 
							
								    CustomAutoNoGIL() : save(PyEval_SaveThread()) {}
							 | 
						||
| 
								 | 
							
								    ~CustomAutoNoGIL() { PyEval_RestoreThread(save); }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    PyThreadState *save;
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								template <typename Acquire, typename Release>
							 | 
						||
| 
								 | 
							
								void gil_acquire_inner() {
							 | 
						||
| 
								 | 
							
								    Acquire acquire_outer;
							 | 
						||
| 
								 | 
							
								    Acquire acquire_inner;
							 | 
						||
| 
								 | 
							
								    Release release;
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								template <typename Acquire, typename Release>
							 | 
						||
| 
								 | 
							
								void gil_acquire_nested() {
							 | 
						||
| 
								 | 
							
								    Acquire acquire_outer;
							 | 
						||
| 
								 | 
							
								    Acquire acquire_inner;
							 | 
						||
| 
								 | 
							
								    Release release;
							 | 
						||
| 
								 | 
							
								    auto thread = std::thread(&gil_acquire_inner<Acquire, Release>);
							 | 
						||
| 
								 | 
							
								    thread.join();
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								constexpr char kModuleName[] = "cross_module_gil_utils";
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								struct PyModuleDef moduledef = {
							 | 
						||
| 
								 | 
							
								    PyModuleDef_HEAD_INIT, kModuleName, nullptr, 0, nullptr, nullptr, nullptr, nullptr, nullptr};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								} // namespace
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#define ADD_FUNCTION(Name, ...)                                                                   \
							 | 
						||
| 
								 | 
							
								    PyModule_AddObject(m, Name, PyLong_FromVoidPtr(reinterpret_cast<void *>(&__VA_ARGS__)));
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								extern "C" PYBIND11_EXPORT PyObject *PyInit_cross_module_gil_utils() {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    PyObject *m = PyModule_Create(&moduledef);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if (m != nullptr) {
							 | 
						||
| 
								 | 
							
								        static_assert(sizeof(&gil_acquire) == sizeof(void *),
							 | 
						||
| 
								 | 
							
								                      "Function pointer must have the same size as void*");
							 | 
						||
| 
								 | 
							
								#ifdef Py_GIL_DISABLED
							 | 
						||
| 
								 | 
							
								        PyUnstable_Module_SetGIL(m, Py_MOD_GIL_NOT_USED);
							 | 
						||
| 
								 | 
							
								#endif
							 | 
						||
| 
								 | 
							
								        ADD_FUNCTION("gil_acquire_funcaddr", gil_acquire)
							 | 
						||
| 
								 | 
							
								        ADD_FUNCTION("gil_multi_acquire_release_funcaddr", gil_multi_acquire_release)
							 | 
						||
| 
								 | 
							
								        ADD_FUNCTION("gil_acquire_inner_custom_funcaddr",
							 | 
						||
| 
								 | 
							
								                     gil_acquire_inner<CustomAutoGIL, CustomAutoNoGIL>)
							 | 
						||
| 
								 | 
							
								        ADD_FUNCTION("gil_acquire_nested_custom_funcaddr",
							 | 
						||
| 
								 | 
							
								                     gil_acquire_nested<CustomAutoGIL, CustomAutoNoGIL>)
							 | 
						||
| 
								 | 
							
								        ADD_FUNCTION("gil_acquire_inner_pybind11_funcaddr",
							 | 
						||
| 
								 | 
							
								                     gil_acquire_inner<py::gil_scoped_acquire, py::gil_scoped_release>)
							 | 
						||
| 
								 | 
							
								        ADD_FUNCTION("gil_acquire_nested_pybind11_funcaddr",
							 | 
						||
| 
								 | 
							
								                     gil_acquire_nested<py::gil_scoped_acquire, py::gil_scoped_release>)
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    return m;
							 | 
						||
| 
								 | 
							
								}
							 |