| 
									
										
										
										
											2022-02-11 09:07:11 +11:00
										 |  |  | /* SPDX-License-Identifier: GPL-2.0-or-later */ | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-18 08:08:12 +11:00
										 |  |  | /** \file
 | 
					
						
							|  |  |  |  * \ingroup bpygpu | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2021-07-20 22:52:31 +10:00
										 |  |  |  * - Use `bpygpu_` for local API. | 
					
						
							|  |  |  |  * - Use `BPyGPU` for public API. | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <Python.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 20:15:56 +02:00
										 |  |  | #include "GPU_index_buffer.h"
 | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "BLI_math.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "../generic/py_capi_utils.h"
 | 
					
						
							|  |  |  | #include "../generic/python_utildefines.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-22 08:26:45 -03:00
										 |  |  | #include "gpu_py.h"
 | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  | #include "gpu_py_element.h" /* own include */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name IndexBuf Type
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-17 10:16:41 -03:00
										 |  |  | static PyObject *pygpu_IndexBuf__tp_new(PyTypeObject *UNUSED(type), PyObject *args, PyObject *kwds) | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-01-03 01:08:26 +11:00
										 |  |  |   BPYGPU_IS_INIT_OR_ERROR_OBJ; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-10 13:35:29 -03:00
										 |  |  |   const char *error_prefix = "IndexBuf.__new__"; | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |   bool ok = true; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-22 08:26:45 -03:00
										 |  |  |   struct PyC_StringEnum prim_type = {bpygpu_primtype_items, GPU_PRIM_NONE}; | 
					
						
							|  |  |  |   PyObject *seq; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |   uint verts_per_prim; | 
					
						
							|  |  |  |   uint index_len; | 
					
						
							|  |  |  |   GPUIndexBufBuilder builder; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |   static const char *_keywords[] = {"type", "seq", NULL}; | 
					
						
							| 
									
										
										
										
											2022-04-08 09:41:28 +10:00
										 |  |  |   static _PyArg_Parser _parser = { | 
					
						
							|  |  |  |       "$O" /* `type` */ | 
					
						
							|  |  |  |       "&O" /* `seq` */ | 
					
						
							|  |  |  |       ":IndexBuf.__new__", | 
					
						
							|  |  |  |       _keywords, | 
					
						
							|  |  |  |       0, | 
					
						
							|  |  |  |   }; | 
					
						
							| 
									
										
										
										
											2019-01-03 01:08:26 +11:00
										 |  |  |   if (!_PyArg_ParseTupleAndKeywordsFast( | 
					
						
							| 
									
										
										
										
											2021-02-22 08:26:45 -03:00
										 |  |  |           args, kwds, &_parser, PyC_ParseStringEnum, &prim_type, &seq)) { | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |     return NULL; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-22 08:26:45 -03:00
										 |  |  |   verts_per_prim = GPU_indexbuf_primitive_len(prim_type.value_found); | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |   if (verts_per_prim == -1) { | 
					
						
							| 
									
										
										
										
											2018-10-01 08:42:26 +10:00
										 |  |  |     PyErr_Format(PyExc_ValueError, | 
					
						
							|  |  |  |                  "The argument 'type' must be " | 
					
						
							|  |  |  |                  "'POINTS', 'LINES', 'TRIS' or 'LINES_ADJ'"); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-22 08:26:45 -03:00
										 |  |  |   if (PyObject_CheckBuffer(seq)) { | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |     Py_buffer pybuffer; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-22 08:26:45 -03:00
										 |  |  |     if (PyObject_GetBuffer(seq, &pybuffer, PyBUF_FORMAT | PyBUF_ND) == -1) { | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |       /* PyObject_GetBuffer already handles error messages. */ | 
					
						
							|  |  |  |       return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |     if (pybuffer.ndim != 1 && pybuffer.shape[1] != verts_per_prim) { | 
					
						
							|  |  |  |       PyErr_Format(PyExc_ValueError, "Each primitive must exactly %d indices", verts_per_prim); | 
					
						
							|  |  |  |       return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-06 01:15:15 -03:00
										 |  |  |     if (pybuffer.itemsize != 4 || | 
					
						
							| 
									
										
										
										
											2018-10-08 08:37:32 +11:00
										 |  |  |         PyC_StructFmt_type_is_float_any(PyC_StructFmt_type_from_str(pybuffer.format))) { | 
					
						
							| 
									
										
										
										
											2018-10-06 01:15:15 -03:00
										 |  |  |       PyErr_Format(PyExc_ValueError, "Each index must be an 4-bytes integer value"); | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |       return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |     index_len = pybuffer.shape[0]; | 
					
						
							|  |  |  |     if (pybuffer.ndim != 1) { | 
					
						
							|  |  |  |       index_len *= pybuffer.shape[1]; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-27 21:15:42 +10:00
										 |  |  |     /* The `vertex_len` parameter is only used for asserts in the Debug build. */ | 
					
						
							|  |  |  |     /* Not very useful in python since scripts are often tested in Release build. */ | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |     /* Use `INT_MAX` instead of the actual number of vertices. */ | 
					
						
							| 
									
										
										
										
											2021-02-22 08:26:45 -03:00
										 |  |  |     GPU_indexbuf_init(&builder, prim_type.value_found, index_len, INT_MAX); | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | #if 0
 | 
					
						
							|  |  |  |     uint *buf = pybuffer.buf; | 
					
						
							|  |  |  |     for (uint i = index_len; i--; buf++) { | 
					
						
							|  |  |  |       GPU_indexbuf_add_generic_vert(&builder, *buf); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2018-10-24 10:57:16 +02:00
										 |  |  |     memcpy(builder.data, pybuffer.buf, index_len * sizeof(*builder.data)); | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |     builder.index_len = index_len; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     PyBuffer_Release(&pybuffer); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							| 
									
										
										
										
											2021-02-22 08:26:45 -03:00
										 |  |  |     PyObject *seq_fast = PySequence_Fast(seq, error_prefix); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |     if (seq_fast == NULL) { | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |     const uint seq_len = PySequence_Fast_GET_SIZE(seq_fast); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |     PyObject **seq_items = PySequence_Fast_ITEMS(seq_fast); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |     index_len = seq_len * verts_per_prim; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-27 21:15:42 +10:00
										 |  |  |     /* The `vertex_len` parameter is only used for asserts in the Debug build. */ | 
					
						
							|  |  |  |     /* Not very useful in python since scripts are often tested in Release build. */ | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |     /* Use `INT_MAX` instead of the actual number of vertices. */ | 
					
						
							| 
									
										
										
										
											2021-02-22 08:26:45 -03:00
										 |  |  |     GPU_indexbuf_init(&builder, prim_type.value_found, index_len, INT_MAX); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |     if (verts_per_prim == 1) { | 
					
						
							|  |  |  |       for (uint i = 0; i < seq_len; i++) { | 
					
						
							|  |  |  |         GPU_indexbuf_add_generic_vert(&builder, PyC_Long_AsU32(seq_items[i])); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2018-10-10 13:35:29 -03:00
										 |  |  |       int values[4]; | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |       for (uint i = 0; i < seq_len; i++) { | 
					
						
							| 
									
										
										
										
											2018-10-10 13:35:29 -03:00
										 |  |  |         PyObject *seq_fast_item = PySequence_Fast(seq_items[i], error_prefix); | 
					
						
							|  |  |  |         if (seq_fast_item == NULL) { | 
					
						
							|  |  |  |           PyErr_Format(PyExc_TypeError, | 
					
						
							|  |  |  |                        "%s: expected a sequence, got %s", | 
					
						
							|  |  |  |                        error_prefix, | 
					
						
							|  |  |  |                        Py_TYPE(seq_items[i])->tp_name); | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |           ok = false; | 
					
						
							|  |  |  |           goto finally; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-27 22:26:33 +10:00
										 |  |  |         ok = PyC_AsArray_FAST(values, | 
					
						
							|  |  |  |                               sizeof(*values), | 
					
						
							|  |  |  |                               seq_fast_item, | 
					
						
							|  |  |  |                               verts_per_prim, | 
					
						
							|  |  |  |                               &PyLong_Type, | 
					
						
							|  |  |  |                               error_prefix) == 0; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-10 13:35:29 -03:00
										 |  |  |         if (ok) { | 
					
						
							|  |  |  |           for (uint j = 0; j < verts_per_prim; j++) { | 
					
						
							|  |  |  |             GPU_indexbuf_add_generic_vert(&builder, values[j]); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |           } | 
					
						
							| 
									
										
										
										
											2018-10-10 13:35:29 -03:00
										 |  |  |         } | 
					
						
							|  |  |  |         Py_DECREF(seq_fast_item); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |     if (PyErr_Occurred()) { | 
					
						
							|  |  |  |       ok = false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |   finally: | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |     Py_DECREF(seq_fast); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |   if (ok == false) { | 
					
						
							|  |  |  |     MEM_freeN(builder.data); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |   return BPyGPUIndexBuf_CreatePyObject(GPU_indexbuf_build(&builder)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-17 10:16:41 -03:00
										 |  |  | static void pygpu_IndexBuf__tp_dealloc(BPyGPUIndexBuf *self) | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  | { | 
					
						
							|  |  |  |   GPU_indexbuf_discard(self->elem); | 
					
						
							|  |  |  |   Py_TYPE(self)->tp_free(self); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-17 10:16:41 -03:00
										 |  |  | PyDoc_STRVAR(pygpu_IndexBuf__tp_doc, | 
					
						
							| 
									
										
										
										
											2018-11-13 14:55:15 +01:00
										 |  |  |              ".. class:: GPUIndexBuf(type, seq)\n" | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |              "\n" | 
					
						
							| 
									
										
										
										
											2018-11-14 09:04:24 +11:00
										 |  |  |              "   Contains an index buffer.\n" | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |              "\n" | 
					
						
							| 
									
										
										
										
											2021-03-16 12:48:00 -03:00
										 |  |  |              "   :arg type: The primitive type this index buffer is composed of.\n" | 
					
						
							|  |  |  |              "      Possible values are `POINTS`, `LINES`, `TRIS` and `LINE_STRIP_ADJ`.\n" | 
					
						
							|  |  |  |              "   :type type: str\n" | 
					
						
							| 
									
										
										
										
											2018-11-14 09:04:24 +11:00
										 |  |  |              "   :param seq: Indices this index buffer will contain.\n" | 
					
						
							|  |  |  |              "      Whether a 1D or 2D sequence is required depends on the type.\n" | 
					
						
							|  |  |  |              "      Optionally the sequence can support the buffer protocol.\n" | 
					
						
							|  |  |  |              "   :type seq: 1D or 2D sequence\n"); | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  | PyTypeObject BPyGPUIndexBuf_Type = { | 
					
						
							|  |  |  |     PyVarObject_HEAD_INIT(NULL, 0).tp_name = "GPUIndexBuf", | 
					
						
							|  |  |  |     .tp_basicsize = sizeof(BPyGPUIndexBuf), | 
					
						
							| 
									
										
										
										
											2021-02-17 10:16:41 -03:00
										 |  |  |     .tp_dealloc = (destructor)pygpu_IndexBuf__tp_dealloc, | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  |     .tp_flags = Py_TPFLAGS_DEFAULT, | 
					
						
							| 
									
										
										
										
											2021-02-17 10:16:41 -03:00
										 |  |  |     .tp_doc = pygpu_IndexBuf__tp_doc, | 
					
						
							|  |  |  |     .tp_new = pygpu_IndexBuf__tp_new, | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Public API
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PyObject *BPyGPUIndexBuf_CreatePyObject(GPUIndexBuf *elem) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   BPyGPUIndexBuf *self; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   self = PyObject_New(BPyGPUIndexBuf, &BPyGPUIndexBuf_Type); | 
					
						
							|  |  |  |   self->elem = elem; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return (PyObject *)self; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \} */ |