|
45 | 45 | import static com.oracle.graal.python.builtins.objects.cext.capi.transitions.ArgDescriptor.Int;
|
46 | 46 | import static com.oracle.graal.python.builtins.objects.cext.capi.transitions.ArgDescriptor.Pointer;
|
47 | 47 | import static com.oracle.graal.python.builtins.objects.cext.capi.transitions.ArgDescriptor.PyFrameObjectTransfer;
|
| 48 | +import static com.oracle.graal.python.builtins.objects.cext.capi.transitions.ArgDescriptor.PyObject; |
48 | 49 | import static com.oracle.graal.python.builtins.objects.cext.capi.transitions.ArgDescriptor.PyObjectBorrowed;
|
49 | 50 | import static com.oracle.graal.python.builtins.objects.cext.capi.transitions.ArgDescriptor.PyThreadState;
|
50 | 51 | import static com.oracle.graal.python.builtins.objects.cext.capi.transitions.ArgDescriptor.Py_ssize_t;
|
51 | 52 | import static com.oracle.graal.python.builtins.objects.cext.capi.transitions.ArgDescriptor.Void;
|
52 | 53 |
|
| 54 | +import com.oracle.graal.python.builtins.modules.cext.PythonCextBuiltins.CApiBinaryBuiltinNode; |
53 | 55 | import com.oracle.graal.python.builtins.modules.cext.PythonCextBuiltins.CApiBuiltin;
|
54 | 56 | import com.oracle.graal.python.builtins.modules.cext.PythonCextBuiltins.CApiNullaryBuiltinNode;
|
55 | 57 | import com.oracle.graal.python.builtins.modules.cext.PythonCextBuiltins.CApiUnaryBuiltinNode;
|
56 | 58 | import com.oracle.graal.python.builtins.objects.PNone;
|
| 59 | +import com.oracle.graal.python.builtins.objects.cext.capi.CApiContext; |
57 | 60 | import com.oracle.graal.python.builtins.objects.cext.capi.PThreadState;
|
| 61 | +import com.oracle.graal.python.builtins.objects.cext.capi.transitions.ArgDescriptor; |
58 | 62 | import com.oracle.graal.python.builtins.objects.dict.PDict;
|
59 | 63 | import com.oracle.graal.python.builtins.objects.frame.PFrame;
|
60 | 64 | import com.oracle.graal.python.builtins.objects.ints.PInt;
|
| 65 | +import com.oracle.graal.python.builtins.objects.thread.PThread; |
| 66 | +import com.oracle.graal.python.nodes.PGuards; |
| 67 | +import com.oracle.graal.python.nodes.PRaiseNode; |
| 68 | +import com.oracle.graal.python.nodes.PRootNode; |
61 | 69 | import com.oracle.graal.python.nodes.frame.GetCurrentFrameRef;
|
62 | 70 | import com.oracle.graal.python.nodes.frame.ReadCallerFrameNode;
|
63 | 71 | import com.oracle.graal.python.nodes.util.CannotCastException;
|
|
67 | 75 | import com.oracle.graal.python.runtime.object.PythonObjectFactory;
|
68 | 76 | import com.oracle.graal.python.util.OverflowException;
|
69 | 77 | import com.oracle.truffle.api.CompilerDirectives.TruffleBoundary;
|
| 78 | +import com.oracle.truffle.api.ThreadLocalAction; |
| 79 | +import com.oracle.truffle.api.TruffleLogger; |
70 | 80 | import com.oracle.truffle.api.dsl.Bind;
|
71 | 81 | import com.oracle.truffle.api.dsl.Cached;
|
72 | 82 | import com.oracle.truffle.api.dsl.Specialization;
|
73 | 83 | import com.oracle.truffle.api.interop.InteropLibrary;
|
74 | 84 | import com.oracle.truffle.api.library.CachedLibrary;
|
75 | 85 | import com.oracle.truffle.api.nodes.Node;
|
| 86 | +import com.oracle.truffle.api.nodes.RootNode; |
76 | 87 |
|
77 | 88 | public final class PythonCextPyStateBuiltins {
|
78 | 89 |
|
@@ -137,6 +148,58 @@ PDict get(@Cached PythonObjectFactory factory) {
|
137 | 148 | }
|
138 | 149 | }
|
139 | 150 |
|
| 151 | + @CApiBuiltin(ret = Int, args = {ArgDescriptor.UNSIGNED_LONG, PyObject}, call = Direct) |
| 152 | + abstract static class PyThreadState_SetAsyncExc extends CApiBinaryBuiltinNode { |
| 153 | + public static final TruffleLogger LOGGER = CApiContext.getLogger(PyThreadState_SetAsyncExc.class); |
| 154 | + |
| 155 | + @Specialization |
| 156 | + @TruffleBoundary |
| 157 | + int doIt(long id, Object exceptionObject) { |
| 158 | + for (Thread thread : getContext().getThreads()) { |
| 159 | + if (PThread.getThreadId(thread) == id) { |
| 160 | + if (PGuards.isNoValue(exceptionObject)) { |
| 161 | + LOGGER.warning("The application used PyThreadState_SetAsyncExc to clear an exception on another thread. " + |
| 162 | + "This is not supported and ignored by GraalPy."); |
| 163 | + return 1; |
| 164 | + } |
| 165 | + ThreadLocalAction action = new ThreadLocalAction(true, false) { |
| 166 | + static final int MAX_MISSED_COUNT = 20; |
| 167 | + int missedCount = 0; |
| 168 | + |
| 169 | + @Override |
| 170 | + protected void perform(Access access) { |
| 171 | + if (missedCount == MAX_MISSED_COUNT) { |
| 172 | + throw PRaiseNode.raiseExceptionObject(null, exceptionObject); |
| 173 | + } |
| 174 | + // If possible, we do not want to raise in some internal code, it could |
| 175 | + // corrupt internal data structures. |
| 176 | + Node location = access.getLocation(); |
| 177 | + if (location != null) { |
| 178 | + RootNode rootNode = location.getRootNode(); |
| 179 | + if (rootNode instanceof PRootNode && !rootNode.isInternal()) { |
| 180 | + throw PRaiseNode.raiseExceptionObject(null, exceptionObject); |
| 181 | + } |
| 182 | + } |
| 183 | + // Heuristic fabricated out of thin air: |
| 184 | + if (missedCount++ < MAX_MISSED_COUNT) { |
| 185 | + if (missedCount % 2 == 0) { |
| 186 | + try { |
| 187 | + Thread.sleep(1); |
| 188 | + } catch (InterruptedException ignored) { |
| 189 | + } |
| 190 | + } |
| 191 | + getContext().getEnv().submitThreadLocal(new Thread[]{thread}, this); |
| 192 | + } |
| 193 | + } |
| 194 | + }; |
| 195 | + getContext().getEnv().submitThreadLocal(new Thread[]{thread}, action); |
| 196 | + return 1; |
| 197 | + } |
| 198 | + } |
| 199 | + return 0; |
| 200 | + } |
| 201 | + } |
| 202 | + |
140 | 203 | @CApiBuiltin(ret = PyFrameObjectTransfer, args = {PyThreadState}, call = Direct)
|
141 | 204 | abstract static class PyThreadState_GetFrame extends CApiUnaryBuiltinNode {
|
142 | 205 | @Specialization
|
|
0 commit comments