_test_extension.c 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258
  1. /* This is a set of functions used by test_extension_interface.py to test the
  2. * Greenlet C API.
  3. */
  4. #include "../greenlet.h"
  5. #ifndef Py_RETURN_NONE
  6. # define Py_RETURN_NONE return Py_INCREF(Py_None), Py_None
  7. #endif
  8. #define TEST_MODULE_NAME "_test_extension"
  9. // CAUTION: MSVC is stupidly picky:
  10. //
  11. // "The compiler ignores, without warning, any __declspec keywords
  12. // placed after * or & and in front of the variable identifier in a
  13. // declaration."
  14. // (https://docs.microsoft.com/en-us/cpp/cpp/declspec?view=msvc-160)
  15. //
  16. // So pointer return types must be handled differently (because of the
  17. // trailing *), or you get inscrutable compiler warnings like "error
  18. // C2059: syntax error: ''"
  19. //
  20. // In C23, there is a standard syntax for attributes, and
  21. // GCC defines an attribute to use with this: [[gnu:noinline]].
  22. // In the future, this is expected to become standard.
  23. #if defined(__GNUC__) || defined(__clang__)
  24. /* We used to check for GCC 4+ or 3.4+, but those compilers are
  25. laughably out of date. Just assume they support it. */
  26. # define UNUSED(x) UNUSED_ ## x __attribute__((__unused__))
  27. #elif defined(_MSC_VER)
  28. /* We used to check for && (_MSC_VER >= 1300) but that's also out of date. */
  29. # define UNUSED(x) UNUSED_ ## x
  30. #endif
  31. static PyObject*
  32. test_switch(PyObject* UNUSED(self), PyObject* greenlet)
  33. {
  34. PyObject* result = NULL;
  35. if (greenlet == NULL || !PyGreenlet_Check(greenlet)) {
  36. PyErr_BadArgument();
  37. return NULL;
  38. }
  39. result = PyGreenlet_Switch((PyGreenlet*)greenlet, NULL, NULL);
  40. if (result == NULL) {
  41. if (!PyErr_Occurred()) {
  42. PyErr_SetString(PyExc_AssertionError,
  43. "greenlet.switch() failed for some reason.");
  44. }
  45. return NULL;
  46. }
  47. Py_INCREF(result);
  48. return result;
  49. }
  50. static PyObject*
  51. test_switch_kwargs(PyObject* UNUSED(self), PyObject* args, PyObject* kwargs)
  52. {
  53. PyGreenlet* g = NULL;
  54. PyObject* result = NULL;
  55. PyArg_ParseTuple(args, "O!", &PyGreenlet_Type, &g);
  56. if (g == NULL || !PyGreenlet_Check(g)) {
  57. PyErr_BadArgument();
  58. return NULL;
  59. }
  60. result = PyGreenlet_Switch(g, NULL, kwargs);
  61. if (result == NULL) {
  62. if (!PyErr_Occurred()) {
  63. PyErr_SetString(PyExc_AssertionError,
  64. "greenlet.switch() failed for some reason.");
  65. }
  66. return NULL;
  67. }
  68. Py_XINCREF(result);
  69. return result;
  70. }
  71. static PyObject*
  72. test_getcurrent(PyObject* UNUSED(self))
  73. {
  74. PyGreenlet* g = PyGreenlet_GetCurrent();
  75. if (g == NULL || !PyGreenlet_Check(g) || !PyGreenlet_ACTIVE(g)) {
  76. PyErr_SetString(PyExc_AssertionError,
  77. "getcurrent() returned an invalid greenlet");
  78. Py_XDECREF(g);
  79. return NULL;
  80. }
  81. Py_DECREF(g);
  82. Py_RETURN_NONE;
  83. }
  84. static PyObject*
  85. test_setparent(PyObject* UNUSED(self), PyObject* arg)
  86. {
  87. PyGreenlet* current;
  88. PyGreenlet* greenlet = NULL;
  89. if (arg == NULL || !PyGreenlet_Check(arg)) {
  90. PyErr_BadArgument();
  91. return NULL;
  92. }
  93. if ((current = PyGreenlet_GetCurrent()) == NULL) {
  94. return NULL;
  95. }
  96. greenlet = (PyGreenlet*)arg;
  97. if (PyGreenlet_SetParent(greenlet, current)) {
  98. Py_DECREF(current);
  99. return NULL;
  100. }
  101. Py_DECREF(current);
  102. if (PyGreenlet_Switch(greenlet, NULL, NULL) == NULL) {
  103. return NULL;
  104. }
  105. Py_RETURN_NONE;
  106. }
  107. static PyObject*
  108. test_new_greenlet(PyObject* UNUSED(self), PyObject* callable)
  109. {
  110. PyObject* result = NULL;
  111. PyGreenlet* greenlet = PyGreenlet_New(callable, NULL);
  112. if (!greenlet) {
  113. return NULL;
  114. }
  115. result = PyGreenlet_Switch(greenlet, NULL, NULL);
  116. Py_CLEAR(greenlet);
  117. if (result == NULL) {
  118. return NULL;
  119. }
  120. Py_INCREF(result);
  121. return result;
  122. }
  123. static PyObject*
  124. test_raise_dead_greenlet(PyObject* UNUSED(self))
  125. {
  126. PyErr_SetString(PyExc_GreenletExit, "test GreenletExit exception.");
  127. return NULL;
  128. }
  129. static PyObject*
  130. test_raise_greenlet_error(PyObject* UNUSED(self))
  131. {
  132. PyErr_SetString(PyExc_GreenletError, "test greenlet.error exception");
  133. return NULL;
  134. }
  135. static PyObject*
  136. test_throw(PyObject* UNUSED(self), PyGreenlet* g)
  137. {
  138. const char msg[] = "take that sucka!";
  139. PyObject* msg_obj = Py_BuildValue("s", msg);
  140. PyGreenlet_Throw(g, PyExc_ValueError, msg_obj, NULL);
  141. Py_DECREF(msg_obj);
  142. if (PyErr_Occurred()) {
  143. return NULL;
  144. }
  145. Py_RETURN_NONE;
  146. }
  147. static PyObject*
  148. test_throw_exact(PyObject* UNUSED(self), PyObject* args)
  149. {
  150. PyGreenlet* g = NULL;
  151. PyObject* typ = NULL;
  152. PyObject* val = NULL;
  153. PyObject* tb = NULL;
  154. if (!PyArg_ParseTuple(args, "OOOO:throw", &g, &typ, &val, &tb)) {
  155. return NULL;
  156. }
  157. PyGreenlet_Throw(g, typ, val, tb);
  158. if (PyErr_Occurred()) {
  159. return NULL;
  160. }
  161. Py_RETURN_NONE;
  162. }
  163. static PyMethodDef test_methods[] = {
  164. {"test_switch",
  165. (PyCFunction)test_switch,
  166. METH_O,
  167. "Switch to the provided greenlet sending provided arguments, and \n"
  168. "return the results."},
  169. {"test_switch_kwargs",
  170. (PyCFunction)test_switch_kwargs,
  171. METH_VARARGS | METH_KEYWORDS,
  172. "Switch to the provided greenlet sending the provided keyword args."},
  173. {"test_getcurrent",
  174. (PyCFunction)test_getcurrent,
  175. METH_NOARGS,
  176. "Test PyGreenlet_GetCurrent()"},
  177. {"test_setparent",
  178. (PyCFunction)test_setparent,
  179. METH_O,
  180. "Se the parent of the provided greenlet and switch to it."},
  181. {"test_new_greenlet",
  182. (PyCFunction)test_new_greenlet,
  183. METH_O,
  184. "Test PyGreenlet_New()"},
  185. {"test_raise_dead_greenlet",
  186. (PyCFunction)test_raise_dead_greenlet,
  187. METH_NOARGS,
  188. "Just raise greenlet.GreenletExit"},
  189. {"test_raise_greenlet_error",
  190. (PyCFunction)test_raise_greenlet_error,
  191. METH_NOARGS,
  192. "Just raise greenlet.error"},
  193. {"test_throw",
  194. (PyCFunction)test_throw,
  195. METH_O,
  196. "Throw a ValueError at the provided greenlet"},
  197. {"test_throw_exact",
  198. (PyCFunction)test_throw_exact,
  199. METH_VARARGS,
  200. "Throw exactly the arguments given at the provided greenlet"},
  201. {NULL, NULL, 0, NULL}
  202. };
  203. #define INITERROR return NULL
  204. static struct PyModuleDef moduledef = {PyModuleDef_HEAD_INIT,
  205. TEST_MODULE_NAME,
  206. NULL,
  207. 0,
  208. test_methods,
  209. NULL,
  210. NULL,
  211. NULL,
  212. NULL};
  213. PyMODINIT_FUNC
  214. PyInit__test_extension(void)
  215. {
  216. PyObject* module = NULL;
  217. module = PyModule_Create(&moduledef);
  218. if (module == NULL) {
  219. return NULL;
  220. }
  221. PyGreenlet_Import();
  222. #ifdef Py_GIL_DISABLED
  223. PyUnstable_Module_SetGIL(module, Py_MOD_GIL_NOT_USED);
  224. #endif
  225. return module;
  226. }