@@ -1909,7 +1909,6 @@ class class_ : public detail::generic_type {
1909
1909
record.type_align = alignof (conditional_t <has_alias, type_alias, type> &);
1910
1910
record.holder_size = sizeof (holder_type);
1911
1911
record.init_instance = init_instance;
1912
- record.dealloc = dealloc;
1913
1912
1914
1913
// A more fitting name would be uses_unique_ptr_holder.
1915
1914
record.default_holder = detail::is_instantiation<std::unique_ptr, holder_type>::value;
@@ -1934,6 +1933,12 @@ class class_ : public detail::generic_type {
1934
1933
/* Process optional arguments, if any */
1935
1934
process_attributes<Extra...>::init (extra..., &record);
1936
1935
1936
+ if (record.release_gil_before_calling_cpp_dtor ) {
1937
+ record.dealloc = dealloc_release_gil_before_calling_cpp_dtor;
1938
+ } else {
1939
+ record.dealloc = dealloc_without_manipulating_gil;
1940
+ }
1941
+
1937
1942
generic_type::initialize (record);
1938
1943
1939
1944
if (has_alias) {
@@ -2323,15 +2328,14 @@ class class_ : public detail::generic_type {
2323
2328
2324
2329
#endif // PYBIND11_SMART_HOLDER_ENABLED
2325
2330
2326
- // / Deallocates an instance; via holder, if constructed; otherwise via operator delete.
2327
- static void dealloc (detail::value_and_holder &v_h) {
2328
- // We could be deallocating because we are cleaning up after a Python exception.
2329
- // If so, the Python error indicator will be set. We need to clear that before
2330
- // running the destructor, in case the destructor code calls more Python.
2331
- // If we don't, the Python API will exit with an exception, and pybind11 will
2332
- // throw error_already_set from the C++ destructor which is forbidden and triggers
2333
- // std::terminate().
2334
- error_scope scope;
2331
+ // Deallocates an instance; via holder, if constructed; otherwise via operator delete.
2332
+ // NOTE: The Python error indicator needs to cleared BEFORE this function is called.
2333
+ // This is because we could be deallocating while cleaning up after a Python exception.
2334
+ // If the error indicator is not cleared but the C++ destructor code makes Python C API
2335
+ // calls, those calls are likely to generate a new exception, and pybind11 will then
2336
+ // throw `error_already_set` from the C++ destructor. This is forbidden and will
2337
+ // trigger std::terminate().
2338
+ static void dealloc_impl (detail::value_and_holder &v_h) {
2335
2339
if (v_h.holder_constructed ()) {
2336
2340
v_h.holder <holder_type>().~holder_type ();
2337
2341
v_h.set_holder_constructed (false );
@@ -2342,6 +2346,32 @@ class class_ : public detail::generic_type {
2342
2346
v_h.value_ptr () = nullptr ;
2343
2347
}
2344
2348
2349
+ static void dealloc_without_manipulating_gil (detail::value_and_holder &v_h) {
2350
+ error_scope scope;
2351
+ dealloc_impl (v_h);
2352
+ }
2353
+
2354
+ static void dealloc_release_gil_before_calling_cpp_dtor (detail::value_and_holder &v_h) {
2355
+ error_scope scope;
2356
+ // Intentionally not using `gil_scoped_release` because the non-simple
2357
+ // version unconditionally calls `get_internals()`.
2358
+ // `Py_BEGIN_ALLOW_THREADS`, `Py_END_ALLOW_THREADS` cannot be used
2359
+ // because those macros include `{` and `}`.
2360
+ PyThreadState *py_ts = PyEval_SaveThread ();
2361
+ try {
2362
+ dealloc_impl (v_h);
2363
+ } catch (...) {
2364
+ // This code path is expected to be unreachable unless there is a
2365
+ // bug in pybind11 itself.
2366
+ // An alternative would be to mark this function, or
2367
+ // `dealloc_impl()`, with `nothrow`, but that would be a subtle
2368
+ // behavior change and could make debugging more difficult.
2369
+ PyEval_RestoreThread (py_ts);
2370
+ throw ;
2371
+ }
2372
+ PyEval_RestoreThread (py_ts);
2373
+ }
2374
+
2345
2375
static detail::function_record *get_function_record (handle h) {
2346
2376
h = detail::get_function (h);
2347
2377
if (!h) {
0 commit comments