This topic is intended for add-on developers who want to port their code to use the gpu module.
As discussed in BGL deprecation, the bgl module is at the end of its days and should be removed to give free way to a switchable drawing API in Blender (Vulkan + OpenGL).
To this end, patches are being proposed to extend the gpu API and make it possible for the bgl API to be replaced:
Could we have a way to render depth buffer into a texture with GPU python modules? Now we can render the 3D View into a texture with “offscreen.draw_view3d” and “offscreen.color_texture”. Could we have something like “offscreen.depth_texture”?
I find in Blender 2.92 glReadPixels will always produce 0.0 for bgl.GL_DEPTH_COMPONENT, while it works well with bgl.GL_RGBA. Making the depth buffer of 3DViewport accessible really helps a lot when adding some effects. Thanks a million!
In fact “offscreen.color_texture” needs to be updated (and “offscreen.depth_texture” will be a good addition).
Currently “offscreen.color_texture” returns a bind value for OpenGL. So it must be deprecated.
Just want to chime in and say +1 to being able to access the depth buffer. Also, is there any indication yet as to what will replace this after the current implementation is deprecated?
We have the GPUFramebufer.read_color, so we could have a GPUFramebufer.read_depth.
But only if it is really necessary to port the current addons as it can limit the vulkan’s implementation.
Deprecation of bgl module in switching to Vulcan is great. Current implementation of the gpu module is lacking though. In particular, use of Buffer or GPUTexture outside this module is pretty bad.
I’ll just explain the my use-case to put everything into some context. So previously I used pyopengl library to generate a set of images and put them inside blender. And it was fine because final read of buffer returns numpy array which is easily turned into flat array of pixels to pass to pixels.foreach_set() and push pixels to blender image. With new gpu module and implementation of framebuffer object I decided to switch to it and get rid of peopengl dependency. The problem is that GPUTexture.read() returns a Buffer object which is shaped according to texture width, height and color depth. Passing this buffer to pixels of some blender image is non-trivial. The problem is that you have to flatten this array. And all the methods that I’ve tried a super slow and take at least 2 second for a simple 1024*1024 32-bit image.
The main bottle-neck seems to be Buffer.to_list() method which returns just a list of lists of pixels and takes around a second. But again this list of lists has to be flattened. numpy.ravel or numpy.flatten are very slow for some reason. Generating a list by stepping through all the indicies is also very slow.
So the whole drawing part takes a hundredth of a second while copying results into blender image pixels takes several seconds which is terrible.
So I wonder if there are any plans to integrate gpu module more tightly into blender. Or maybe I’m overlooking some simple solution which you can maybe share
I also wonder, whether this can be still a part of 2.93 release. It would be great if I could release an addon without forcing users to use 3.0
To ‘read’ any part of 3d-viewport (using glReadPixels) to 2d buf - which later can be saved to img.
Anyone know how to redo this using only gpu module?
There is offscreen.draw_view3d() - but it requires 3d camera to be present at scene for projection_matrix - and as it is mentioned in example , there is no way to get projection_matrix without cam.
I forgot to mention I wanted to render view_buffer to image in blender 2.93.
But even in latest Blender 3.0 I have trouble to get the code to work (I get black image when reading framebuffer with read_color).
Here is the simple operator script that I made to render current 3d view to image: ViewportBufferToImg_b30.txt (1.5 KB)
Im not familiar with OpengGL stuff so I’m probably doing something wrong… Any help would be appreciated
GPUOffScreen is like an empty screen that can be drawn in the background.
If nothing is drawn it is black.
It has its own GPUFramebuffer. GPUFramebuffer is a container for GPUTextures. (A texture can be used by more than one framebuffer).
Blender has a framebuffer for each viewport.
If you call gpu.state.active_framebuffer_get() inside a draw callback, you can access these framebuffers.
I am happy to see that GPUFrameBuffer will get a function read_color (*x*, *y*, *xsize*, *ysize*, *channels*, *slot*, *format*, *data=data* ) which allows us to read a rectangle from a FrameBuffer texture.
Question: Is there also a function to write a rectangle into a texture or FrameBuffer object? Specifically I’d like to have the functionality of glBlitFrameBuffer