Skip to content
Merged
Show file tree
Hide file tree
Changes from 2 commits
Commits
Show all changes
57 commits
Select commit Hold shift + click to select a range
c9b872f
gh-61103: support double complex (_Complex) type in ctypes
skirpichev Jun 22, 2024
b402300
+ ifdef
skirpichev Jun 23, 2024
ad39e6d
+ fix rst formatting
skirpichev Jun 23, 2024
fd9135b
+ CMPLX
skirpichev Jun 23, 2024
c296fb3
Apply suggestions from code review
skirpichev Jun 23, 2024
7453ca5
+ support _Complex in c-analyzer
skirpichev Jun 23, 2024
e444c9c
+1
skirpichev Jun 23, 2024
2e90643
expand docs
skirpichev Jun 23, 2024
32ffacf
drop ctypes.__STDC_IEC_559_COMPLEX__
skirpichev Jun 23, 2024
9163950
And more tests, finally)
skirpichev Jun 23, 2024
bf81682
Apply suggestions from code review
skirpichev Jun 23, 2024
af2fdce
Apply suggestions from code review
skirpichev Jun 23, 2024
6d9d5ab
+ allow _Complex specifier before float/double/etc
skirpichev Jun 23, 2024
f9c709c
+ comment on clang workaround
skirpichev Jun 23, 2024
ccd418d
Address review:
skirpichev Jun 24, 2024
eb313dc
+ @requires_IEEE_754 for test_csqrt()
skirpichev Jun 24, 2024
693c04e
Revert "+ @requires_IEEE_754 for test_csqrt()"
skirpichev Jun 24, 2024
0ee7049
Adjust test_csqrt() to not use special numbers
skirpichev Jun 24, 2024
de91bbe
Revert "Adjust test_csqrt() to not use special numbers"
skirpichev Jun 24, 2024
b79200e
+1
skirpichev Jun 24, 2024
be6a685
+1
skirpichev Jun 24, 2024
04e89c2
revert test + fix typo
skirpichev Jun 24, 2024
0bc87da
Adjust test_csqrt() to not use special numbers
skirpichev Jun 24, 2024
6347412
+ make CMPLX static
skirpichev Jun 24, 2024
e757fb6
try bad CMPLX()
skirpichev Jun 24, 2024
1eb989c
more tests
skirpichev Jun 24, 2024
b44aae2
cexp test
skirpichev Jun 24, 2024
2153608
use my_csquare() in test
skirpichev Jun 24, 2024
af42aa8
revert CMPLX
skirpichev Jun 24, 2024
e57198e
Address review:
skirpichev Jun 24, 2024
36732bc
revert tests
skirpichev Jun 24, 2024
892b241
XXX skip test on MacOs arm64
skirpichev Jun 24, 2024
ee953c3
Revert "XXX skip test on MacOs arm64"
skirpichev Jun 24, 2024
875cbb8
XXX (to pass)
skirpichev Jun 24, 2024
f8e3a47
Address review: guard _complex.h with ifndef
skirpichev Jun 24, 2024
f447919
Apply suggestions from code review
skirpichev Jun 24, 2024
7b1c366
Merge branch 'master' into ctypes-c_complex-61103
skirpichev Jun 25, 2024
2ea028a
Add _Complex (sic!) to union result
skirpichev Jun 25, 2024
837add4
add missing ifdef + restore tests
skirpichev Jun 25, 2024
1b8f37e
Adjust test_csqrt() to not use special numbers
skirpichev Jun 24, 2024
0bd1ebe
Add _Complex also to tagPyCArgObject struct
skirpichev Jun 25, 2024
6eba445
cleanup: use "complex" instead of _Complex
skirpichev Jun 25, 2024
a6ac463
address review:
skirpichev Jun 26, 2024
615e8de
Update Modules/_ctypes/_ctypes.c
skirpichev Jun 26, 2024
f7d9973
address review: renamed macro (->PY_HAVE_C_COMPLEX)
skirpichev Jun 26, 2024
3d170bd
+ autoreconf
skirpichev Jun 26, 2024
4dd044c
+ test to ensure CMPLX preserves special components
skirpichev Jun 26, 2024
4cf7045
address review: rename macros to Py_HAVE_C_COMPLEX
skirpichev Jun 28, 2024
264fa7d
fix typos and adjust Makefile.pre.in
skirpichev Jun 28, 2024
f54b964
enable Py_HAVE_C_COMPLEX only if libff support complex numbers
skirpichev Jun 29, 2024
23db200
Revert "enable Py_HAVE_C_COMPLEX only if libff support complex numbers"
skirpichev Jun 29, 2024
5a7e366
ifdef Py_HAVE_C_COMPLEX -> if defined(Py_HAVE_C_COMPLEX) && defined(F…
skirpichev Jun 29, 2024
388e2ca
+ missing ffi.h imports
skirpichev Jun 29, 2024
e888565
+ add libffi's cflags/ldflags to _ctypes_test.c deps
skirpichev Jun 29, 2024
56b524c
Merge branch 'master' into ctypes-c_complex-61103
skirpichev Jun 29, 2024
e9cab4a
Adjust test in _ctypes.c to not use FFI_TYPE_COMPLEX define
skirpichev Jun 29, 2024
6d5bf66
sed: FFI_TYPE_COMPLEX -> FFI_TARGET_HAS_COMPLEX_TYPE
skirpichev Jun 30, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
11 changes: 11 additions & 0 deletions Doc/library/ctypes.rst
Original file line number Diff line number Diff line change
Expand Up @@ -266,6 +266,17 @@ Fundamental data types
(1)
The constructor accepts any object with a truth value.

Additionally, if IEC 60559 compatible complex arithmetic (Annex G) is supported, following
complex types are available:

+----------------------------------+---------------------------------+-----------------+
| ctypes type | C type | Python type |
+==================================+===================================================+
| :class:`!c_double_complex` | :c:expr:`double complex` | complex |
+----------------------------------+---------------------------------+-----------------+


(1)
All these types can be created by calling them with an optional initializer of
the correct type and value::

Expand Down
5 changes: 5 additions & 0 deletions Lib/ctypes/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@
from _ctypes import RTLD_LOCAL, RTLD_GLOBAL
from _ctypes import ArgumentError
from _ctypes import SIZEOF_TIME_T
from _ctypes import __STDC_IEC_559_COMPLEX__

from struct import calcsize as _calcsize

Expand Down Expand Up @@ -205,6 +206,10 @@ class c_longdouble(_SimpleCData):
if sizeof(c_longdouble) == sizeof(c_double):
c_longdouble = c_double

if __STDC_IEC_559_COMPLEX__:
class c_double_complex(_SimpleCData):
_type_ = "C"

if _calcsize("l") == _calcsize("q"):
# if long and long long have the same size, make c_longlong an alias for c_long
c_longlong = c_long
Expand Down
10 changes: 10 additions & 0 deletions Lib/test/test_ctypes/test_libc.py
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
import ctypes
import math
import unittest
from ctypes import (CDLL, CFUNCTYPE, POINTER, create_string_buffer, sizeof,
Expand All @@ -21,6 +22,15 @@ def test_sqrt(self):
self.assertEqual(lib.my_sqrt(4.0), 2.0)
self.assertEqual(lib.my_sqrt(2.0), math.sqrt(2.0))

def test_csqrt(self):
lib.my_csqrt.argtypes = ctypes.c_double_complex,
lib.my_csqrt.restype = ctypes.c_double_complex
self.assertEqual(lib.my_csqrt(4.0), 2+0j)
self.assertEqual(lib.my_csqrt(complex(-1, +0.)), complex(0, +1))
self.assertEqual(lib.my_csqrt(complex(-1, -0.)), complex(0, -1))

@unittest.skipUnless(ctypes.__STDC_IEC_559_COMPLEX__,
"requires C11 complex type")
def test_qsort(self):
comparefunc = CFUNCTYPE(c_int, POINTER(c_char), POINTER(c_char))
lib.my_qsort.argtypes = c_void_p, c_size_t, c_size_t, comparefunc
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
Support :c:expr:`double complex` C type in :mod:`ctypes` if compiler has C11
complex arithmetic. Patch by Sergey B Kirpichev.
22 changes: 21 additions & 1 deletion Modules/_ctypes/_ctypes.c
Original file line number Diff line number Diff line change
Expand Up @@ -1750,7 +1750,11 @@ class _ctypes.c_void_p "PyObject *" "clinic_state_sub()->PyCSimpleType_Type"
[clinic start generated code]*/
/*[clinic end generated code: output=da39a3ee5e6b4b0d input=dd4d9646c56f43a9]*/

#ifdef __STDC_IEC_559_COMPLEX__
static const char SIMPLE_TYPE_CHARS[] = "cbBhHiIlLdCfuzZqQPXOv?g";
#else
static const char SIMPLE_TYPE_CHARS[] = "cbBhHiIlLdfuzZqQPXOv?g";
#endif

/*[clinic input]
_ctypes.c_wchar_p.from_param as c_wchar_p_from_param
Expand Down Expand Up @@ -2226,7 +2230,17 @@ PyCSimpleType_init(PyObject *self, PyObject *args, PyObject *kwds)
goto error;
}

stginfo->ffi_type_pointer = *fmt->pffi_type;
if (fmt->pffi_type->type != FFI_TYPE_COMPLEX) {
stginfo->ffi_type_pointer = *fmt->pffi_type;
}
else {
stginfo->ffi_type_pointer.size = fmt->pffi_type->size;
stginfo->ffi_type_pointer.alignment = fmt->pffi_type->alignment;
stginfo->ffi_type_pointer.type = fmt->pffi_type->type;
stginfo->ffi_type_pointer.elements = PyMem_Malloc(2*sizeof(ffi_type));
memcpy(stginfo->ffi_type_pointer.elements,
fmt->pffi_type->elements, 2*sizeof(ffi_type));
}
stginfo->align = fmt->pffi_type->alignment;
stginfo->length = 0;
stginfo->size = fmt->pffi_type->size;
Expand Down Expand Up @@ -5811,6 +5825,12 @@ _ctypes_add_objects(PyObject *mod)
MOD_ADD("_cast_addr", PyLong_FromVoidPtr(cast));
MOD_ADD("_wstring_at_addr", PyLong_FromVoidPtr(wstring_at));

#ifdef __STDC_IEC_559_COMPLEX__
MOD_ADD("__STDC_IEC_559_COMPLEX__", PyLong_FromLong(__STDC_IEC_559_COMPLEX__));
#else
MOD_ADD("__STDC_IEC_559_COMPLEX__", PyLong_FromLong(0));
#endif

/* If RTLD_LOCAL is not defined (Windows!), set it to zero. */
#if !HAVE_DECL_RTLD_LOCAL
# define RTLD_LOCAL 0
Expand Down
11 changes: 11 additions & 0 deletions Modules/_ctypes/_ctypes_test.c
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,10 @@

#include <Python.h>

#ifdef __STDC_IEC_559_COMPLEX__
# include <complex.h> // csqrt()
# undef I
#endif
#include <stdio.h> // printf()
#include <stdlib.h> // qsort()
#include <string.h> // memset()
Expand Down Expand Up @@ -443,6 +447,13 @@ EXPORT(double) my_sqrt(double a)
return sqrt(a);
}

#ifdef __STDC_IEC_559_COMPLEX__
EXPORT(double complex) my_csqrt(double complex a)
{
return csqrt(a);
}
#endif

EXPORT(void) my_qsort(void *base, size_t num, size_t width, int(*compare)(const void*, const void*))
{
qsort(base, num, width, compare);
Expand Down
34 changes: 34 additions & 0 deletions Modules/_ctypes/cfield.c
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,10 @@
#include "pycore_bitutils.h" // _Py_bswap32()
#include "pycore_call.h" // _PyObject_CallNoArgs()

#ifdef __STDC_IEC_559_COMPLEX__
# include <complex.h> // complex
#endif

#include <ffi.h>
#include "ctypes.h"

Expand Down Expand Up @@ -1087,6 +1091,30 @@ d_get(void *ptr, Py_ssize_t size)
return PyFloat_FromDouble(val);
}

#ifdef __STDC_IEC_559_COMPLEX__
static PyObject *
C_set(void *ptr, PyObject *value, Py_ssize_t size)
{
Py_complex c = PyComplex_AsCComplex(value);
double complex x = CMPLX(c.real, c.imag);

if (c.real == -1 && PyErr_Occurred()) {
return NULL;
}
memcpy(ptr, &x, sizeof(x));
_RET(value);
}

static PyObject *
C_get(void *ptr, Py_ssize_t size)
{
double complex x;

memcpy(&x, ptr, sizeof(x));
return PyComplex_FromDoubles(creal(x), cimag(x));
}
#endif

static PyObject *
d_set_sw(void *ptr, PyObject *value, Py_ssize_t size)
{
Expand Down Expand Up @@ -1592,6 +1620,9 @@ static struct fielddesc formattable[] = {
{ 'B', B_set, B_get, NULL},
{ 'c', c_set, c_get, NULL},
{ 'd', d_set, d_get, NULL, d_set_sw, d_get_sw},
#ifdef __STDC_IEC_559_COMPLEX__
{ 'C', C_set, C_get, NULL},
#endif
{ 'g', g_set, g_get, NULL},
{ 'f', f_set, f_get, NULL, f_set_sw, f_get_sw},
{ 'h', h_set, h_get, NULL, h_set_sw, h_get_sw},
Expand Down Expand Up @@ -1642,6 +1673,9 @@ _ctypes_init_fielddesc(void)
case 'B': fd->pffi_type = &ffi_type_uchar; break;
case 'c': fd->pffi_type = &ffi_type_schar; break;
case 'd': fd->pffi_type = &ffi_type_double; break;
#ifdef __STDC_IEC_559_COMPLEX__
case 'C': fd->pffi_type = &ffi_type_complex_double; break;
#endif
case 'g': fd->pffi_type = &ffi_type_longdouble; break;
case 'f': fd->pffi_type = &ffi_type_float; break;
case 'h': fd->pffi_type = &ffi_type_sshort; break;
Expand Down