| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU General Public License | 
					
						
							|  |  |  |  * as published by the Free Software Foundation; either version 2 | 
					
						
							|  |  |  |  * of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  * GNU General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  * along with this program; if not, write to the Free Software Foundation, | 
					
						
							|  |  |  |  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The Original Code is Copyright (C) 2005 Blender Foundation. | 
					
						
							|  |  |  |  * All rights reserved. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-18 11:23:17 +11:00
										 |  |  | /** \file
 | 
					
						
							|  |  |  |  * \ingroup gpu | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | #include "BLI_string.h"
 | 
					
						
							| 
									
										
										
										
											2020-02-15 12:17:39 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-19 15:48:13 +02:00
										 |  |  | #include "GPU_framebuffer.h"
 | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | #include "GPU_texture.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | #include "gpu_backend.hh"
 | 
					
						
							| 
									
										
										
										
											2020-08-08 03:01:45 +02:00
										 |  |  | #include "gpu_context_private.hh"
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | #include "gpu_framebuffer_private.hh"
 | 
					
						
							| 
									
										
										
										
											2020-09-06 16:40:07 +02:00
										 |  |  | #include "gpu_vertex_buffer_private.hh"
 | 
					
						
							| 
									
										
										
										
											2018-07-19 15:48:13 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-02 01:25:32 +02:00
										 |  |  | #include "gpu_texture_private.hh"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace blender::gpu { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Creation & Deletion
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Texture::Texture(const char *name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (name) { | 
					
						
							|  |  |  |     BLI_strncpy(name_, name, sizeof(name_)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     name_[0] = '\0'; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:37:01 +02:00
										 |  |  |   for (int i = 0; i < ARRAY_SIZE(fb_); i++) { | 
					
						
							| 
									
										
										
										
											2020-11-06 17:49:09 +01:00
										 |  |  |     fb_[i] = nullptr; | 
					
						
							| 
									
										
										
										
											2020-03-24 12:09:39 -03:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-05-04 21:22:41 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | Texture::~Texture() | 
					
						
							| 
									
										
										
										
											2018-12-11 21:14:52 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:37:01 +02:00
										 |  |  |   for (int i = 0; i < ARRAY_SIZE(fb_); i++) { | 
					
						
							| 
									
										
										
										
											2020-11-06 17:49:09 +01:00
										 |  |  |     if (fb_[i] != nullptr) { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:37:01 +02:00
										 |  |  |       fb_[i]->attachment_remove(fb_attachment_[i]); | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-29 14:50:20 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-04-29 14:48:59 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-30 11:20:39 -03:00
										 |  |  | #ifndef GPU_NO_USE_PY_REFERENCES
 | 
					
						
							| 
									
										
										
										
											2021-04-29 14:48:59 -03:00
										 |  |  |   if (this->py_ref) { | 
					
						
							|  |  |  |     *this->py_ref = nullptr; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2018-12-11 21:14:52 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | bool Texture::init_1D(int w, int layers, eGPUTextureFormat format) | 
					
						
							| 
									
										
										
										
											2020-09-02 00:27:41 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   w_ = w; | 
					
						
							|  |  |  |   h_ = layers; | 
					
						
							|  |  |  |   d_ = 0; | 
					
						
							|  |  |  |   format_ = format; | 
					
						
							|  |  |  |   format_flag_ = to_format_flag(format); | 
					
						
							|  |  |  |   type_ = (layers > 0) ? GPU_TEXTURE_1D_ARRAY : GPU_TEXTURE_1D; | 
					
						
							|  |  |  |   if ((format_flag_ & (GPU_FORMAT_DEPTH_STENCIL | GPU_FORMAT_INTEGER)) == 0) { | 
					
						
							|  |  |  |     sampler_state = GPU_SAMPLER_FILTER; | 
					
						
							| 
									
										
										
										
											2018-07-10 13:17:32 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   return this->init_internal(); | 
					
						
							| 
									
										
										
										
											2018-07-10 13:17:32 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | bool Texture::init_2D(int w, int h, int layers, eGPUTextureFormat format) | 
					
						
							| 
									
										
										
										
											2020-03-11 17:12:01 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   w_ = w; | 
					
						
							|  |  |  |   h_ = h; | 
					
						
							|  |  |  |   d_ = layers; | 
					
						
							|  |  |  |   format_ = format; | 
					
						
							|  |  |  |   format_flag_ = to_format_flag(format); | 
					
						
							|  |  |  |   type_ = (layers > 0) ? GPU_TEXTURE_2D_ARRAY : GPU_TEXTURE_2D; | 
					
						
							|  |  |  |   if ((format_flag_ & (GPU_FORMAT_DEPTH_STENCIL | GPU_FORMAT_INTEGER)) == 0) { | 
					
						
							|  |  |  |     sampler_state = GPU_SAMPLER_FILTER; | 
					
						
							| 
									
										
										
										
											2020-03-11 17:12:01 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   return this->init_internal(); | 
					
						
							| 
									
										
										
										
											2020-03-11 17:12:01 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | bool Texture::init_3D(int w, int h, int d, eGPUTextureFormat format) | 
					
						
							| 
									
										
										
										
											2018-07-10 13:17:32 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   w_ = w; | 
					
						
							|  |  |  |   h_ = h; | 
					
						
							|  |  |  |   d_ = d; | 
					
						
							|  |  |  |   format_ = format; | 
					
						
							|  |  |  |   format_flag_ = to_format_flag(format); | 
					
						
							|  |  |  |   type_ = GPU_TEXTURE_3D; | 
					
						
							|  |  |  |   if ((format_flag_ & (GPU_FORMAT_DEPTH_STENCIL | GPU_FORMAT_INTEGER)) == 0) { | 
					
						
							|  |  |  |     sampler_state = GPU_SAMPLER_FILTER; | 
					
						
							| 
									
										
										
										
											2018-07-10 13:17:32 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   return this->init_internal(); | 
					
						
							| 
									
										
										
										
											2018-07-10 13:17:32 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | bool Texture::init_cubemap(int w, int layers, eGPUTextureFormat format) | 
					
						
							| 
									
										
										
										
											2018-07-10 13:17:32 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   w_ = w; | 
					
						
							|  |  |  |   h_ = w; | 
					
						
							|  |  |  |   d_ = max_ii(1, layers) * 6; | 
					
						
							|  |  |  |   format_ = format; | 
					
						
							|  |  |  |   format_flag_ = to_format_flag(format); | 
					
						
							|  |  |  |   type_ = (layers > 0) ? GPU_TEXTURE_CUBE_ARRAY : GPU_TEXTURE_CUBE; | 
					
						
							|  |  |  |   if ((format_flag_ & (GPU_FORMAT_DEPTH_STENCIL | GPU_FORMAT_INTEGER)) == 0) { | 
					
						
							|  |  |  |     sampler_state = GPU_SAMPLER_FILTER; | 
					
						
							| 
									
										
										
										
											2020-08-07 12:39:35 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   return this->init_internal(); | 
					
						
							| 
									
										
										
										
											2018-07-10 13:17:32 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | bool Texture::init_buffer(GPUVertBuf *vbo, eGPUTextureFormat format) | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   /* See to_texture_format(). */ | 
					
						
							|  |  |  |   if (format == GPU_DEPTH_COMPONENT24) { | 
					
						
							|  |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2017-02-03 16:01:32 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-09-06 23:45:51 +02:00
										 |  |  |   w_ = GPU_vertbuf_get_vertex_len(vbo); | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   h_ = 0; | 
					
						
							|  |  |  |   d_ = 0; | 
					
						
							|  |  |  |   format_ = format; | 
					
						
							|  |  |  |   format_flag_ = to_format_flag(format); | 
					
						
							|  |  |  |   type_ = GPU_TEXTURE_BUFFER; | 
					
						
							|  |  |  |   return this->init_internal(vbo); | 
					
						
							| 
									
										
										
										
											2017-02-03 16:01:32 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | /** \} */ | 
					
						
							| 
									
										
										
										
											2018-04-30 16:02:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Operation
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:37:01 +02:00
										 |  |  | void Texture::attach_to(FrameBuffer *fb, GPUAttachmentType type) | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:37:01 +02:00
										 |  |  |   for (int i = 0; i < ARRAY_SIZE(fb_); i++) { | 
					
						
							| 
									
										
										
										
											2020-11-06 17:49:09 +01:00
										 |  |  |     if (fb_[i] == nullptr) { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:37:01 +02:00
										 |  |  |       fb_attachment_[i] = type; | 
					
						
							|  |  |  |       fb_[i] = fb; | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   BLI_assert(!"GPU: Error: Texture: Not enough attachment"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Texture::detach_from(FrameBuffer *fb) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   for (int i = 0; i < ARRAY_SIZE(fb_); i++) { | 
					
						
							|  |  |  |     if (fb_[i] == fb) { | 
					
						
							|  |  |  |       fb_[i]->attachment_remove(fb_attachment_[i]); | 
					
						
							| 
									
										
										
										
											2020-11-06 17:49:09 +01:00
										 |  |  |       fb_[i] = nullptr; | 
					
						
							| 
									
										
										
										
											2020-09-05 17:37:01 +02:00
										 |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   BLI_assert(!"GPU: Error: Texture: Framebuffer is not attached"); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | void Texture::update(eGPUDataFormat format, const void *data) | 
					
						
							| 
									
										
										
										
											2018-12-14 21:30:50 -02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   int mip = 0; | 
					
						
							|  |  |  |   int extent[3], offset[3] = {0, 0, 0}; | 
					
						
							|  |  |  |   this->mip_size_get(mip, extent); | 
					
						
							|  |  |  |   this->update_sub(mip, offset, extent, format, data); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-08-11 20:06:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | /** \} */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | }  // namespace blender::gpu
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name C-API
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | using namespace blender; | 
					
						
							|  |  |  | using namespace blender::gpu; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:33:56 +02:00
										 |  |  | /* ------ Memory Management ------ */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | uint GPU_texture_memory_usage_get(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-19 14:32:41 +10:00
										 |  |  |   /* TODO(fclem): Do that inside the new Texture class. */ | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   return 0; | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-08 03:22:35 +02:00
										 |  |  | /* ------ Creation ------ */ | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:33:56 +02:00
										 |  |  | static inline GPUTexture *gpu_texture_create(const char *name, | 
					
						
							|  |  |  |                                              const int w, | 
					
						
							|  |  |  |                                              const int h, | 
					
						
							|  |  |  |                                              const int d, | 
					
						
							|  |  |  |                                              const eGPUTextureType type, | 
					
						
							|  |  |  |                                              int UNUSED(mips), | 
					
						
							|  |  |  |                                              eGPUTextureFormat tex_format, | 
					
						
							| 
									
										
										
										
											2020-09-15 21:21:14 +05:30
										 |  |  |                                              eGPUDataFormat data_format, | 
					
						
							|  |  |  |                                              const void *pixels) | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:33:56 +02:00
										 |  |  |   Texture *tex = GPUBackend::get()->texture_alloc(name); | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   bool success = false; | 
					
						
							| 
									
										
										
										
											2020-09-05 17:33:56 +02:00
										 |  |  |   switch (type) { | 
					
						
							|  |  |  |     case GPU_TEXTURE_1D: | 
					
						
							|  |  |  |     case GPU_TEXTURE_1D_ARRAY: | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |       success = tex->init_1D(w, h, tex_format); | 
					
						
							|  |  |  |       break; | 
					
						
							| 
									
										
										
										
											2020-09-05 17:33:56 +02:00
										 |  |  |     case GPU_TEXTURE_2D: | 
					
						
							|  |  |  |     case GPU_TEXTURE_2D_ARRAY: | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |       success = tex->init_2D(w, h, d, tex_format); | 
					
						
							|  |  |  |       break; | 
					
						
							| 
									
										
										
										
											2020-09-05 17:33:56 +02:00
										 |  |  |     case GPU_TEXTURE_3D: | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |       success = tex->init_3D(w, h, d, tex_format); | 
					
						
							|  |  |  |       break; | 
					
						
							| 
									
										
										
										
											2020-09-05 17:33:56 +02:00
										 |  |  |     case GPU_TEXTURE_CUBE: | 
					
						
							|  |  |  |     case GPU_TEXTURE_CUBE_ARRAY: | 
					
						
							|  |  |  |       success = tex->init_cubemap(w, d, tex_format); | 
					
						
							|  |  |  |       break; | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |     default: | 
					
						
							|  |  |  |       break; | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   if (!success) { | 
					
						
							|  |  |  |     delete tex; | 
					
						
							| 
									
										
										
										
											2020-11-06 17:49:09 +01:00
										 |  |  |     return nullptr; | 
					
						
							| 
									
										
										
										
											2017-02-03 16:01:32 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-09-15 21:21:14 +05:30
										 |  |  |   if (pixels) { | 
					
						
							|  |  |  |     tex->update(data_format, pixels); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   return reinterpret_cast<GPUTexture *>(tex); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:33:56 +02:00
										 |  |  | GPUTexture *GPU_texture_create_1d( | 
					
						
							|  |  |  |     const char *name, int w, int mips, eGPUTextureFormat format, const float *data) | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-15 21:21:14 +05:30
										 |  |  |   return gpu_texture_create(name, w, 0, 0, GPU_TEXTURE_1D, mips, format, GPU_DATA_FLOAT, data); | 
					
						
							| 
									
										
										
										
											2020-09-05 17:33:56 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:33:56 +02:00
										 |  |  | GPUTexture *GPU_texture_create_1d_array( | 
					
						
							|  |  |  |     const char *name, int w, int h, int mips, eGPUTextureFormat format, const float *data) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-15 21:21:14 +05:30
										 |  |  |   return gpu_texture_create( | 
					
						
							|  |  |  |       name, w, h, 0, GPU_TEXTURE_1D_ARRAY, mips, format, GPU_DATA_FLOAT, data); | 
					
						
							| 
									
										
										
										
											2020-09-05 17:33:56 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GPUTexture *GPU_texture_create_2d( | 
					
						
							|  |  |  |     const char *name, int w, int h, int mips, eGPUTextureFormat format, const float *data) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-15 21:21:14 +05:30
										 |  |  |   return gpu_texture_create(name, w, h, 0, GPU_TEXTURE_2D, mips, format, GPU_DATA_FLOAT, data); | 
					
						
							| 
									
										
										
										
											2020-09-05 17:33:56 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GPUTexture *GPU_texture_create_2d_array( | 
					
						
							|  |  |  |     const char *name, int w, int h, int d, int mips, eGPUTextureFormat format, const float *data) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-15 21:21:14 +05:30
										 |  |  |   return gpu_texture_create( | 
					
						
							|  |  |  |       name, w, h, d, GPU_TEXTURE_2D_ARRAY, mips, format, GPU_DATA_FLOAT, data); | 
					
						
							| 
									
										
										
										
											2020-09-05 17:33:56 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-15 21:21:14 +05:30
										 |  |  | GPUTexture *GPU_texture_create_3d(const char *name, | 
					
						
							|  |  |  |                                   int w, | 
					
						
							|  |  |  |                                   int h, | 
					
						
							|  |  |  |                                   int d, | 
					
						
							|  |  |  |                                   int mips, | 
					
						
							|  |  |  |                                   eGPUTextureFormat texture_format, | 
					
						
							|  |  |  |                                   eGPUDataFormat data_format, | 
					
						
							|  |  |  |                                   const void *data) | 
					
						
							| 
									
										
										
										
											2020-09-05 17:33:56 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-15 21:21:14 +05:30
										 |  |  |   return gpu_texture_create( | 
					
						
							|  |  |  |       name, w, h, d, GPU_TEXTURE_3D, mips, texture_format, data_format, data); | 
					
						
							| 
									
										
										
										
											2020-09-05 17:33:56 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GPUTexture *GPU_texture_create_cube( | 
					
						
							|  |  |  |     const char *name, int w, int mips, eGPUTextureFormat format, const float *data) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-15 21:21:14 +05:30
										 |  |  |   return gpu_texture_create(name, w, w, 0, GPU_TEXTURE_CUBE, mips, format, GPU_DATA_FLOAT, data); | 
					
						
							| 
									
										
										
										
											2020-09-05 17:33:56 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GPUTexture *GPU_texture_create_cube_array( | 
					
						
							|  |  |  |     const char *name, int w, int d, int mips, eGPUTextureFormat format, const float *data) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-15 21:21:14 +05:30
										 |  |  |   return gpu_texture_create( | 
					
						
							|  |  |  |       name, w, w, d, GPU_TEXTURE_CUBE_ARRAY, mips, format, GPU_DATA_FLOAT, data); | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | /* DDS texture loading. Return NULL if support is not available. */ | 
					
						
							| 
									
										
										
										
											2020-09-05 17:33:56 +02:00
										 |  |  | GPUTexture *GPU_texture_create_compressed_2d( | 
					
						
							|  |  |  |     const char *name, int w, int h, int miplen, eGPUTextureFormat tex_format, const void *data) | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:33:56 +02:00
										 |  |  |   Texture *tex = GPUBackend::get()->texture_alloc(name); | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   bool success = tex->init_2D(w, h, 0, tex_format); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   if (!success) { | 
					
						
							|  |  |  |     delete tex; | 
					
						
							| 
									
										
										
										
											2020-11-06 17:49:09 +01:00
										 |  |  |     return nullptr; | 
					
						
							| 
									
										
										
										
											2018-05-13 22:12:32 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   if (data) { | 
					
						
							|  |  |  |     size_t ofs = 0; | 
					
						
							|  |  |  |     for (int mip = 0; mip < miplen; mip++) { | 
					
						
							|  |  |  |       int extent[3], offset[3] = {0, 0, 0}; | 
					
						
							|  |  |  |       tex->mip_size_get(mip, extent); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |       size_t size = ((extent[0] + 3) / 4) * ((extent[1] + 3) / 4) * to_block_size(tex_format); | 
					
						
							|  |  |  |       tex->update_sub(mip, offset, extent, to_data_format(tex_format), (uchar *)data + ofs); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |       ofs += size; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-05-13 22:12:32 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   return reinterpret_cast<GPUTexture *>(tex); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:33:56 +02:00
										 |  |  | GPUTexture *GPU_texture_create_from_vertbuf(const char *name, GPUVertBuf *vert) | 
					
						
							| 
									
										
										
										
											2020-07-26 20:17:01 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-06 23:45:51 +02:00
										 |  |  |   eGPUTextureFormat tex_format = to_texture_format(GPU_vertbuf_get_format(vert)); | 
					
						
							| 
									
										
										
										
											2020-09-05 17:33:56 +02:00
										 |  |  |   Texture *tex = GPUBackend::get()->texture_alloc(name); | 
					
						
							| 
									
										
										
										
											2020-07-29 04:55:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   bool success = tex->init_buffer(vert, tex_format); | 
					
						
							|  |  |  |   if (!success) { | 
					
						
							|  |  |  |     delete tex; | 
					
						
							| 
									
										
										
										
											2020-11-06 17:49:09 +01:00
										 |  |  |     return nullptr; | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   return reinterpret_cast<GPUTexture *>(tex); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:33:56 +02:00
										 |  |  | /* Create an error texture that will bind an invalid texture (pink) at draw time. */ | 
					
						
							|  |  |  | GPUTexture *GPU_texture_create_error(int dimension, bool is_array) | 
					
						
							| 
									
										
										
										
											2017-02-03 16:01:32 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:33:56 +02:00
										 |  |  |   float pixel[4] = {1.0f, 0.0f, 1.0f, 1.0f}; | 
					
						
							|  |  |  |   int w = 1; | 
					
						
							|  |  |  |   int h = (dimension < 2 && !is_array) ? 0 : 1; | 
					
						
							|  |  |  |   int d = (dimension < 3 && !is_array) ? 0 : 1; | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:33:56 +02:00
										 |  |  |   eGPUTextureType type = GPU_TEXTURE_3D; | 
					
						
							|  |  |  |   type = (dimension == 2) ? (is_array ? GPU_TEXTURE_2D_ARRAY : GPU_TEXTURE_2D) : type; | 
					
						
							|  |  |  |   type = (dimension == 1) ? (is_array ? GPU_TEXTURE_1D_ARRAY : GPU_TEXTURE_1D) : type; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-15 21:21:14 +05:30
										 |  |  |   return gpu_texture_create("invalid_tex", w, h, d, type, 1, GPU_RGBA8, GPU_DATA_FLOAT, pixel); | 
					
						
							| 
									
										
										
										
											2017-11-13 23:25:46 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-08 03:22:35 +02:00
										 |  |  | /* ------ Update ------ */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:33:56 +02:00
										 |  |  | void GPU_texture_update_mipmap(GPUTexture *tex_, | 
					
						
							|  |  |  |                                int miplvl, | 
					
						
							| 
									
										
										
										
											2020-09-07 13:59:22 +02:00
										 |  |  |                                eGPUDataFormat data_format, | 
					
						
							| 
									
										
										
										
											2020-09-05 17:33:56 +02:00
										 |  |  |                                const void *pixels) | 
					
						
							| 
									
										
										
										
											2018-07-10 13:17:32 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   Texture *tex = reinterpret_cast<Texture *>(tex_); | 
					
						
							|  |  |  |   int extent[3] = {1, 1, 1}, offset[3] = {0, 0, 0}; | 
					
						
							|  |  |  |   tex->mip_size_get(miplvl, extent); | 
					
						
							| 
									
										
										
										
											2020-09-07 13:59:22 +02:00
										 |  |  |   reinterpret_cast<Texture *>(tex)->update_sub(miplvl, offset, extent, data_format, pixels); | 
					
						
							| 
									
										
										
										
											2018-07-10 13:17:32 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-22 08:18:02 +02:00
										 |  |  | void GPU_texture_update_sub(GPUTexture *tex, | 
					
						
							| 
									
										
										
										
											2020-09-07 13:59:22 +02:00
										 |  |  |                             eGPUDataFormat data_format, | 
					
						
							| 
									
										
										
										
											2019-01-23 14:15:43 +11:00
										 |  |  |                             const void *pixels, | 
					
						
							| 
									
										
										
										
											2018-06-22 08:18:02 +02:00
										 |  |  |                             int offset_x, | 
					
						
							|  |  |  |                             int offset_y, | 
					
						
							|  |  |  |                             int offset_z, | 
					
						
							|  |  |  |                             int width, | 
					
						
							|  |  |  |                             int height, | 
					
						
							|  |  |  |                             int depth) | 
					
						
							| 
									
										
										
										
											2017-08-22 10:22:11 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   int offset[3] = {offset_x, offset_y, offset_z}; | 
					
						
							|  |  |  |   int extent[3] = {width, height, depth}; | 
					
						
							| 
									
										
										
										
											2020-09-07 13:59:22 +02:00
										 |  |  |   reinterpret_cast<Texture *>(tex)->update_sub(0, offset, extent, data_format, pixels); | 
					
						
							| 
									
										
										
										
											2017-08-22 10:22:11 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | void *GPU_texture_read(GPUTexture *tex_, eGPUDataFormat data_format, int miplvl) | 
					
						
							| 
									
										
										
										
											2018-06-21 18:35:37 -06:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   Texture *tex = reinterpret_cast<Texture *>(tex_); | 
					
						
							|  |  |  |   return tex->read(miplvl, data_format); | 
					
						
							| 
									
										
										
										
											2018-07-10 13:17:32 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Fills the whole texture with the same data for all pixels. | 
					
						
							|  |  |  |  * \warning Only work for 2D texture for now. | 
					
						
							|  |  |  |  * \warning Only clears the mip 0 of the texture. | 
					
						
							| 
									
										
										
										
											2020-09-08 09:10:17 +10:00
										 |  |  |  * \param data_format: data format of the pixel data. | 
					
						
							|  |  |  |  * \param data: 1 pixel worth of data to fill the texture with. | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |  */ | 
					
						
							|  |  |  | void GPU_texture_clear(GPUTexture *tex, eGPUDataFormat data_format, const void *data) | 
					
						
							| 
									
										
										
										
											2020-02-15 12:17:39 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-11-06 17:49:09 +01:00
										 |  |  |   BLI_assert(data != nullptr); /* Do not accept NULL as parameter. */ | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   reinterpret_cast<Texture *>(tex)->clear(data_format, data); | 
					
						
							| 
									
										
										
										
											2020-02-15 12:17:39 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:33:56 +02:00
										 |  |  | /* NOTE: Updates only mip 0. */ | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | void GPU_texture_update(GPUTexture *tex, eGPUDataFormat data_format, const void *data) | 
					
						
							| 
									
										
										
										
											2018-07-10 13:17:32 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   reinterpret_cast<Texture *>(tex)->update(data_format, data); | 
					
						
							| 
									
										
										
										
											2018-06-21 18:35:37 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-08 00:30:38 +02:00
										 |  |  | /* Makes data interpretation aware of the source layout.
 | 
					
						
							|  |  |  |  * Skipping pixels correctly when changing rows when doing partial update.*/ | 
					
						
							|  |  |  | void GPU_unpack_row_length_set(uint len) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-08 04:12:12 +02:00
										 |  |  |   Context::get()->state_manager->texture_unpack_row_length_set(len); | 
					
						
							| 
									
										
										
										
											2020-09-08 00:30:38 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-08 03:22:35 +02:00
										 |  |  | /* ------ Binding ------ */ | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | void GPU_texture_bind_ex(GPUTexture *tex_, | 
					
						
							|  |  |  |                          eGPUSamplerState state, | 
					
						
							|  |  |  |                          int unit, | 
					
						
							|  |  |  |                          const bool UNUSED(set_number)) | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   Texture *tex = reinterpret_cast<Texture *>(tex_); | 
					
						
							| 
									
										
										
										
											2020-09-05 17:31:53 +02:00
										 |  |  |   state = (state >= GPU_SAMPLER_MAX) ? tex->sampler_state : state; | 
					
						
							| 
									
										
										
										
											2020-09-08 04:12:12 +02:00
										 |  |  |   Context::get()->state_manager->texture_bind(tex, state, unit); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | void GPU_texture_bind(GPUTexture *tex_, int unit) | 
					
						
							| 
									
										
										
										
											2020-06-02 18:14:28 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   Texture *tex = reinterpret_cast<Texture *>(tex_); | 
					
						
							| 
									
										
										
										
											2020-09-08 04:12:12 +02:00
										 |  |  |   Context::get()->state_manager->texture_bind(tex, tex->sampler_state, unit); | 
					
						
							| 
									
										
										
										
											2020-06-02 18:14:28 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | void GPU_texture_unbind(GPUTexture *tex_) | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   Texture *tex = reinterpret_cast<Texture *>(tex_); | 
					
						
							| 
									
										
										
										
											2020-09-08 04:12:12 +02:00
										 |  |  |   Context::get()->state_manager->texture_unbind(tex); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-02 18:14:28 +02:00
										 |  |  | void GPU_texture_unbind_all(void) | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-08 04:12:12 +02:00
										 |  |  |   Context::get()->state_manager->texture_unbind_all(); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-12 06:10:11 +02:00
										 |  |  | void GPU_texture_image_bind(GPUTexture *tex, int unit) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Context::get()->state_manager->image_bind(unwrap(tex), unit); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GPU_texture_image_unbind(GPUTexture *tex) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Context::get()->state_manager->image_unbind(unwrap(tex)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GPU_texture_image_unbind_all(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Context::get()->state_manager->image_unbind_all(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-18 11:21:17 +02:00
										 |  |  | void GPU_texture_generate_mipmap(GPUTexture *tex) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   reinterpret_cast<Texture *>(tex)->generate_mipmap(); | 
					
						
							| 
									
										
										
										
											2020-04-26 12:25:32 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-22 21:22:07 +02:00
										 |  |  | /* Copy a texture content to a similar texture. Only Mip 0 is copied. */ | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | void GPU_texture_copy(GPUTexture *dst_, GPUTexture *src_) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Texture *src = reinterpret_cast<Texture *>(src_); | 
					
						
							|  |  |  |   Texture *dst = reinterpret_cast<Texture *>(dst_); | 
					
						
							|  |  |  |   src->copy_to(dst); | 
					
						
							| 
									
										
										
										
											2020-04-22 21:22:07 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | void GPU_texture_compare_mode(GPUTexture *tex_, bool use_compare) | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   Texture *tex = reinterpret_cast<Texture *>(tex_); | 
					
						
							|  |  |  |   /* Only depth formats does support compare mode. */ | 
					
						
							|  |  |  |   BLI_assert(!(use_compare) || (tex->format_flag_get() & GPU_FORMAT_DEPTH)); | 
					
						
							| 
									
										
										
										
											2020-06-02 10:47:45 +02:00
										 |  |  |   SET_FLAG_FROM_TEST(tex->sampler_state, use_compare, GPU_SAMPLER_COMPARE); | 
					
						
							| 
									
										
										
										
											2017-02-03 16:01:32 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | void GPU_texture_filter_mode(GPUTexture *tex_, bool use_filter) | 
					
						
							| 
									
										
										
										
											2017-02-03 16:01:32 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   Texture *tex = reinterpret_cast<Texture *>(tex_); | 
					
						
							| 
									
										
										
										
											2018-03-24 20:27:39 +01:00
										 |  |  |   /* Stencil and integer format does not support filtering. */ | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   BLI_assert(!(use_filter) || | 
					
						
							|  |  |  |              !(tex->format_flag_get() & (GPU_FORMAT_STENCIL | GPU_FORMAT_INTEGER))); | 
					
						
							| 
									
										
										
										
											2020-06-02 10:47:45 +02:00
										 |  |  |   SET_FLAG_FROM_TEST(tex->sampler_state, use_filter, GPU_SAMPLER_FILTER); | 
					
						
							| 
									
										
										
										
											2017-02-03 16:01:32 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | void GPU_texture_mipmap_mode(GPUTexture *tex_, bool use_mipmap, bool use_filter) | 
					
						
							| 
									
										
										
										
											2017-04-18 11:21:17 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   Texture *tex = reinterpret_cast<Texture *>(tex_); | 
					
						
							| 
									
										
										
										
											2018-03-24 20:27:39 +01:00
										 |  |  |   /* Stencil and integer format does not support filtering. */ | 
					
						
							| 
									
										
										
										
											2020-06-02 10:47:45 +02:00
										 |  |  |   BLI_assert(!(use_filter || use_mipmap) || | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |              !(tex->format_flag_get() & (GPU_FORMAT_STENCIL | GPU_FORMAT_INTEGER))); | 
					
						
							| 
									
										
										
										
											2020-06-02 10:47:45 +02:00
										 |  |  |   SET_FLAG_FROM_TEST(tex->sampler_state, use_mipmap, GPU_SAMPLER_MIPMAP); | 
					
						
							|  |  |  |   SET_FLAG_FROM_TEST(tex->sampler_state, use_filter, GPU_SAMPLER_FILTER); | 
					
						
							| 
									
										
										
										
											2017-04-18 11:21:17 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | void GPU_texture_anisotropic_filter(GPUTexture *tex_, bool use_aniso) | 
					
						
							| 
									
										
										
										
											2020-07-29 04:55:21 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   Texture *tex = reinterpret_cast<Texture *>(tex_); | 
					
						
							| 
									
										
										
										
											2020-07-29 04:55:21 +02:00
										 |  |  |   /* Stencil and integer format does not support filtering. */ | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   BLI_assert(!(use_aniso) || | 
					
						
							|  |  |  |              !(tex->format_flag_get() & (GPU_FORMAT_STENCIL | GPU_FORMAT_INTEGER))); | 
					
						
							| 
									
										
										
										
											2020-07-29 04:55:21 +02:00
										 |  |  |   SET_FLAG_FROM_TEST(tex->sampler_state, use_aniso, GPU_SAMPLER_ANISO); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | void GPU_texture_wrap_mode(GPUTexture *tex_, bool use_repeat, bool use_clamp) | 
					
						
							| 
									
										
										
										
											2017-02-03 16:01:32 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   Texture *tex = reinterpret_cast<Texture *>(tex_); | 
					
						
							| 
									
										
										
										
											2020-06-02 10:47:45 +02:00
										 |  |  |   SET_FLAG_FROM_TEST(tex->sampler_state, use_repeat, GPU_SAMPLER_REPEAT); | 
					
						
							|  |  |  |   SET_FLAG_FROM_TEST(tex->sampler_state, !use_clamp, GPU_SAMPLER_CLAMP_BORDER); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-17 02:15:41 +02:00
										 |  |  | void GPU_texture_swizzle_set(GPUTexture *tex, const char swizzle[4]) | 
					
						
							| 
									
										
										
										
											2020-03-11 13:56:28 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   reinterpret_cast<Texture *>(tex)->swizzle_set(swizzle); | 
					
						
							| 
									
										
										
										
											2020-03-11 13:56:28 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | void GPU_texture_free(GPUTexture *tex_) | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   Texture *tex = reinterpret_cast<Texture *>(tex_); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  |   tex->refcount--; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-22 09:32:37 +10:00
										 |  |  |   if (tex->refcount < 0) { | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  |     fprintf(stderr, "GPUTexture: negative refcount\n"); | 
					
						
							| 
									
										
										
										
											2019-04-22 09:32:37 +10:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-06-04 09:09:12 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  |   if (tex->refcount == 0) { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |     delete tex; | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GPU_texture_ref(GPUTexture *tex) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   reinterpret_cast<Texture *>(tex)->refcount++; | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int GPU_texture_width(const GPUTexture *tex) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   return reinterpret_cast<const Texture *>(tex)->width_get(); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int GPU_texture_height(const GPUTexture *tex) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   return reinterpret_cast<const Texture *>(tex)->height_get(); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-24 22:09:04 +10:00
										 |  |  | int GPU_texture_orig_width(const GPUTexture *tex) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   return reinterpret_cast<const Texture *>(tex)->src_w; | 
					
						
							| 
									
										
										
										
											2019-06-24 22:09:04 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int GPU_texture_orig_height(const GPUTexture *tex) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   return reinterpret_cast<const Texture *>(tex)->src_h; | 
					
						
							| 
									
										
										
										
											2019-06-24 22:09:04 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | void GPU_texture_orig_size_set(GPUTexture *tex_, int w, int h) | 
					
						
							| 
									
										
										
										
											2018-07-10 13:17:32 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   Texture *tex = reinterpret_cast<Texture *>(tex_); | 
					
						
							|  |  |  |   tex->src_w = w; | 
					
						
							|  |  |  |   tex->src_h = h; | 
					
						
							| 
									
										
										
										
											2018-07-10 13:17:32 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-23 14:15:43 +11:00
										 |  |  | eGPUTextureFormat GPU_texture_format(const GPUTexture *tex) | 
					
						
							| 
									
										
										
										
											2017-05-16 02:59:25 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   return reinterpret_cast<const Texture *>(tex)->format_get(); | 
					
						
							| 
									
										
										
										
											2018-03-14 03:20:39 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-18 01:26:08 +01:00
										 |  |  | bool GPU_texture_depth(const GPUTexture *tex) | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   return (reinterpret_cast<const Texture *>(tex)->format_flag_get() & GPU_FORMAT_DEPTH) != 0; | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-18 01:26:08 +01:00
										 |  |  | bool GPU_texture_stencil(const GPUTexture *tex) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   return (reinterpret_cast<const Texture *>(tex)->format_flag_get() & GPU_FORMAT_STENCIL) != 0; | 
					
						
							| 
									
										
										
										
											2018-03-24 20:27:39 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool GPU_texture_integer(const GPUTexture *tex) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   return (reinterpret_cast<const Texture *>(tex)->format_flag_get() & GPU_FORMAT_INTEGER) != 0; | 
					
						
							| 
									
										
										
										
											2018-03-24 20:27:39 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool GPU_texture_cube(const GPUTexture *tex) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   return (reinterpret_cast<const Texture *>(tex)->type_get() & GPU_TEXTURE_CUBE) != 0; | 
					
						
							| 
									
										
										
										
											2017-03-18 01:26:08 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | bool GPU_texture_array(const GPUTexture *tex) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return (reinterpret_cast<const Texture *>(tex)->type_get() & GPU_TEXTURE_ARRAY) != 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-30 11:20:39 -03:00
										 |  |  | #ifndef GPU_NO_USE_PY_REFERENCES
 | 
					
						
							| 
									
										
										
										
											2021-04-29 14:48:59 -03:00
										 |  |  | void **GPU_texture_py_reference_get(GPUTexture *tex) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return unwrap(tex)->py_ref; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GPU_texture_py_reference_set(GPUTexture *tex, void **py_ref) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   BLI_assert(py_ref == nullptr || unwrap(tex)->py_ref == nullptr); | 
					
						
							|  |  |  |   unwrap(tex)->py_ref = py_ref; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | /* TODO remove */ | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | int GPU_texture_opengl_bindcode(const GPUTexture *tex) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   return reinterpret_cast<const Texture *>(tex)->gl_bindcode_get(); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  | void GPU_texture_get_mipmap_size(GPUTexture *tex, int lvl, int *r_size) | 
					
						
							| 
									
										
										
										
											2018-07-10 13:17:32 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:29:51 +02:00
										 |  |  |   return reinterpret_cast<Texture *>(tex)->mip_size_get(lvl, r_size); | 
					
						
							| 
									
										
										
										
											2018-07-10 13:17:32 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2020-06-02 10:47:45 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-02 01:25:32 +02:00
										 |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-02 10:47:45 +02:00
										 |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name GPU Sampler Objects
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Simple wrapper around opengl sampler objects. | 
					
						
							|  |  |  |  * Override texture sampler state for one sampler unit only. | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:31:53 +02:00
										 |  |  | /* Update user defined sampler states. */ | 
					
						
							|  |  |  | void GPU_samplers_update(void) | 
					
						
							| 
									
										
										
										
											2020-06-02 10:47:45 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:31:53 +02:00
										 |  |  |   GPUBackend::get()->samplers_update(); | 
					
						
							| 
									
										
										
										
											2020-06-02 10:47:45 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \} */ | 
					
						
							| 
									
										
										
											
												Python: gpu module: add new submodules and types
This commit extends the gpu python API with:
```
gpu.types.Buffer         #"__init__", "to_list"
gpu.types.GPUTexture     #"__init__", "clear", "read", "format"
gpu.types.GPUFrameBuffer #"__init__", "bind", "clear", "is_bound", "viewport", ("__enter__",  "__exit__" with "GPUFrameBufferStackContext")
gpu.types.GPUUniformBuf  #"__init__", "update"
gpu.state                #"blend_set",  "blend_get",  "depth_test_set",  "depth_test_get",  "depth_mask_set",  "depth_mask_get",  "viewport_set",  "viewport_get",  "line_width_set",  "line_width_get",  "point_size_set",  "color_mask_set",  "face_culling_set", "front_facing_set",  "program_point_size_set"
```
Add these methods to existing objects:
```
gpu.types.GPUShader  #"uniform_sample", "uniform_buffer"
```
Maniphest Tasks: T80481
Differential Revision: https://developer.blender.org/D8826
											
										 
											2021-02-17 10:48:08 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name GPU texture utilities
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t GPU_texture_component_len(eGPUTextureFormat tex_format) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return to_component_len(tex_format); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t GPU_texture_dataformat_size(eGPUDataFormat data_format) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return to_bytesize(data_format); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \} */ |