File: test_arraydatatype.py

package info (click to toggle)
pyopengl 3.1.10%2Bdfsg-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 15,024 kB
  • sloc: python: 108,056; sh: 13; makefile: 8
file content (419 lines) | stat: -rw-r--r-- 14,206 bytes parent folder | download | duplicates (2)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
import basetestcase
import os, sys
import OpenGL
from OpenGL.arrays import arraydatatype
from OpenGL.GL import *

try:
    import numpy as np
except ImportError as err:
    np = None
HERE = os.path.abspath(os.path.dirname(__file__))
import pytest
from OpenGL import acceleratesupport


class TestCoreDatatype(basetestcase.BaseTest):
    def test_arrayPointer(self):
        dt = arraydatatype.GLuintArray
        d = dt.zeros((3,))
        dp = dt.typedPointer(d)
        assert dp[0] == 0
        assert dp[1] == 0
        assert dp[2] == 0
        dp[1] = 1
        assert dp[1] == 1
        assert d[1] == 1

    def test_ctypes_array(self):
        color = (GLfloat * 3)(0, 1, 0)
        glColor3fv(color)

    if not OpenGL.ERROR_ON_COPY:

        def test_pointers(self):
            """Test that basic pointer functions work"""
            vertex = GLdouble * 3
            vArray = vertex * 2
            glVertexPointerd([[2, 3, 4, 5], [2, 3, 4, 5]])
            glVertexPointeri(([2, 3, 4, 5], [2, 3, 4, 5]))
            glVertexPointers([[2, 3, 4, 5], [2, 3, 4, 5]])
            glVertexPointerd(vArray(vertex(2, 3, 4), vertex(2, 3, 4)))
            myVector = vArray(vertex(2, 3, 4), vertex(2, 3, 4))
            glVertexPointer(
                3, GL_DOUBLE, 0, ctypes.cast(myVector, ctypes.POINTER(GLdouble))
            )

            repr(glVertexPointerb([[2, 3], [4, 5]]))
            glVertexPointerf([[2, 3], [4, 5]])
            assert arrays.ArrayDatatype.dataPointer(None) == None
            glVertexPointerf(None)

            glNormalPointerd([[2, 3, 4], [2, 3, 4]])
            glNormalPointerd(None)

            glTexCoordPointerd([[2, 3, 4], [2, 3, 4]])
            glTexCoordPointerd(None)

            glColorPointerd([[2, 3, 4], [2, 3, 4]])
            glColorPointerd(None)

            glEdgeFlagPointerb([0, 1, 0, 0, 1, 0])
            glEdgeFlagPointerb(None)

            glIndexPointerd([0, 1, 0, 0, 1, 0])
            glIndexPointerd(None)

            glColor4fv([0, 0, 0, 1])

            # string data-types...
            import struct

            s = struct.pack('>iiii', 2, 3, 4, 5) * 2
            glVertexPointer(4, GL_INT, 0, s)

        def test_texture(self):
            """Test texture (requires OpenGLContext and PIL)"""
            try:
                from OpenGLContext import texture
                import Image
                from OpenGL.GLUT import glutSolidTeapot
            except ImportError:
                pass
            else:
                assert glutSolidTeapot
                glEnable(GL_TEXTURE_2D)
                ourTexture = texture.Texture(
                    Image.open(os.path.join(HERE, 'yingyang.png'))
                )
                ourTexture()

                result = glGetTexImageub(GL_TEXTURE_2D, 0, GL_RGBA)
                assert isinstance(result, bytes), type(result)
                result = glGetTexImage(GL_TEXTURE_2D, 0, GL_RGBA, GL_UNSIGNED_BYTE)
                assert isinstance(result, bytes), type(result)

                glEnable(GL_LIGHTING)
                glEnable(GL_LIGHT0)
                glBegin(GL_TRIANGLES)
                try:
                    try:
                        glTexCoord2f(0.5, 1)
                        glVertex3f(0.0, 1.0, 0.0)
                    except Exception:
                        traceback.print_exc()
                    glTexCoord2f(0, 0)
                    glVertex3fv([-1, 0, 0])
                    glTexCoord2f(1, 0)
                    glVertex3dv([1, 0, 0])
                    try:
                        glVertex3dv([1, 0])
                    except ValueError:
                        assert (
                            OpenGL.ARRAY_SIZE_CHECKING
                        ), """Should have raised ValueError when doing array size checking"""
                    else:
                        assert (
                            not OpenGL.ARRAY_SIZE_CHECKING
                        ), """Should not have raised ValueError when not doing array size checking"""
                finally:
                    glEnd()

    @pytest.mark.skipif(not np, reason="Numpy not available")
    def test_numpyConversion(self):
        """Test that we can run a numpy conversion from double to float for glColorArray"""
        a = np.arange(0, 1.2, 0.1, 'd').reshape((-1, 3))
        glEnableClientState(GL_VERTEX_ARRAY)
        try:
            glColorPointerf(a)
            glColorPointerd(a)
        finally:
            glDisableClientState(GL_VERTEX_ARRAY)

    @pytest.mark.skipif(not np, reason="Numpy not available")
    def test_glbuffersubdata_numeric(self):
        from OpenGL.arrays import vbo

        assert vbo.get_implementation()
        points = np.array(
            [
                [0, 0, 0],
                [0, 1, 0],
                [1, 0.5, 0],
                [1, 0, 0],
                [1.5, 0.5, 0],
                [1.5, 0, 0],
            ],
            dtype='f',
        )
        d = vbo.VBO(points)
        with d:
            glBufferSubData(
                d.target,
                12,
                12,
                np.array([1, 1, 1], dtype='f'),
            )

    @pytest.mark.skipif(not np, reason="Numpy not available")
    @pytest.mark.skipif(OpenGL.ERROR_ON_COPY, reason="Test requires array copy")
    def test_copyNonContiguous(self):
        """Test that a non-contiguous (transposed) array gets applied as a copy"""
        glMatrixMode(GL_MODELVIEW)
        glPushMatrix()
        try:
            transf = np.identity(4, dtype=np.float32)
            # some arbitrary transformation...
            transf[0, 3] = 2.5
            transf[2, 3] = -80

            # what do we get with the un-transposed version...
            glMatrixMode(GL_MODELVIEW)
            glLoadIdentity()
            glMultMatrixf(transf)
            untransposed = glGetFloatv(GL_MODELVIEW_MATRIX)
            # now transposed...

            # with a copy it works...
            t2 = transf.transpose().copy()
            # This doesn't work:
            glLoadIdentity()
            glMultMatrixf(t2)
            # This does work:
            # glMultMatrixf(transf.transpose().copy())
            transposed = glGetFloatv(GL_MODELVIEW_MATRIX)

            assert not np.allclose(transposed, untransposed), (transposed, untransposed)

            t2 = transf.transpose()
            # This doesn't work:
            glLoadIdentity()
            glMultMatrixf(t2)
            # This does work:
            # glMultMatrixf(transf.transpose().copy())
            transposed = glGetFloatv(GL_MODELVIEW_MATRIX)

            assert not np.allclose(transposed, untransposed), (transposed, untransposed)
        finally:
            glMatrixMode(GL_MODELVIEW)
            glPopMatrix()

    def test_bytes_array_support(self):
        color = b'\000' * 12
        glColor3fv(color)

    @pytest.mark.skipif(
        not (OpenGL.USE_ACCELERATE and acceleratesupport.ACCELERATE_AVAILABLE),
        reason="Need OpenGL_accelerate for buffer support",
    )
    def test_bytearray_support(self):
        import struct

        data = struct.pack(b'fff', 0.5, 0.4, 0.3)
        color = bytearray(data)
        glColor3fv(color)

    @pytest.mark.skipif(
        not (OpenGL.USE_ACCELERATE and acceleratesupport.ACCELERATE_AVAILABLE),
        reason="Need OpenGL_accelerate for buffer support",
    )
    def test_buffer_api_basic(self):
        import array as silly_array

        structures = []
        if sys.version_info[:2] >= (3, 9):
            structures.append(
                (b'this and that', 13, 1, True, 1, b'B', [13], [1]),
            )

        if sys.version_info[:2] >= (2, 7):
            # GH#92 Big-endian hosts report different formats because yeah, obviously
            if sys.byteorder == 'little':  # x86 cases
                int_formats = [b'(3)<i', b'(3)<l', b'<i', b'<l']
            else:
                int_formats = [b'(3)>i', b'(3)>l', b'>i', b'>l']

            if sys.version_info[:2] not in [(3, 8), (3, 7)]:

                structures.append(
                    # on Python 3.4 we do *not* get the (3) prefix :(
                    (
                        (GLint * 3)(1, 2, 3),
                        12,
                        4,
                        False,
                        1,
                        int_formats,
                        [3],
                        None,
                    ),
                )

        if sys.version_info[:2] >= (3, 0) and sys.version_info[:2] not in [
            (3, 8),
            (3, 7),
        ]:
            # only supports buffer protocol in 3.x
            structures.extend(
                [
                    (
                        silly_array.array('I', [1, 2, 3]),
                        12,
                        4,
                        False,
                        1,
                        b'I',
                        [3],
                        [4],
                    ),
                ]
            )
        try:
            if sys.version_info[:2] not in [(3, 8), (3, 7)]:
                structures.append((memoryview(b'this'), 4, 1, True, 1, b'B', [4], [1]))
        except NameError:
            # Python 2.6 doesn't have memory view
            pass
        try:
            if array:
                structures.extend(
                    [
                        (
                            arange(0, 9, dtype='I').reshape((3, 3)),
                            36,
                            4,
                            False,
                            2,
                            b'I',
                            [3, 3],
                            [12, 4],
                        ),
                        (
                            arange(0, 9, dtype='I').reshape((3, 3))[:, 1],
                            12,
                            4,
                            False,
                            1,
                            b'I',
                            [3],
                            [12],
                        ),
                    ]
                )
        except NameError:
            # Don't have numpy installed...
            pass

        from OpenGL.arrays import _buffers

        for (
            object,
            length,
            itemsize,
            readonly,
            ndim,
            format,
            shape,
            strides,
        ) in structures:
            buf = _buffers.Py_buffer.from_object(
                object, _buffers.PyBUF_STRIDES | _buffers.PyBUF_FORMAT
            )
            with buf:
                assert buf.len == length, (object, length, buf.len)
                assert buf.itemsize == itemsize, (object, itemsize, buf.itemsize)
                assert buf.readonly == readonly, (object, readonly, buf.readonly)
                assert buf.ndim == ndim, (object, ndim, buf.ndim)
                if isinstance(format, list):
                    assert buf.format in format, (object, format, buf.format)
                else:
                    assert buf.format == format, (object, format, buf.format)
                assert buf.shape[: buf.ndim] == shape, (
                    object,
                    shape,
                    buf.shape[: buf.ndim],
                )
                assert buf.dims == shape, (object, shape, buf.dims)
                assert buf.buf
                if strides is None:
                    assert not buf.strides
                else:
                    assert buf.strides[: buf.ndim] == strides, (
                        object,
                        strides,
                        buf.strides[: buf.ndim],
                    )
            assert buf.obj == None, buf.obj
            del buf

    @pytest.mark.skipif(
        not (OpenGL.USE_ACCELERATE and acceleratesupport.ACCELERATE_AVAILABLE),
        reason="Need OpenGL_accelerate for buffer support",
    )
    def test_memoryview_support(self):
        color = bytearray(b'\000' * 12)
        mem = memoryview(color)
        glColor3fv(mem)

    def test_void_dp_for_void_dp_is_self(self):
        array = ctypes.c_voidp(12)
        translated = ArrayDatatype.voidDataPointer(array)
        assert translated.value == array.value, translated

    def test_params_python3_strings(self):
        try:
            glGetUniformBlockIndex(0, unicode("Moo"))
        except ArgumentError:
            assert (
                OpenGL.ERROR_ON_COPY
            ), """Shouldn't have raised error on copy for unicode"""
        except TypeError:
            raise
        except GLError:
            # expected error, as we don't have a shader there...
            pass

    @pytest.mark.skipif(not np, reason="Numpy not available")
    def test_array_subclass(self):
        s = Subclassed([0, 1, 2, 3, 4])
        result = arraydatatype.ArrayDatatype.asArray(s)
        assert isinstance(result, Subclassed)

    @pytest.mark.skipif(not np, reason="Numpy not available")
    def test_byte_count_numpy(self):
        for a, expected in [
            (np.array([1, 2], dtype='e'), 4),
            (np.array([1, 2], dtype='f'), 8),
            (np.array([1, 2], dtype='d'), 16),
            (np.array([1, 2], dtype='B'), 2),
            (np.array([[1], [2]], dtype='B'), 2),
            (np.float32(), 4),
            (np.float64(), 8),
        ]:
            handler = arraydatatype.ArrayDatatype.getHandler(a)

            assert type(a) in handler.HANDLED_TYPES, type(a)
            calculated = arraydatatype.ArrayDatatype.arrayByteCount(a)
            assert calculated == expected, "Byte count for %s was %s, expected %" % (
                a,
                calculated,
                expected,
            )

    def test_byte_count(self):
        for a, expected in [
            (ctypes.c_float(), 4),
            ((ctypes.c_float * 3 * 4)(), 4 * 3 * 4),
        ]:
            calculated = arraydatatype.ArrayDatatype.arrayByteCount(a)
            assert calculated == expected, "Byte count for %s was %s, expected %" % (
                a,
                calculated,
                expected,
            )


if np:

    class Subclassed(np.ndarray):
        pass