2023-08-31 18:41:18 -03:00
|
|
|
/*
|
|
|
|
* C Extension module to smoke test pyatomic.h API.
|
|
|
|
*
|
|
|
|
* This only tests basic functionality, not any synchronizing ordering.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#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();
|
2024-07-08 15:52:07 -03:00
|
|
|
_Py_atomic_fence_acquire();
|
2023-08-31 18:41:18 -03:00
|
|
|
_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;
|
|
|
|
}
|
|
|
|
|
2023-10-16 19:32:50 -03:00
|
|
|
static PyObject *
|
|
|
|
test_atomic_load_store_int_release_acquire(PyObject *self, PyObject *obj) { \
|
|
|
|
int x = 0;
|
|
|
|
int y = 1;
|
|
|
|
int z = 2;
|
|
|
|
assert(_Py_atomic_load_int_acquire(&x) == 0);
|
|
|
|
_Py_atomic_store_int_release(&x, y);
|
|
|
|
assert(x == y);
|
|
|
|
assert(_Py_atomic_load_int_acquire(&x) == y);
|
|
|
|
_Py_atomic_store_int_release(&x, z);
|
|
|
|
assert(x == z);
|
|
|
|
assert(_Py_atomic_load_int_acquire(&x) == z);
|
|
|
|
Py_RETURN_NONE;
|
|
|
|
}
|
|
|
|
|
2023-08-31 18:41:18 -03:00
|
|
|
// 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},
|
2023-10-16 19:32:50 -03:00
|
|
|
{"test_atomic_load_store_int_release_acquire", test_atomic_load_store_int_release_acquire, METH_NOARGS},
|
2023-08-31 18:41:18 -03:00
|
|
|
{NULL, NULL} /* sentinel */
|
|
|
|
};
|
|
|
|
|
|
|
|
int
|
|
|
|
_PyTestCapi_Init_PyAtomic(PyObject *mod)
|
|
|
|
{
|
|
|
|
if (PyModule_AddFunctions(mod, test_methods) < 0) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|