blob: 9cfbe3f02b6af6f1f745ee907bee1866661ff45a (
plain) (
blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
|
from libcpp cimport bool
cdef extern from * nogil:
"""
// <thread> is defined in C++ and may define __cpp_lib_jthread
// Including it in this way lets us cimport it in the C++11 condition_variable wrappings
#include <thread>
#if defined(__cpp_lib_jthread)
#include <stop_token>
#endif
"""
cdef extern from * namespace "std" nogil:
cdef cppclass stop_token:
bool stop_requested() noexcept
bool stop_possible() noexcept
cdef cppclass nostopstate_t:
pass
nostopstate_t nostopstate
cdef cppclass stop_source:
stop_source() except+
stop_source(nostopstate_t) noexcept
bool request_stop() noexcept
void swap(stop_source& other) noexcept
stop_token get_token() noexcept
bool stop_requested() noexcept
bool stop_possible() noexcept
# stop_callback is not copyable or moveable which currently means it must
# be heap-allocated in Cython (although use with cpp_locals should eventually be supported too)
cdef cppclass stop_callback[Callback]:
# in principle the second argument is a template argument with a "std::constructable_from" constraint,
# but for Cython's purposes it probably makes sense to assume no conversion
stop_callback(stop_token st, Callback cb) noexcept
cdef extern from *:
"""
#if defined(__cpp_lib_jthread)
#include <optional>
#include <utility>
namespace {
using __pyx_func_ptr_stop_callback = std::stop_callback<void (*)()>;
class __pyx_python_stop_callback_holder {
class callable_py_object_holder {
PyObject *o;
public:
explicit callable_py_object_holder(PyObject *o)
: o(o)
{
Py_INCREF(o);
}
callable_py_object_holder(callable_py_object_holder&& rhs)
: o(std::exchange(rhs.o, nullptr))
{
}
~callable_py_object_holder() {
if (o) {
PyGILState_STATE state = PyGILState_Ensure();
Py_DECREF(o);
PyGILState_Release(state);
}
}
callable_py_object_holder(const callable_py_object_holder&) = delete;
callable_py_object_holder& operator=(const callable_py_object_holder&) = delete;
void operator()() const {
PyGILState_STATE state = PyGILState_Ensure();
PyObject *result = PyObject_CallObject(o, NULL);
if (!result) {
PyObject *s = PyUnicode_FromString("python_stop_callback_holder callback");
PyErr_WriteUnraisable(s);
Py_XDECREF(s);
} else {
Py_DECREF(result);
}
PyGILState_Release(state);
}
};
std::optional<std::stop_callback<callable_py_object_holder>> callback;
public:
__pyx_python_stop_callback_holder() = default;
__pyx_python_stop_callback_holder(std::stop_token token, PyObject *callable) {
initialize(std::move(token), callable);
}
__pyx_python_stop_callback_holder(const __pyx_python_stop_callback_holder&) = delete;
__pyx_python_stop_callback_holder& operator=(const __pyx_python_stop_callback_holder&) = delete;
void initialize(std::stop_token token, PyObject *callable) {
callback.emplace(std::move(token), callable_py_object_holder(callable));
}
};
}
#endif // defined(__cpp_lib_jthread)
"""
# This is provided as a convenience mainly as a reminder to use nogil functions!
ctypedef stop_callback[void (*)() noexcept nogil] func_ptr_stop_callback "__pyx_func_ptr_stop_callback"
# A fairly thin wrapper to let you create a stop callback with a Python object.
# For most uses, it should be created empty and then filled with "initialize"
cdef cppclass python_stop_callback_holder "__pyx_python_stop_callback_holder":
python_stop_callback_holder()
python_stop_callback_holder(stop_token token, object callable)
void initialize(stop_token token, object callable)
|