198 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			198 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* SPDX-License-Identifier: GPL-2.0-or-later */
 | |
| 
 | |
| /** \file
 | |
|  * \ingroup bpygpu
 | |
|  *
 | |
|  * This file defines the uniform buffer functionalities of the 'gpu' module
 | |
|  *
 | |
|  * - Use `bpygpu_` for local API.
 | |
|  * - Use `BPyGPU` for public API.
 | |
|  */
 | |
| 
 | |
| #include <Python.h>
 | |
| 
 | |
| #include "BLI_string.h"
 | |
| 
 | |
| #include "GPU_context.h"
 | |
| #include "GPU_uniform_buffer.h"
 | |
| 
 | |
| #include "../generic/py_capi_utils.h"
 | |
| 
 | |
| #include "gpu_py.h"
 | |
| 
 | |
| #include "gpu_py_uniformbuffer.h" /* own include */
 | |
| 
 | |
| /* -------------------------------------------------------------------- */
 | |
| /** \name GPUUniformBuf Common Utilities
 | |
|  * \{ */
 | |
| 
 | |
| static int pygpu_uniformbuffer_valid_check(BPyGPUUniformBuf *bpygpu_ub)
 | |
| {
 | |
|   if (UNLIKELY(bpygpu_ub->ubo == NULL)) {
 | |
|     PyErr_SetString(PyExc_ReferenceError,
 | |
| #ifdef BPYGPU_USE_GPUOBJ_FREE_METHOD
 | |
|                     "GPU uniform buffer was freed, no further access is valid");
 | |
| #else
 | |
| 
 | |
|                     "GPU uniform buffer: internal error");
 | |
| #endif
 | |
|     return -1;
 | |
|   }
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| #define BPYGPU_UNIFORMBUF_CHECK_OBJ(bpygpu) \
 | |
|   { \
 | |
|     if (UNLIKELY(pygpu_uniformbuffer_valid_check(bpygpu) == -1)) { \
 | |
|       return NULL; \
 | |
|     } \
 | |
|   } \
 | |
|   ((void)0)
 | |
| 
 | |
| /** \} */
 | |
| 
 | |
| /* -------------------------------------------------------------------- */
 | |
| /** \name GPUUniformBuf Type
 | |
|  * \{ */
 | |
| 
 | |
| static PyObject *pygpu_uniformbuffer__tp_new(PyTypeObject *UNUSED(self),
 | |
|                                              PyObject *args,
 | |
|                                              PyObject *kwds)
 | |
| {
 | |
|   BPYGPU_IS_INIT_OR_ERROR_OBJ;
 | |
| 
 | |
|   GPUUniformBuf *ubo = NULL;
 | |
|   PyObject *pybuffer_obj;
 | |
|   char err_out[256] = "unknown error. See console";
 | |
| 
 | |
|   static const char *_keywords[] = {"data", NULL};
 | |
|   static _PyArg_Parser _parser = {
 | |
|       "O" /* `data` */
 | |
|       ":GPUUniformBuf.__new__",
 | |
|       _keywords,
 | |
|       0,
 | |
|   };
 | |
|   if (!_PyArg_ParseTupleAndKeywordsFast(args, kwds, &_parser, &pybuffer_obj)) {
 | |
|     return NULL;
 | |
|   }
 | |
| 
 | |
|   if (!GPU_context_active_get()) {
 | |
|     STRNCPY(err_out, "No active GPU context found");
 | |
|   }
 | |
|   else {
 | |
|     Py_buffer pybuffer;
 | |
|     if (PyObject_GetBuffer(pybuffer_obj, &pybuffer, PyBUF_SIMPLE) == -1) {
 | |
|       /* PyObject_GetBuffer raise a PyExc_BufferError */
 | |
|       return NULL;
 | |
|     }
 | |
| 
 | |
|     if ((pybuffer.len % 16) != 0) {
 | |
|       STRNCPY(err_out, "UBO is not padded to size of vec4");
 | |
|     }
 | |
|     else {
 | |
|       ubo = GPU_uniformbuf_create_ex(pybuffer.len, pybuffer.buf, "python_uniformbuffer");
 | |
|     }
 | |
|     PyBuffer_Release(&pybuffer);
 | |
|   }
 | |
| 
 | |
|   if (ubo == NULL) {
 | |
|     PyErr_Format(PyExc_RuntimeError, "GPUUniformBuf.__new__(...) failed with '%s'", err_out);
 | |
|     return NULL;
 | |
|   }
 | |
| 
 | |
|   return BPyGPUUniformBuf_CreatePyObject(ubo);
 | |
| }
 | |
| 
 | |
| PyDoc_STRVAR(pygpu_uniformbuffer_update_doc,
 | |
|              ".. method:: update(data)\n"
 | |
|              "\n"
 | |
|              "   Update the data of the uniform buffer object.\n");
 | |
| static PyObject *pygpu_uniformbuffer_update(BPyGPUUniformBuf *self, PyObject *obj)
 | |
| {
 | |
|   BPYGPU_UNIFORMBUF_CHECK_OBJ(self);
 | |
| 
 | |
|   Py_buffer pybuffer;
 | |
|   if (PyObject_GetBuffer(obj, &pybuffer, PyBUF_SIMPLE) == -1) {
 | |
|     /* PyObject_GetBuffer raise a PyExc_BufferError */
 | |
|     return NULL;
 | |
|   }
 | |
| 
 | |
|   GPU_uniformbuf_update(self->ubo, pybuffer.buf);
 | |
|   PyBuffer_Release(&pybuffer);
 | |
|   Py_RETURN_NONE;
 | |
| }
 | |
| 
 | |
| #ifdef BPYGPU_USE_GPUOBJ_FREE_METHOD
 | |
| PyDoc_STRVAR(pygpu_uniformbuffer_free_doc,
 | |
|              ".. method::free()\n"
 | |
|              "\n"
 | |
|              "   Free the uniform buffer object.\n"
 | |
|              "   The uniform buffer object will no longer be accessible.\n");
 | |
| static PyObject *pygpu_uniformbuffer_free(BPyGPUUniformBuf *self)
 | |
| {
 | |
|   BPYGPU_UNIFORMBUF_CHECK_OBJ(self);
 | |
| 
 | |
|   GPU_uniformbuf_free(self->ubo);
 | |
|   self->ubo = NULL;
 | |
|   Py_RETURN_NONE;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| static void BPyGPUUniformBuf__tp_dealloc(BPyGPUUniformBuf *self)
 | |
| {
 | |
|   if (self->ubo) {
 | |
|     GPU_uniformbuf_free(self->ubo);
 | |
|   }
 | |
|   Py_TYPE(self)->tp_free((PyObject *)self);
 | |
| }
 | |
| 
 | |
| static PyGetSetDef pygpu_uniformbuffer__tp_getseters[] = {
 | |
|     {NULL, NULL, NULL, NULL, NULL} /* Sentinel */
 | |
| };
 | |
| 
 | |
| static struct PyMethodDef pygpu_uniformbuffer__tp_methods[] = {
 | |
|     {"update", (PyCFunction)pygpu_uniformbuffer_update, METH_O, pygpu_uniformbuffer_update_doc},
 | |
| #ifdef BPYGPU_USE_GPUOBJ_FREE_METHOD
 | |
|     {"free", (PyCFunction)pygpu_uniformbuffer_free, METH_NOARGS, pygpu_uniformbuffer_free_doc},
 | |
| #endif
 | |
|     {NULL, NULL, 0, NULL},
 | |
| };
 | |
| 
 | |
| PyDoc_STRVAR(pygpu_uniformbuffer__tp_doc,
 | |
|              ".. class:: GPUUniformBuf(data)\n"
 | |
|              "\n"
 | |
|              "   This object gives access to off uniform buffers.\n"
 | |
|              "\n"
 | |
|              "   :arg data: Data to fill the buffer.\n"
 | |
|              "   :type data: object exposing buffer interface\n");
 | |
| PyTypeObject BPyGPUUniformBuf_Type = {
 | |
|     PyVarObject_HEAD_INIT(NULL, 0).tp_name = "GPUUniformBuf",
 | |
|     .tp_basicsize = sizeof(BPyGPUUniformBuf),
 | |
|     .tp_dealloc = (destructor)BPyGPUUniformBuf__tp_dealloc,
 | |
|     .tp_flags = Py_TPFLAGS_DEFAULT,
 | |
|     .tp_doc = pygpu_uniformbuffer__tp_doc,
 | |
|     .tp_methods = pygpu_uniformbuffer__tp_methods,
 | |
|     .tp_getset = pygpu_uniformbuffer__tp_getseters,
 | |
|     .tp_new = pygpu_uniformbuffer__tp_new,
 | |
| };
 | |
| 
 | |
| /** \} */
 | |
| 
 | |
| /* -------------------------------------------------------------------- */
 | |
| /** \name Public API
 | |
|  * \{ */
 | |
| 
 | |
| PyObject *BPyGPUUniformBuf_CreatePyObject(GPUUniformBuf *ubo)
 | |
| {
 | |
|   BPyGPUUniformBuf *self;
 | |
| 
 | |
|   self = PyObject_New(BPyGPUUniformBuf, &BPyGPUUniformBuf_Type);
 | |
|   self->ubo = ubo;
 | |
| 
 | |
|   return (PyObject *)self;
 | |
| }
 | |
| 
 | |
| /** \} */
 | |
| 
 | |
| #undef BPYGPU_UNIFORMBUF_CHECK_OBJ
 |