blob: db1871e6aeed0fde8dfed97a912e18831b20dfed [file] [log] [blame]
Henry Schreinerd8c7ee02020-07-20 13:35:21 -04001# -*- coding: utf-8 -*-
Sebastian Koslowskidc65d662019-11-24 08:33:05 +01002import io
Bruce Merryfe0cf8b2017-05-17 10:52:33 +02003import struct
Sebastian Koslowskidc65d662019-11-24 08:33:05 +01004
Dean Moldovana0c1ccf2016-08-12 13:50:00 +02005import pytest
Sebastian Koslowskidc65d662019-11-24 08:33:05 +01006
Jason Rhinelander391c7542017-07-25 16:47:36 -04007from pybind11_tests import buffers as m
8from pybind11_tests import ConstructorStats
Dean Moldovana0c1ccf2016-08-12 13:50:00 +02009
Dean Moldovand47febc2017-03-10 15:42:42 +010010pytestmark = pytest.requires_numpy
11
Dean Moldovana0c1ccf2016-08-12 13:50:00 +020012with pytest.suppress(ImportError):
13 import numpy as np
14
15
Dean Moldovana0c1ccf2016-08-12 13:50:00 +020016def test_from_python():
17 with pytest.raises(RuntimeError) as excinfo:
Jason Rhinelander391c7542017-07-25 16:47:36 -040018 m.Matrix(np.array([1, 2, 3])) # trying to assign a 1D array
Dean Moldovana0c1ccf2016-08-12 13:50:00 +020019 assert str(excinfo.value) == "Incompatible buffer format!"
20
21 m3 = np.array([[1, 2, 3], [4, 5, 6]]).astype(np.float32)
Jason Rhinelander391c7542017-07-25 16:47:36 -040022 m4 = m.Matrix(m3)
Dean Moldovana0c1ccf2016-08-12 13:50:00 +020023
24 for i in range(m4.rows()):
25 for j in range(m4.cols()):
26 assert m3[i, j] == m4[i, j]
27
Jason Rhinelander391c7542017-07-25 16:47:36 -040028 cstats = ConstructorStats.get(m.Matrix)
Dean Moldovana0c1ccf2016-08-12 13:50:00 +020029 assert cstats.alive() == 1
30 del m3, m4
31 assert cstats.alive() == 0
32 assert cstats.values() == ["2x3 matrix"]
33 assert cstats.copy_constructions == 0
34 # assert cstats.move_constructions >= 0 # Don't invoke any
35 assert cstats.copy_assignments == 0
36 assert cstats.move_assignments == 0
Wenzel Jakob1d1f81b2016-12-16 15:00:46 +010037
38
39# PyPy: Memory leak in the "np.array(m, copy=False)" call
40# https://bitbucket.org/pypy/pypy/issues/2444
41@pytest.unsupported_on_pypy
Wenzel Jakob1d1f81b2016-12-16 15:00:46 +010042def test_to_python():
Ansgar Burchardta22dd2d2017-09-21 23:07:48 +020043 mat = m.Matrix(5, 4)
44 assert memoryview(mat).shape == (5, 4)
Wenzel Jakob1d1f81b2016-12-16 15:00:46 +010045
Jason Rhinelander391c7542017-07-25 16:47:36 -040046 assert mat[2, 3] == 0
Ansgar Burchardta22dd2d2017-09-21 23:07:48 +020047 mat[2, 3] = 4.0
48 mat[3, 2] = 7.0
Jason Rhinelander391c7542017-07-25 16:47:36 -040049 assert mat[2, 3] == 4
Ansgar Burchardta22dd2d2017-09-21 23:07:48 +020050 assert mat[3, 2] == 7
51 assert struct.unpack_from('f', mat, (3 * 4 + 2) * 4) == (7, )
52 assert struct.unpack_from('f', mat, (2 * 4 + 3) * 4) == (4, )
Wenzel Jakob1d1f81b2016-12-16 15:00:46 +010053
Jason Rhinelander391c7542017-07-25 16:47:36 -040054 mat2 = np.array(mat, copy=False)
Ansgar Burchardta22dd2d2017-09-21 23:07:48 +020055 assert mat2.shape == (5, 4)
56 assert abs(mat2).sum() == 11
57 assert mat2[2, 3] == 4 and mat2[3, 2] == 7
Jason Rhinelander391c7542017-07-25 16:47:36 -040058 mat2[2, 3] = 5
59 assert mat2[2, 3] == 5
Wenzel Jakob1d1f81b2016-12-16 15:00:46 +010060
Jason Rhinelander391c7542017-07-25 16:47:36 -040061 cstats = ConstructorStats.get(m.Matrix)
Wenzel Jakob1d1f81b2016-12-16 15:00:46 +010062 assert cstats.alive() == 1
Jason Rhinelander391c7542017-07-25 16:47:36 -040063 del mat
Wenzel Jakob1d1f81b2016-12-16 15:00:46 +010064 pytest.gc_collect()
65 assert cstats.alive() == 1
Jason Rhinelander391c7542017-07-25 16:47:36 -040066 del mat2 # holds a mat reference
Wenzel Jakob1d1f81b2016-12-16 15:00:46 +010067 pytest.gc_collect()
68 assert cstats.alive() == 0
Ansgar Burchardta22dd2d2017-09-21 23:07:48 +020069 assert cstats.values() == ["5x4 matrix"]
Wenzel Jakob1d1f81b2016-12-16 15:00:46 +010070 assert cstats.copy_constructions == 0
71 # assert cstats.move_constructions >= 0 # Don't invoke any
72 assert cstats.copy_assignments == 0
73 assert cstats.move_assignments == 0
Bruce Merryfe0cf8b2017-05-17 10:52:33 +020074
75
76@pytest.unsupported_on_pypy
Dean Moldovan427e4af2017-05-28 16:35:02 +020077def test_inherited_protocol():
78 """SquareMatrix is derived from Matrix and inherits the buffer protocol"""
Dean Moldovan427e4af2017-05-28 16:35:02 +020079
Jason Rhinelander391c7542017-07-25 16:47:36 -040080 matrix = m.SquareMatrix(5)
Dean Moldovan427e4af2017-05-28 16:35:02 +020081 assert memoryview(matrix).shape == (5, 5)
82 assert np.asarray(matrix).shape == (5, 5)
83
84
85@pytest.unsupported_on_pypy
Jason Rhinelander391c7542017-07-25 16:47:36 -040086def test_pointer_to_member_fn():
87 for cls in [m.Buffer, m.ConstBuffer, m.DerivedBuffer]:
Bruce Merryfe0cf8b2017-05-17 10:52:33 +020088 buf = cls()
89 buf.value = 0x12345678
90 value = struct.unpack('i', bytearray(buf))[0]
91 assert value == 0x12345678
Sebastian Koslowskidc65d662019-11-24 08:33:05 +010092
93
94@pytest.unsupported_on_pypy
95def test_readonly_buffer():
96 buf = m.BufferReadOnly(0x64)
97 view = memoryview(buf)
Eric Cousineauebdd0d32020-08-14 14:03:43 -040098 assert view[0] == b'd' if pytest.PY2 else 0x64
Sebastian Koslowskidc65d662019-11-24 08:33:05 +010099 assert view.readonly
100
101
102@pytest.unsupported_on_pypy
103def test_selective_readonly_buffer():
104 buf = m.BufferReadOnlySelect()
105
Eric Cousineauebdd0d32020-08-14 14:03:43 -0400106 memoryview(buf)[0] = b'd' if pytest.PY2 else 0x64
Sebastian Koslowskidc65d662019-11-24 08:33:05 +0100107 assert buf.value == 0x64
108
109 io.BytesIO(b'A').readinto(buf)
110 assert buf.value == ord(b'A')
111
112 buf.readonly = True
113 with pytest.raises(TypeError):
Eric Cousineauebdd0d32020-08-14 14:03:43 -0400114 memoryview(buf)[0] = b'\0' if pytest.PY2 else 0
Sebastian Koslowskidc65d662019-11-24 08:33:05 +0100115 with pytest.raises(TypeError):
116 io.BytesIO(b'1').readinto(buf)