/* * C Extension module to smoke test pyatomic.h API. * * This only tests basic functionality, not any synchronizing ordering. */ /* Always enable assertions */ #undef NDEBUG #include "Python.h" #include "cpython/pyatomic.h" #include "parts.h" // We define atomic bitwise operations on these types #define FOR_BITWISE_TYPES(V) \ V(uint8, uint8_t) \ V(uint16, uint16_t) \ V(uint32, uint32_t) \ V(uint64, uint64_t) \ V(uintptr, uintptr_t) // We define atomic addition on these types #define FOR_ARITHMETIC_TYPES(V) \ FOR_BITWISE_TYPES(V) \ V(int, int) \ V(uint, unsigned int) \ V(int8, int8_t) \ V(int16, int16_t) \ V(int32, int32_t) \ V(int64, int64_t) \ V(intptr, intptr_t) \ V(ssize, Py_ssize_t) // We define atomic load, store, exchange, and compare_exchange on these types #define FOR_ALL_TYPES(V) \ FOR_ARITHMETIC_TYPES(V) \ V(ptr, void*) #define IMPL_TEST_ADD(suffix, dtype) \ static PyObject * \ test_atomic_add_##suffix(PyObject *self, PyObject *obj) { \ dtype x = 0; \ assert(_Py_atomic_add_##suffix(&x, 1) == 0); \ assert(x == 1); \ assert(_Py_atomic_add_##suffix(&x, 2) == 1); \ assert(x == 3); \ assert(_Py_atomic_add_##suffix(&x, -2) == 3); \ assert(x == 1); \ assert(_Py_atomic_add_##suffix(&x, -1) == 1); \ assert(x == 0); \ assert(_Py_atomic_add_##suffix(&x, -1) == 0); \ assert(x == (dtype)-1); \ assert(_Py_atomic_add_##suffix(&x, -2) == (dtype)-1); \ assert(x == (dtype)-3); \ assert(_Py_atomic_add_##suffix(&x, 2) == (dtype)-3); \ assert(x == (dtype)-1); \ Py_RETURN_NONE; \ } FOR_ARITHMETIC_TYPES(IMPL_TEST_ADD) #define IMPL_TEST_COMPARE_EXCHANGE(suffix, dtype) \ static PyObject * \ test_atomic_compare_exchange_##suffix(PyObject *self, PyObject *obj) { \ dtype x = (dtype)0; \ dtype y = (dtype)1; \ dtype z = (dtype)2; \ assert(_Py_atomic_compare_exchange_##suffix(&x, &y, z) == 0); \ assert(x == 0); \ assert(y == 0); \ assert(_Py_atomic_compare_exchange_##suffix(&x, &y, z) == 1); \ assert(x == z); \ assert(y == 0); \ assert(_Py_atomic_compare_exchange_##suffix(&x, &y, z) == 0); \ assert(x == z); \ assert(y == z); \ Py_RETURN_NONE; \ } FOR_ALL_TYPES(IMPL_TEST_COMPARE_EXCHANGE) #define IMPL_TEST_EXCHANGE(suffix, dtype) \ static PyObject * \ test_atomic_exchange_##suffix(PyObject *self, PyObject *obj) { \ dtype x = (dtype)0; \ dtype y = (dtype)1; \ dtype z = (dtype)2; \ assert(_Py_atomic_exchange_##suffix(&x, y) == (dtype)0); \ assert(x == (dtype)1); \ assert(_Py_atomic_exchange_##suffix(&x, z) == (dtype)1); \ assert(x == (dtype)2); \ assert(_Py_atomic_exchange_##suffix(&x, y) == (dtype)2); \ assert(x == (dtype)1); \ Py_RETURN_NONE; \ } FOR_ALL_TYPES(IMPL_TEST_EXCHANGE) #define IMPL_TEST_LOAD_STORE(suffix, dtype) \ static PyObject * \ test_atomic_load_store_##suffix(PyObject *self, PyObject *obj) { \ dtype x = (dtype)0; \ dtype y = (dtype)1; \ dtype z = (dtype)2; \ assert(_Py_atomic_load_##suffix(&x) == (dtype)0); \ assert(x == (dtype)0); \ _Py_atomic_store_##suffix(&x, y); \ assert(_Py_atomic_load_##suffix(&x) == (dtype)1); \ assert(x == (dtype)1); \ _Py_atomic_store_##suffix##_relaxed(&x, z); \ assert(_Py_atomic_load_##suffix##_relaxed(&x) == (dtype)2); \ assert(x == (dtype)2); \ Py_RETURN_NONE; \ } FOR_ALL_TYPES(IMPL_TEST_LOAD_STORE) #define IMPL_TEST_AND_OR(suffix, dtype) \ static PyObject * \ test_atomic_and_or_##suffix(PyObject *self, PyObject *obj) { \ dtype x = (dtype)0; \ dtype y = (dtype)1; \ dtype z = (dtype)3; \ assert(_Py_atomic_or_##suffix(&x, z) == (dtype)0); \ assert(x == (dtype)3); \ assert(_Py_atomic_and_##suffix(&x, y) == (dtype)3); \ assert(x == (dtype)1); \ Py_RETURN_NONE; \ } FOR_BITWISE_TYPES(IMPL_TEST_AND_OR) static PyObject * test_atomic_fences(PyObject *self, PyObject *obj) { // Just make sure that the fences compile. We are not // testing any synchronizing ordering. _Py_atomic_fence_seq_cst(); _Py_atomic_fence_release(); Py_RETURN_NONE; } static PyObject * test_atomic_release_acquire(PyObject *self, PyObject *obj) { void *x = NULL; void *y = &y; assert(_Py_atomic_load_ptr_acquire(&x) == NULL); _Py_atomic_store_ptr_release(&x, y); assert(x == y); assert(_Py_atomic_load_ptr_acquire(&x) == y); Py_RETURN_NONE; } // NOTE: all tests should start with "test_atomic_" to be included // in test_pyatomic.py #define BIND_TEST_ADD(suffix, dtype) \ {"test_atomic_add_" #suffix, test_atomic_add_##suffix, METH_NOARGS}, #define BIND_TEST_COMPARE_EXCHANGE(suffix, dtype) \ {"test_atomic_compare_exchange_" #suffix, test_atomic_compare_exchange_##suffix, METH_NOARGS}, #define BIND_TEST_EXCHANGE(suffix, dtype) \ {"test_atomic_exchange_" #suffix, test_atomic_exchange_##suffix, METH_NOARGS}, #define BIND_TEST_LOAD_STORE(suffix, dtype) \ {"test_atomic_load_store_" #suffix, test_atomic_load_store_##suffix, METH_NOARGS}, #define BIND_TEST_AND_OR(suffix, dtype) \ {"test_atomic_and_or_" #suffix, test_atomic_and_or_##suffix, METH_NOARGS}, static PyMethodDef test_methods[] = { FOR_ARITHMETIC_TYPES(BIND_TEST_ADD) FOR_ALL_TYPES(BIND_TEST_COMPARE_EXCHANGE) FOR_ALL_TYPES(BIND_TEST_EXCHANGE) FOR_ALL_TYPES(BIND_TEST_LOAD_STORE) FOR_BITWISE_TYPES(BIND_TEST_AND_OR) {"test_atomic_fences", test_atomic_fences, METH_NOARGS}, {"test_atomic_release_acquire", test_atomic_release_acquire, METH_NOARGS}, {NULL, NULL} /* sentinel */ }; int _PyTestCapi_Init_PyAtomic(PyObject *mod) { if (PyModule_AddFunctions(mod, test_methods) < 0) { return -1; } return 0; }