123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164 |
- import io
- import struct
- import ctypes
- import pytest
- import env
- from pybind11_tests import buffers as m
- from pybind11_tests import ConstructorStats
- np = pytest.importorskip("numpy")
- def test_from_python():
- with pytest.raises(RuntimeError) as excinfo:
- m.Matrix(np.array([1, 2, 3]))
- assert str(excinfo.value) == "Incompatible buffer format!"
- m3 = np.array([[1, 2, 3], [4, 5, 6]]).astype(np.float32)
- m4 = m.Matrix(m3)
- for i in range(m4.rows()):
- for j in range(m4.cols()):
- assert m3[i, j] == m4[i, j]
- cstats = ConstructorStats.get(m.Matrix)
- assert cstats.alive() == 1
- del m3, m4
- assert cstats.alive() == 0
- assert cstats.values() == ["2x3 matrix"]
- assert cstats.copy_constructions == 0
-
- assert cstats.copy_assignments == 0
- assert cstats.move_assignments == 0
- def test_to_python():
- mat = m.Matrix(5, 4)
- assert memoryview(mat).shape == (5, 4)
- assert mat[2, 3] == 0
- mat[2, 3] = 4.0
- mat[3, 2] = 7.0
- assert mat[2, 3] == 4
- assert mat[3, 2] == 7
- assert struct.unpack_from("f", mat, (3 * 4 + 2) * 4) == (7,)
- assert struct.unpack_from("f", mat, (2 * 4 + 3) * 4) == (4,)
- mat2 = np.array(mat, copy=False)
- assert mat2.shape == (5, 4)
- assert abs(mat2).sum() == 11
- assert mat2[2, 3] == 4 and mat2[3, 2] == 7
- mat2[2, 3] = 5
- assert mat2[2, 3] == 5
- cstats = ConstructorStats.get(m.Matrix)
- assert cstats.alive() == 1
- del mat
- pytest.gc_collect()
- assert cstats.alive() == 1
- del mat2
- pytest.gc_collect()
- assert cstats.alive() == 0
- assert cstats.values() == ["5x4 matrix"]
- assert cstats.copy_constructions == 0
-
- assert cstats.copy_assignments == 0
- assert cstats.move_assignments == 0
- def test_inherited_protocol():
- """SquareMatrix is derived from Matrix and inherits the buffer protocol"""
- matrix = m.SquareMatrix(5)
- assert memoryview(matrix).shape == (5, 5)
- assert np.asarray(matrix).shape == (5, 5)
- def test_pointer_to_member_fn():
- for cls in [m.Buffer, m.ConstBuffer, m.DerivedBuffer]:
- buf = cls()
- buf.value = 0x12345678
- value = struct.unpack("i", bytearray(buf))[0]
- assert value == 0x12345678
- def test_readonly_buffer():
- buf = m.BufferReadOnly(0x64)
- view = memoryview(buf)
- assert view[0] == b"d" if env.PY2 else 0x64
- assert view.readonly
- with pytest.raises(TypeError):
- view[0] = b"\0" if env.PY2 else 0
- def test_selective_readonly_buffer():
- buf = m.BufferReadOnlySelect()
- memoryview(buf)[0] = b"d" if env.PY2 else 0x64
- assert buf.value == 0x64
- io.BytesIO(b"A").readinto(buf)
- assert buf.value == ord(b"A")
- buf.readonly = True
- with pytest.raises(TypeError):
- memoryview(buf)[0] = b"\0" if env.PY2 else 0
- with pytest.raises(TypeError):
- io.BytesIO(b"1").readinto(buf)
- def test_ctypes_array_1d():
- char1d = (ctypes.c_char * 10)()
- int1d = (ctypes.c_int * 15)()
- long1d = (ctypes.c_long * 7)()
- for carray in (char1d, int1d, long1d):
- info = m.get_buffer_info(carray)
- assert info.itemsize == ctypes.sizeof(carray._type_)
- assert info.size == len(carray)
- assert info.ndim == 1
- assert info.shape == [info.size]
- assert info.strides == [info.itemsize]
- assert not info.readonly
- def test_ctypes_array_2d():
- char2d = ((ctypes.c_char * 10) * 4)()
- int2d = ((ctypes.c_int * 15) * 3)()
- long2d = ((ctypes.c_long * 7) * 2)()
- for carray in (char2d, int2d, long2d):
- info = m.get_buffer_info(carray)
- assert info.itemsize == ctypes.sizeof(carray[0]._type_)
- assert info.size == len(carray) * len(carray[0])
- assert info.ndim == 2
- assert info.shape == [len(carray), len(carray[0])]
- assert info.strides == [info.itemsize * len(carray[0]), info.itemsize]
- assert not info.readonly
- @pytest.mark.skipif(
- "env.PYPY and env.PY2", reason="PyPy2 bytes buffer not reported as readonly"
- )
- def test_ctypes_from_buffer():
- test_pystr = b"0123456789"
- for pyarray in (test_pystr, bytearray(test_pystr)):
- pyinfo = m.get_buffer_info(pyarray)
- if pyinfo.readonly:
- cbytes = (ctypes.c_char * len(pyarray)).from_buffer_copy(pyarray)
- cinfo = m.get_buffer_info(cbytes)
- else:
- cbytes = (ctypes.c_char * len(pyarray)).from_buffer(pyarray)
- cinfo = m.get_buffer_info(cbytes)
- assert cinfo.size == pyinfo.size
- assert cinfo.ndim == pyinfo.ndim
- assert cinfo.shape == pyinfo.shape
- assert cinfo.strides == pyinfo.strides
- assert not cinfo.readonly
|