| // clinic/float.c.h uses internal pycore_modsupport.h API |
| #define PYTESTCAPI_NEED_INTERNAL_API |
| |
| #include "parts.h" |
| #include "util.h" |
| #include "clinic/float.c.h" |
| |
| |
| /*[clinic input] |
| module _testcapi |
| [clinic start generated code]*/ |
| /*[clinic end generated code: output=da39a3ee5e6b4b0d input=6361033e795369fc]*/ |
| |
| /*[clinic input] |
| _testcapi.float_pack |
| |
| size: int |
| d: double |
| le: int |
| / |
| |
| Test PyFloat_Pack2(), PyFloat_Pack4() and PyFloat_Pack8() |
| [clinic start generated code]*/ |
| |
| static PyObject * |
| _testcapi_float_pack_impl(PyObject *module, int size, double d, int le) |
| /*[clinic end generated code: output=7899bd98f8b6cb04 input=52c9115121999c98]*/ |
| { |
| switch (size) |
| { |
| case 2: |
| { |
| char data[2]; |
| if (PyFloat_Pack2(d, data, le) < 0) { |
| return NULL; |
| } |
| return PyBytes_FromStringAndSize(data, Py_ARRAY_LENGTH(data)); |
| } |
| case 4: |
| { |
| char data[4]; |
| if (PyFloat_Pack4(d, data, le) < 0) { |
| return NULL; |
| } |
| return PyBytes_FromStringAndSize(data, Py_ARRAY_LENGTH(data)); |
| } |
| case 8: |
| { |
| char data[8]; |
| if (PyFloat_Pack8(d, data, le) < 0) { |
| return NULL; |
| } |
| return PyBytes_FromStringAndSize(data, Py_ARRAY_LENGTH(data)); |
| } |
| default: break; |
| } |
| |
| PyErr_SetString(PyExc_ValueError, "size must 2, 4 or 8"); |
| return NULL; |
| } |
| |
| |
| /*[clinic input] |
| _testcapi.float_unpack |
| |
| data: str(accept={robuffer}, zeroes=True) |
| le: int |
| / |
| |
| Test PyFloat_Unpack2(), PyFloat_Unpack4() and PyFloat_Unpack8() |
| [clinic start generated code]*/ |
| |
| static PyObject * |
| _testcapi_float_unpack_impl(PyObject *module, const char *data, |
| Py_ssize_t data_length, int le) |
| /*[clinic end generated code: output=617059f889ddbfe4 input=c095e4bb75a696cd]*/ |
| { |
| assert(!PyErr_Occurred()); |
| double d; |
| switch (data_length) |
| { |
| case 2: |
| d = PyFloat_Unpack2(data, le); |
| break; |
| case 4: |
| d = PyFloat_Unpack4(data, le); |
| break; |
| case 8: |
| d = PyFloat_Unpack8(data, le); |
| break; |
| default: |
| PyErr_SetString(PyExc_ValueError, "data length must 2, 4 or 8 bytes"); |
| return NULL; |
| } |
| |
| if (d == -1.0 && PyErr_Occurred()) { |
| return NULL; |
| } |
| return PyFloat_FromDouble(d); |
| } |
| |
| |
| /* Test PyOS_string_to_double. */ |
| static PyObject * |
| test_string_to_double(PyObject *self, PyObject *Py_UNUSED(ignored)) |
| { |
| double result; |
| const char *msg; |
| |
| #define CHECK_STRING(STR, expected) \ |
| do { \ |
| result = PyOS_string_to_double(STR, NULL, NULL); \ |
| if (result == -1.0 && PyErr_Occurred()) { \ |
| return NULL; \ |
| } \ |
| if (result != (double)expected) { \ |
| msg = "conversion of " STR " to float failed"; \ |
| goto fail; \ |
| } \ |
| } while (0) |
| |
| #define CHECK_INVALID(STR) \ |
| do { \ |
| result = PyOS_string_to_double(STR, NULL, NULL); \ |
| if (result == -1.0 && PyErr_Occurred()) { \ |
| if (PyErr_ExceptionMatches(PyExc_ValueError)) { \ |
| PyErr_Clear(); \ |
| } \ |
| else { \ |
| return NULL; \ |
| } \ |
| } \ |
| else { \ |
| msg = "conversion of " STR " didn't raise ValueError"; \ |
| goto fail; \ |
| } \ |
| } while (0) |
| |
| CHECK_STRING("0.1", 0.1); |
| CHECK_STRING("1.234", 1.234); |
| CHECK_STRING("-1.35", -1.35); |
| CHECK_STRING(".1e01", 1.0); |
| CHECK_STRING("2.e-2", 0.02); |
| |
| CHECK_INVALID(" 0.1"); |
| CHECK_INVALID("\t\n-3"); |
| CHECK_INVALID(".123 "); |
| CHECK_INVALID("3\n"); |
| CHECK_INVALID("123abc"); |
| |
| Py_RETURN_NONE; |
| fail: |
| PyErr_Format(PyExc_AssertionError, "test_string_to_double: %s", msg); |
| return NULL; |
| #undef CHECK_STRING |
| #undef CHECK_INVALID |
| } |
| |
| |
| static PyMethodDef test_methods[] = { |
| _TESTCAPI_FLOAT_PACK_METHODDEF |
| _TESTCAPI_FLOAT_UNPACK_METHODDEF |
| {"test_string_to_double", test_string_to_double, METH_NOARGS}, |
| {NULL}, |
| }; |
| |
| int |
| _PyTestCapi_Init_Float(PyObject *mod) |
| { |
| if (PyModule_AddFunctions(mod, test_methods) < 0) { |
| return -1; |
| } |
| |
| return 0; |
| } |