| 
									
										
										
										
											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 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "BLI_blenlib.h"
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | #include "BLI_math_base.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "BLI_utildefines.h"
 | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-22 19:57:58 +01:00
										 |  |  | #include "GPU_batch.h"
 | 
					
						
							| 
									
										
										
										
											2017-10-10 17:18:29 +02:00
										 |  |  | #include "GPU_extensions.h"
 | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | #include "GPU_shader.h"
 | 
					
						
							|  |  |  | #include "GPU_texture.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | #include "gpu_backend.hh"
 | 
					
						
							| 
									
										
										
										
											2020-08-08 03:01:45 +02:00
										 |  |  | #include "gpu_context_private.hh"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "gpu_private.h"
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | #include "gpu_texture_private.hh"
 | 
					
						
							| 
									
										
										
										
											2018-06-11 20:50:14 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-25 23:27:40 +02:00
										 |  |  | #include "gpu_framebuffer_private.hh"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | namespace blender::gpu { | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Constructor / Destructor
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | FrameBuffer::FrameBuffer(const char *name) | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   if (name) { | 
					
						
							|  |  |  |     BLI_strncpy(name_, name, sizeof(name_)); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   else { | 
					
						
							|  |  |  |     name_[0] = '\0'; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   /* Force config on first use. */ | 
					
						
							|  |  |  |   dirty_attachments_ = true; | 
					
						
							| 
									
										
										
										
											2020-08-29 15:17:13 +02:00
										 |  |  |   dirty_state_ = true; | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   for (int i = 0; i < ARRAY_SIZE(attachments_); i++) { | 
					
						
							|  |  |  |     attachments_[i].tex = NULL; | 
					
						
							|  |  |  |     attachments_[i].mip = -1; | 
					
						
							|  |  |  |     attachments_[i].layer = -1; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | FrameBuffer::~FrameBuffer() | 
					
						
							| 
									
										
										
										
											2018-06-11 20:50:14 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   for (int i = 0; i < ARRAY_SIZE(attachments_); i++) { | 
					
						
							|  |  |  |     if (attachments_[i].tex != NULL) { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:37:01 +02:00
										 |  |  |       reinterpret_cast<Texture *>(attachments_[i].tex)->detach_from(this); | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-06-11 20:50:14 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | /** \} */ | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							| 
									
										
										
										
											2020-09-02 09:58:26 +10:00
										 |  |  | /** \name Attachments Management
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |  * \{ */ | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | void FrameBuffer::attachment_set(GPUAttachmentType type, const GPUAttachment &new_attachment) | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   if (new_attachment.mip == -1) { | 
					
						
							|  |  |  |     return; /* GPU_ATTACHMENT_LEAVE */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-02 10:04:50 +10:00
										 |  |  |   if (type >= GPU_FB_MAX_ATTACHMENT) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     fprintf(stderr, | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |             "GPUFramebuffer: Error: Trying to attach texture to type %d but maximum slot is %d.\n", | 
					
						
							|  |  |  |             type - GPU_FB_COLOR_ATTACHMENT0, | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             GPU_FB_MAX_COLOR_ATTACHMENT); | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-06-08 20:12:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   if (new_attachment.tex) { | 
					
						
							|  |  |  |     if (new_attachment.layer > 0) { | 
					
						
							| 
									
										
										
										
											2020-09-03 21:52:30 +02:00
										 |  |  |       BLI_assert(GPU_texture_cube(new_attachment.tex) || GPU_texture_array(new_attachment.tex)); | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (GPU_texture_stencil(new_attachment.tex)) { | 
					
						
							|  |  |  |       BLI_assert(ELEM(type, GPU_FB_DEPTH_STENCIL_ATTACHMENT)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (GPU_texture_depth(new_attachment.tex)) { | 
					
						
							|  |  |  |       BLI_assert(ELEM(type, GPU_FB_DEPTH_ATTACHMENT)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   GPUAttachment &attachment = attachments_[type]; | 
					
						
							| 
									
										
										
										
											2017-06-08 20:12:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   if (attachment.tex == new_attachment.tex && attachment.layer == new_attachment.layer && | 
					
						
							|  |  |  |       attachment.mip == new_attachment.mip) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     return; /* Exact same texture already bound here. */ | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   /* Unbind previous and bind new. */ | 
					
						
							|  |  |  |   /* TODO(fclem) cleanup the casts. */ | 
					
						
							|  |  |  |   if (attachment.tex) { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:37:01 +02:00
										 |  |  |     reinterpret_cast<Texture *>(attachment.tex)->detach_from(this); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-06-08 20:12:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   attachment = new_attachment; | 
					
						
							| 
									
										
										
										
											2017-06-08 20:12:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   /* Might be null if this is for unbinding. */ | 
					
						
							|  |  |  |   if (attachment.tex) { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:37:01 +02:00
										 |  |  |     reinterpret_cast<Texture *>(attachment.tex)->attach_to(this, type); | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     /* GPU_ATTACHMENT_NONE */ | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-06-16 13:25:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   dirty_attachments_ = true; | 
					
						
							| 
									
										
										
										
											2017-06-16 13:25:22 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:37:01 +02:00
										 |  |  | void FrameBuffer::attachment_remove(GPUAttachmentType type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   attachments_[type] = GPU_ATTACHMENT_NONE; | 
					
						
							|  |  |  |   dirty_attachments_ = true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | void FrameBuffer::recursive_downsample(int max_lvl, | 
					
						
							|  |  |  |                                        void (*callback)(void *userData, int level), | 
					
						
							|  |  |  |                                        void *userData) | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-02 09:58:26 +10:00
										 |  |  |   /* Bind to make sure the frame-buffer is up to date. */ | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   this->bind(true); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-05 17:36:53 +02:00
										 |  |  |   /* FIXME(fclem): This assumes all mips are defined which may not be the case. */ | 
					
						
							|  |  |  |   max_lvl = min_ii(max_lvl, floor(log2(max_ii(width_, height_)))); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (int mip_lvl = 1; mip_lvl <= max_lvl; mip_lvl++) { | 
					
						
							| 
									
										
										
										
											2020-09-02 09:58:26 +10:00
										 |  |  |     /* Replace attached mip-level for each attachment. */ | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |     for (int att = 0; att < ARRAY_SIZE(attachments_); att++) { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:36:53 +02:00
										 |  |  |       Texture *tex = reinterpret_cast<Texture *>(attachments_[att].tex); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       if (tex != NULL) { | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |         /* Some Intel HDXXX have issue with rendering to a mipmap that is below
 | 
					
						
							|  |  |  |          * the texture GL_TEXTURE_MAX_LEVEL. So even if it not correct, in this case | 
					
						
							|  |  |  |          * we allow GL_TEXTURE_MAX_LEVEL to be one level lower. In practice it does work! */ | 
					
						
							| 
									
										
										
										
											2020-09-05 17:36:53 +02:00
										 |  |  |         int mip_max = (GPU_mip_render_workaround()) ? mip_lvl : (mip_lvl - 1); | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |         /* Restrict fetches only to previous level. */ | 
					
						
							| 
									
										
										
										
											2020-09-05 17:36:53 +02:00
										 |  |  |         tex->mip_range_set(mip_lvl - 1, mip_max); | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |         /* Bind next level. */ | 
					
						
							|  |  |  |         attachments_[att].mip = mip_lvl; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |     /* Update the internal attachments and viewport size. */ | 
					
						
							|  |  |  |     dirty_attachments_ = true; | 
					
						
							|  |  |  |     this->bind(true); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |     callback(userData, mip_lvl); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   for (int att = 0; att < ARRAY_SIZE(attachments_); att++) { | 
					
						
							|  |  |  |     if (attachments_[att].tex != NULL) { | 
					
						
							|  |  |  |       /* Reset mipmap level range. */ | 
					
						
							| 
									
										
										
										
											2020-09-05 17:36:53 +02:00
										 |  |  |       reinterpret_cast<Texture *>(attachments_[att].tex)->mip_range_set(0, max_lvl); | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |       /* Reset base level. NOTE: might not be the one bound at the start of this function. */ | 
					
						
							|  |  |  |       attachments_[att].mip = 0; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-08-29 17:46:24 +02:00
										 |  |  |   dirty_attachments_ = true; | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | /** \} */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | }  // namespace blender::gpu
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name C-API
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | using namespace blender; | 
					
						
							|  |  |  | using namespace blender::gpu; | 
					
						
							| 
									
										
										
										
											2018-11-06 15:24:13 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:28:26 +02:00
										 |  |  | GPUFrameBuffer *GPU_framebuffer_create(const char *name) | 
					
						
							| 
									
										
										
										
											2018-11-06 15:24:13 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   /* We generate the FB object later at first use in order to
 | 
					
						
							| 
									
										
										
										
											2020-09-02 09:58:26 +10:00
										 |  |  |    * create the frame-buffer in the right opengl context. */ | 
					
						
							| 
									
										
										
										
											2020-08-29 01:28:26 +02:00
										 |  |  |   return (GPUFrameBuffer *)GPUBackend::get()->framebuffer_alloc(name); | 
					
						
							| 
									
										
										
										
											2018-11-06 15:24:13 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | void GPU_framebuffer_free(GPUFrameBuffer *gpu_fb) | 
					
						
							| 
									
										
										
										
											2018-11-06 15:24:13 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   delete reinterpret_cast<FrameBuffer *>(gpu_fb); | 
					
						
							| 
									
										
										
										
											2018-11-06 15:24:13 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | /* ---------- Binding ----------- */ | 
					
						
							| 
									
										
										
										
											2018-11-06 15:24:13 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-25 23:27:40 +02:00
										 |  |  | void GPU_framebuffer_bind(GPUFrameBuffer *gpu_fb) | 
					
						
							| 
									
										
										
										
											2017-02-07 11:20:15 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-08-25 23:27:40 +02:00
										 |  |  |   FrameBuffer *fb = reinterpret_cast<FrameBuffer *>(gpu_fb); | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   const bool enable_srgb = true; | 
					
						
							|  |  |  |   fb->bind(enable_srgb); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-08-25 23:27:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-02 09:58:26 +10:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Workaround for binding a SRGB frame-buffer without doing the SRGB transform. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | void GPU_framebuffer_bind_no_srgb(GPUFrameBuffer *gpu_fb) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   FrameBuffer *fb = reinterpret_cast<FrameBuffer *>(gpu_fb); | 
					
						
							|  |  |  |   const bool enable_srgb = false; | 
					
						
							|  |  |  |   fb->bind(enable_srgb); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2017-02-07 11:20:15 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-02 09:58:26 +10:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * For stereo rendering. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | void GPU_backbuffer_bind(eGPUBackBuffer buffer) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   GPUContext *ctx = GPU_context_active_get(); | 
					
						
							| 
									
										
										
										
											2020-04-14 20:44:45 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   if (buffer == GPU_BACKBUFFER_LEFT) { | 
					
						
							|  |  |  |     ctx->back_left->bind(false); | 
					
						
							| 
									
										
										
										
											2019-04-22 09:32:37 +10:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   else { | 
					
						
							|  |  |  |     ctx->back_right->bind(false); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2017-02-07 11:20:15 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | void GPU_framebuffer_restore(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   GPU_context_active_get()->back_left->bind(false); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2017-02-07 11:20:15 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | GPUFrameBuffer *GPU_framebuffer_active_get(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   GPUContext *ctx = GPU_context_active_get(); | 
					
						
							|  |  |  |   return reinterpret_cast<GPUFrameBuffer *>(ctx ? ctx->active_fb : NULL); | 
					
						
							| 
									
										
										
										
											2017-02-07 11:20:15 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-02 09:58:26 +10:00
										 |  |  | /* Returns the default frame-buffer. Will always exists even if it's just a dummy. */ | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | GPUFrameBuffer *GPU_framebuffer_back_get(void) | 
					
						
							| 
									
										
										
										
											2020-08-18 14:43:18 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   GPUContext *ctx = GPU_context_active_get(); | 
					
						
							|  |  |  |   return reinterpret_cast<GPUFrameBuffer *>(ctx ? ctx->back_left : NULL); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-08-25 23:27:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | bool GPU_framebuffer_bound(GPUFrameBuffer *gpu_fb) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return (gpu_fb == GPU_framebuffer_active_get()); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-08-18 14:43:18 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | /* ---------- Attachment Management ----------- */ | 
					
						
							| 
									
										
										
										
											2020-08-18 14:43:18 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | bool GPU_framebuffer_check_valid(GPUFrameBuffer *gpu_fb, char err_out[256]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return reinterpret_cast<FrameBuffer *>(gpu_fb)->check(err_out); | 
					
						
							| 
									
										
										
										
											2020-08-18 14:43:18 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-02 10:04:50 +10:00
										 |  |  | void GPU_framebuffer_texture_attach_ex(GPUFrameBuffer *gpu_fb, GPUAttachment attachment, int slot) | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-02 01:25:32 +02:00
										 |  |  |   Texture *tex = reinterpret_cast<Texture *>(attachment.tex); | 
					
						
							|  |  |  |   GPUAttachmentType type = tex->attachment_type(slot); | 
					
						
							| 
									
										
										
										
											2020-09-02 10:04:50 +10:00
										 |  |  |   reinterpret_cast<FrameBuffer *>(gpu_fb)->attachment_set(type, attachment); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-30 19:07:49 +02:00
										 |  |  | void GPU_framebuffer_texture_attach(GPUFrameBuffer *fb, GPUTexture *tex, int slot, int mip) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-02 10:04:50 +10:00
										 |  |  |   GPUAttachment attachment = GPU_ATTACHMENT_TEXTURE_MIP(tex, mip); | 
					
						
							|  |  |  |   GPU_framebuffer_texture_attach_ex(fb, attachment, slot); | 
					
						
							| 
									
										
										
										
											2020-08-30 19:07:49 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GPU_framebuffer_texture_layer_attach( | 
					
						
							|  |  |  |     GPUFrameBuffer *fb, GPUTexture *tex, int slot, int layer, int mip) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-02 10:04:50 +10:00
										 |  |  |   GPUAttachment attachment = GPU_ATTACHMENT_TEXTURE_LAYER_MIP(tex, layer, mip); | 
					
						
							|  |  |  |   GPU_framebuffer_texture_attach_ex(fb, attachment, slot); | 
					
						
							| 
									
										
										
										
											2020-08-30 19:07:49 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GPU_framebuffer_texture_cubeface_attach( | 
					
						
							|  |  |  |     GPUFrameBuffer *fb, GPUTexture *tex, int slot, int face, int mip) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-02 10:04:50 +10:00
										 |  |  |   GPUAttachment attachment = GPU_ATTACHMENT_TEXTURE_CUBEFACE_MIP(tex, face, mip); | 
					
						
							|  |  |  |   GPU_framebuffer_texture_attach_ex(fb, attachment, slot); | 
					
						
							| 
									
										
										
										
											2020-08-30 19:07:49 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | void GPU_framebuffer_texture_detach(GPUFrameBuffer *gpu_fb, GPUTexture *tex) | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:37:01 +02:00
										 |  |  |   FrameBuffer *fb = reinterpret_cast<FrameBuffer *>(gpu_fb); | 
					
						
							|  |  |  |   reinterpret_cast<Texture *>(tex)->detach_from(fb); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * First GPUAttachment in *config is always the depth/depth_stencil buffer. | 
					
						
							|  |  |  |  * Following GPUAttachments are color buffers. | 
					
						
							|  |  |  |  * Setting GPUAttachment.mip to -1 will leave the texture in this slot. | 
					
						
							|  |  |  |  * Setting GPUAttachment.tex to NULL will detach the texture in this slot. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void GPU_framebuffer_config_array(GPUFrameBuffer *gpu_fb, | 
					
						
							|  |  |  |                                   const GPUAttachment *config, | 
					
						
							|  |  |  |                                   int config_len) | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   FrameBuffer *fb = reinterpret_cast<FrameBuffer *>(gpu_fb); | 
					
						
							| 
									
										
										
										
											2018-02-17 14:05:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-02 10:04:50 +10:00
										 |  |  |   const GPUAttachment &depth_attachment = config[0]; | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   Span<GPUAttachment> color_attachments(config + 1, config_len - 1); | 
					
						
							| 
									
										
										
										
											2016-08-19 00:52:52 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-02 10:04:50 +10:00
										 |  |  |   if (depth_attachment.mip == -1) { | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |     /* GPU_ATTACHMENT_LEAVE */ | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-09-02 10:04:50 +10:00
										 |  |  |   else if (depth_attachment.tex == NULL) { | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |     /* GPU_ATTACHMENT_NONE: Need to clear both targets. */ | 
					
						
							| 
									
										
										
										
											2020-09-02 10:04:50 +10:00
										 |  |  |     fb->attachment_set(GPU_FB_DEPTH_STENCIL_ATTACHMENT, depth_attachment); | 
					
						
							|  |  |  |     fb->attachment_set(GPU_FB_DEPTH_ATTACHMENT, depth_attachment); | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							| 
									
										
										
										
											2020-09-02 10:04:50 +10:00
										 |  |  |     GPUAttachmentType type = GPU_texture_stencil(depth_attachment.tex) ? | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |                                  GPU_FB_DEPTH_STENCIL_ATTACHMENT : | 
					
						
							|  |  |  |                                  GPU_FB_DEPTH_ATTACHMENT; | 
					
						
							| 
									
										
										
										
											2020-09-02 10:04:50 +10:00
										 |  |  |     fb->attachment_set(type, depth_attachment); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-08-19 00:52:52 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   GPUAttachmentType type = GPU_FB_COLOR_ATTACHMENT0; | 
					
						
							| 
									
										
										
										
											2020-09-02 10:04:50 +10:00
										 |  |  |   for (const GPUAttachment &attachment : color_attachments) { | 
					
						
							|  |  |  |     fb->attachment_set(type, attachment); | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |     ++type; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 15:17:13 +02:00
										 |  |  | /* ---------- Viewport & Scissor Region ----------- */ | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-02 09:58:26 +10:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Viewport and scissor size is stored per frame-buffer. | 
					
						
							|  |  |  |  * It is only reset to its original dimensions explicitly OR when binding the frame-buffer after | 
					
						
							|  |  |  |  * modifying its attachments. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-08-29 15:17:13 +02:00
										 |  |  | void GPU_framebuffer_viewport_set(GPUFrameBuffer *gpu_fb, int x, int y, int width, int height) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   int viewport_rect[4] = {x, y, width, height}; | 
					
						
							|  |  |  |   reinterpret_cast<FrameBuffer *>(gpu_fb)->viewport_set(viewport_rect); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-03-25 14:18:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 15:17:13 +02:00
										 |  |  | void GPU_framebuffer_viewport_get(GPUFrameBuffer *gpu_fb, int r_viewport[4]) | 
					
						
							| 
									
										
										
										
											2018-03-25 14:18:39 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-08-29 15:17:13 +02:00
										 |  |  |   reinterpret_cast<FrameBuffer *>(gpu_fb)->viewport_get(r_viewport); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-03-25 14:18:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-02 09:58:26 +10:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Reset to its attachment(s) size. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-08-29 15:17:13 +02:00
										 |  |  | void GPU_framebuffer_viewport_reset(GPUFrameBuffer *gpu_fb) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   reinterpret_cast<FrameBuffer *>(gpu_fb)->viewport_reset(); | 
					
						
							| 
									
										
										
										
											2018-03-25 14:18:39 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 15:17:13 +02:00
										 |  |  | /* ---------- Framebuffer Operations ----------- */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-25 23:27:40 +02:00
										 |  |  | void GPU_framebuffer_clear(GPUFrameBuffer *gpu_fb, | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |                            eGPUFrameBufferBits buffers, | 
					
						
							|  |  |  |                            const float clear_col[4], | 
					
						
							|  |  |  |                            float clear_depth, | 
					
						
							|  |  |  |                            uint clear_stencil) | 
					
						
							| 
									
										
										
										
											2018-03-25 14:18:39 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   reinterpret_cast<FrameBuffer *>(gpu_fb)->clear(buffers, clear_col, clear_depth, clear_stencil); | 
					
						
							| 
									
										
										
										
											2018-03-25 14:18:39 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-02 09:58:26 +10:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Clear all textures attached to this frame-buffer with a different color. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-08-25 23:27:40 +02:00
										 |  |  | void GPU_framebuffer_multi_clear(GPUFrameBuffer *gpu_fb, const float (*clear_cols)[4]) | 
					
						
							| 
									
										
										
										
											2020-03-09 16:27:24 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   reinterpret_cast<FrameBuffer *>(gpu_fb)->clear_multi(clear_cols); | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | void GPU_clear_color(float red, float green, float blue, float alpha) | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   float clear_col[4] = {red, green, blue, alpha}; | 
					
						
							|  |  |  |   GPU_context_active_get()->active_fb->clear(GPU_COLOR_BIT, clear_col, 0.0f, 0x0); | 
					
						
							| 
									
										
										
										
											2020-07-16 02:50:55 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | void GPU_clear_depth(float depth) | 
					
						
							| 
									
										
										
										
											2020-07-16 02:50:55 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   float clear_col[4] = {0}; | 
					
						
							|  |  |  |   GPU_context_active_get()->active_fb->clear(GPU_DEPTH_BIT, clear_col, depth, 0x0); | 
					
						
							| 
									
										
										
										
											2020-07-16 02:50:55 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | void GPU_framebuffer_read_depth(GPUFrameBuffer *gpu_fb, int x, int y, int w, int h, float *data) | 
					
						
							| 
									
										
										
										
											2020-07-16 02:50:55 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   int rect[4] = {x, y, w, h}; | 
					
						
							|  |  |  |   reinterpret_cast<FrameBuffer *>(gpu_fb)->read(GPU_DEPTH_BIT, GPU_DATA_FLOAT, rect, 1, 1, data); | 
					
						
							| 
									
										
										
										
											2020-07-16 02:50:55 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-25 23:27:40 +02:00
										 |  |  | void GPU_framebuffer_read_color(GPUFrameBuffer *gpu_fb, | 
					
						
							| 
									
										
										
										
											2020-07-16 02:50:55 +02:00
										 |  |  |                                 int x, | 
					
						
							|  |  |  |                                 int y, | 
					
						
							|  |  |  |                                 int w, | 
					
						
							|  |  |  |                                 int h, | 
					
						
							|  |  |  |                                 int channels, | 
					
						
							|  |  |  |                                 int slot, | 
					
						
							|  |  |  |                                 eGPUDataFormat format, | 
					
						
							|  |  |  |                                 void *data) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   int rect[4] = {x, y, w, h}; | 
					
						
							|  |  |  |   reinterpret_cast<FrameBuffer *>(gpu_fb)->read(GPU_COLOR_BIT, format, rect, channels, slot, data); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* TODO(fclem) rename to read_color. */ | 
					
						
							|  |  |  | void GPU_frontbuffer_read_pixels( | 
					
						
							|  |  |  |     int x, int y, int w, int h, int channels, eGPUDataFormat format, void *data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   int rect[4] = {x, y, w, h}; | 
					
						
							|  |  |  |   GPU_context_active_get()->front_left->read(GPU_COLOR_BIT, format, rect, channels, 0, data); | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* read_slot and write_slot are only used for color buffers. */ | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | /* TODO(fclem) port as texture operation. */ | 
					
						
							| 
									
										
										
										
											2020-08-25 23:27:40 +02:00
										 |  |  | void GPU_framebuffer_blit(GPUFrameBuffer *gpufb_read, | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |                           int read_slot, | 
					
						
							| 
									
										
										
										
											2020-08-25 23:27:40 +02:00
										 |  |  |                           GPUFrameBuffer *gpufb_write, | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |                           int write_slot, | 
					
						
							|  |  |  |                           eGPUFrameBufferBits blit_buffers) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-08-25 23:27:40 +02:00
										 |  |  |   FrameBuffer *fb_read = reinterpret_cast<FrameBuffer *>(gpufb_read); | 
					
						
							|  |  |  |   FrameBuffer *fb_write = reinterpret_cast<FrameBuffer *>(gpufb_write); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BLI_assert(blit_buffers != 0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-25 23:27:40 +02:00
										 |  |  |   FrameBuffer *prev_fb = GPU_context_active_get()->active_fb; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | #ifndef NDEBUG
 | 
					
						
							|  |  |  |   GPUTexture *read_tex, *write_tex; | 
					
						
							|  |  |  |   if (blit_buffers & (GPU_DEPTH_BIT | GPU_STENCIL_BIT)) { | 
					
						
							|  |  |  |     read_tex = fb_read->depth_tex(); | 
					
						
							|  |  |  |     write_tex = fb_write->depth_tex(); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   else { | 
					
						
							|  |  |  |     read_tex = fb_read->color_tex(read_slot); | 
					
						
							|  |  |  |     write_tex = fb_write->color_tex(write_slot); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   if (blit_buffers & GPU_DEPTH_BIT) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     BLI_assert(GPU_texture_depth(read_tex) && GPU_texture_depth(write_tex)); | 
					
						
							|  |  |  |     BLI_assert(GPU_texture_format(read_tex) == GPU_texture_format(write_tex)); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   if (blit_buffers & GPU_STENCIL_BIT) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     BLI_assert(GPU_texture_stencil(read_tex) && GPU_texture_stencil(write_tex)); | 
					
						
							|  |  |  |     BLI_assert(GPU_texture_format(read_tex) == GPU_texture_format(write_tex)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (GPU_texture_samples(write_tex) != 0 || GPU_texture_samples(read_tex) != 0) { | 
					
						
							|  |  |  |     /* Can only blit multisample textures to another texture of the same size. */ | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |     BLI_assert((GPU_texture_width(write_tex) == GPU_texture_width(read_tex)) && | 
					
						
							|  |  |  |                (GPU_texture_height(write_tex) == GPU_texture_height(read_tex))); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   fb_read->blit_to(blit_buffers, read_slot, fb_write, write_slot, 0, 0); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   /* FIXME(fclem) sRGB is not saved. */ | 
					
						
							|  |  |  |   prev_fb->bind(true); | 
					
						
							| 
									
										
										
										
											2017-02-15 15:15:42 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-22 02:01:58 +02:00
										 |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2020-09-02 09:58:26 +10:00
										 |  |  |  * Use this if you need to custom down-sample your texture and use the previous mip-level as | 
					
						
							| 
									
										
										
										
											2020-07-16 02:50:55 +02:00
										 |  |  |  * input. This function only takes care of the correct texture handling. It execute the callback | 
					
						
							|  |  |  |  * for each texture level. | 
					
						
							| 
									
										
										
										
											2019-03-19 15:17:46 +11:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-08-25 23:27:40 +02:00
										 |  |  | void GPU_framebuffer_recursive_downsample(GPUFrameBuffer *gpu_fb, | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |                                           int max_lvl, | 
					
						
							|  |  |  |                                           void (*callback)(void *userData, int level), | 
					
						
							|  |  |  |                                           void *userData) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   reinterpret_cast<FrameBuffer *>(gpu_fb)->recursive_downsample(max_lvl, callback, userData); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | /** \} */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name GPUOffScreen
 | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-09-02 09:58:26 +10:00
										 |  |  |  * Container that holds a frame-buffer and its textures. | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |  * Might be bound to multiple contexts. | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | #define FRAMEBUFFER_STACK_DEPTH 16
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | static struct { | 
					
						
							|  |  |  |   GPUFrameBuffer *framebuffers[FRAMEBUFFER_STACK_DEPTH]; | 
					
						
							|  |  |  |   uint top; | 
					
						
							|  |  |  | } FrameBufferStack = {{0}}; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | static void gpuPushFrameBuffer(GPUFrameBuffer *fb) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   BLI_assert(FrameBufferStack.top < FRAMEBUFFER_STACK_DEPTH); | 
					
						
							|  |  |  |   FrameBufferStack.framebuffers[FrameBufferStack.top] = fb; | 
					
						
							|  |  |  |   FrameBufferStack.top++; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | static GPUFrameBuffer *gpuPopFrameBuffer(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   BLI_assert(FrameBufferStack.top > 0); | 
					
						
							|  |  |  |   FrameBufferStack.top--; | 
					
						
							|  |  |  |   return FrameBufferStack.framebuffers[FrameBufferStack.top]; | 
					
						
							| 
									
										
										
										
											2017-06-22 02:01:58 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | #undef FRAMEBUFFER_STACK_DEPTH
 | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-13 21:31:46 +02:00
										 |  |  | #define MAX_CTX_FB_LEN 3
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | struct GPUOffScreen { | 
					
						
							| 
									
										
										
										
											2019-06-13 21:31:46 +02:00
										 |  |  |   struct { | 
					
						
							|  |  |  |     GPUContext *ctx; | 
					
						
							|  |  |  |     GPUFrameBuffer *fb; | 
					
						
							|  |  |  |   } framebuffers[MAX_CTX_FB_LEN]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   GPUTexture *color; | 
					
						
							|  |  |  |   GPUTexture *depth; | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-02 09:58:26 +10:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Returns the correct frame-buffer for the current context. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2019-06-13 21:31:46 +02:00
										 |  |  | static GPUFrameBuffer *gpu_offscreen_fb_get(GPUOffScreen *ofs) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   GPUContext *ctx = GPU_context_active_get(); | 
					
						
							|  |  |  |   BLI_assert(ctx); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (int i = 0; i < MAX_CTX_FB_LEN; i++) { | 
					
						
							|  |  |  |     if (ofs->framebuffers[i].fb == NULL) { | 
					
						
							|  |  |  |       ofs->framebuffers[i].ctx = ctx; | 
					
						
							| 
									
										
										
										
											2020-07-25 18:40:19 +02:00
										 |  |  |       GPU_framebuffer_ensure_config(&ofs->framebuffers[i].fb, | 
					
						
							|  |  |  |                                     { | 
					
						
							|  |  |  |                                         GPU_ATTACHMENT_TEXTURE(ofs->depth), | 
					
						
							|  |  |  |                                         GPU_ATTACHMENT_TEXTURE(ofs->color), | 
					
						
							|  |  |  |                                     }); | 
					
						
							| 
									
										
										
										
											2019-06-13 21:31:46 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (ofs->framebuffers[i].ctx == ctx) { | 
					
						
							|  |  |  |       return ofs->framebuffers[i].fb; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* List is full, this should never happen or
 | 
					
						
							|  |  |  |    * it might just slow things down if it happens | 
					
						
							| 
									
										
										
										
											2019-07-07 15:38:41 +10:00
										 |  |  |    * regularly. In this case we just empty the list | 
					
						
							| 
									
										
										
										
											2019-06-13 21:31:46 +02:00
										 |  |  |    * and start over. This is most likely never going | 
					
						
							|  |  |  |    * to happen under normal usage. */ | 
					
						
							|  |  |  |   BLI_assert(0); | 
					
						
							|  |  |  |   printf( | 
					
						
							|  |  |  |       "Warning: GPUOffscreen used in more than 3 GPUContext. " | 
					
						
							|  |  |  |       "This may create performance drop.\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (int i = 0; i < MAX_CTX_FB_LEN; i++) { | 
					
						
							|  |  |  |     GPU_framebuffer_free(ofs->framebuffers[i].fb); | 
					
						
							|  |  |  |     ofs->framebuffers[i].fb = NULL; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return gpu_offscreen_fb_get(ofs); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | GPUOffScreen *GPU_offscreen_create( | 
					
						
							| 
									
										
										
										
											2020-07-02 17:28:30 +02:00
										 |  |  |     int width, int height, bool depth, bool high_bitdepth, char err_out[256]) | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-07-25 18:40:19 +02:00
										 |  |  |   GPUOffScreen *ofs = (GPUOffScreen *)MEM_callocN(sizeof(GPUOffScreen), __func__); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* Sometimes areas can have 0 height or width and this will
 | 
					
						
							|  |  |  |    * create a 1D texture which we don't want. */ | 
					
						
							|  |  |  |   height = max_ii(1, height); | 
					
						
							|  |  |  |   width = max_ii(1, width); | 
					
						
							| 
									
										
										
										
											2019-01-22 16:30:17 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-02 17:28:30 +02:00
										 |  |  |   ofs->color = GPU_texture_create_2d( | 
					
						
							| 
									
										
										
										
											2020-09-05 17:33:56 +02:00
										 |  |  |       "ofs_color", width, height, 1, (high_bitdepth) ? GPU_RGBA16F : GPU_RGBA8, NULL); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (depth) { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:33:56 +02:00
										 |  |  |     ofs->depth = GPU_texture_create_2d("ofs_depth", width, height, 1, GPU_DEPTH24_STENCIL8, NULL); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if ((depth && !ofs->depth) || !ofs->color) { | 
					
						
							| 
									
										
										
										
											2020-09-05 17:33:56 +02:00
										 |  |  |     BLI_snprintf(err_out, 256, "GPUTexture: Texture allocation failed."); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     GPU_offscreen_free(ofs); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-04-20 20:51:54 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-13 21:31:46 +02:00
										 |  |  |   GPUFrameBuffer *fb = gpu_offscreen_fb_get(ofs); | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* check validity at the very end! */ | 
					
						
							| 
									
										
										
										
											2019-06-13 21:31:46 +02:00
										 |  |  |   if (!GPU_framebuffer_check_valid(fb, err_out)) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     GPU_offscreen_free(ofs); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   GPU_framebuffer_restore(); | 
					
						
							|  |  |  |   return ofs; | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GPU_offscreen_free(GPUOffScreen *ofs) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-06-13 21:31:46 +02:00
										 |  |  |   for (int i = 0; i < MAX_CTX_FB_LEN; i++) { | 
					
						
							|  |  |  |     if (ofs->framebuffers[i].fb) { | 
					
						
							|  |  |  |       GPU_framebuffer_free(ofs->framebuffers[i].fb); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-22 09:32:37 +10:00
										 |  |  |   } | 
					
						
							|  |  |  |   if (ofs->color) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     GPU_texture_free(ofs->color); | 
					
						
							| 
									
										
										
										
											2019-04-22 09:32:37 +10:00
										 |  |  |   } | 
					
						
							|  |  |  |   if (ofs->depth) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     GPU_texture_free(ofs->depth); | 
					
						
							| 
									
										
										
										
											2019-04-22 09:32:37 +10:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-06-04 09:09:12 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   MEM_freeN(ofs); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GPU_offscreen_bind(GPUOffScreen *ofs, bool save) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (save) { | 
					
						
							|  |  |  |     GPUFrameBuffer *fb = GPU_framebuffer_active_get(); | 
					
						
							| 
									
										
										
										
											2020-08-25 23:27:40 +02:00
										 |  |  |     gpuPushFrameBuffer(reinterpret_cast<GPUFrameBuffer *>(fb)); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-09-01 17:04:08 +02:00
										 |  |  |   reinterpret_cast<FrameBuffer *>(gpu_offscreen_fb_get(ofs))->bind(false); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-01 17:04:08 +02:00
										 |  |  | void GPU_offscreen_unbind(GPUOffScreen *UNUSED(ofs), bool restore) | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   GPUFrameBuffer *fb = NULL; | 
					
						
							|  |  |  |   if (restore) { | 
					
						
							|  |  |  |     fb = gpuPopFrameBuffer(); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-11-06 15:24:13 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (fb) { | 
					
						
							|  |  |  |     GPU_framebuffer_bind(fb); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     GPU_framebuffer_restore(); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-27 10:22:37 +02:00
										 |  |  | void GPU_offscreen_draw_to_screen(GPUOffScreen *ofs, int x, int y) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   GPUContext *ctx = GPU_context_active_get(); | 
					
						
							| 
									
										
										
										
											2020-08-25 23:27:40 +02:00
										 |  |  |   FrameBuffer *ofs_fb = reinterpret_cast<FrameBuffer *>(gpu_offscreen_fb_get(ofs)); | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   ofs_fb->blit_to(GPU_COLOR_BIT, 0, ctx->active_fb, 0, x, y); | 
					
						
							| 
									
										
										
										
											2018-04-27 10:22:37 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  | void GPU_offscreen_read_pixels(GPUOffScreen *ofs, eGPUDataFormat format, void *pixels) | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   BLI_assert(ELEM(format, GPU_DATA_UNSIGNED_BYTE, GPU_DATA_FLOAT)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const int w = GPU_texture_width(ofs->color); | 
					
						
							|  |  |  |   const int h = GPU_texture_height(ofs->color); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-29 01:13:54 +02:00
										 |  |  |   GPUFrameBuffer *ofs_fb = gpu_offscreen_fb_get(ofs); | 
					
						
							|  |  |  |   GPU_framebuffer_read_color(ofs_fb, 0, 0, w, h, 4, 0, format, pixels); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int GPU_offscreen_width(const GPUOffScreen *ofs) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return GPU_texture_width(ofs->color); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int GPU_offscreen_height(const GPUOffScreen *ofs) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return GPU_texture_height(ofs->color); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-13 18:18:04 +01:00
										 |  |  | GPUTexture *GPU_offscreen_color_texture(const GPUOffScreen *ofs) | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return ofs->color; | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-02 09:58:26 +10:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * \note only to be used by viewport code! | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | void GPU_offscreen_viewport_data_get(GPUOffScreen *ofs, | 
					
						
							|  |  |  |                                      GPUFrameBuffer **r_fb, | 
					
						
							|  |  |  |                                      GPUTexture **r_color, | 
					
						
							|  |  |  |                                      GPUTexture **r_depth) | 
					
						
							| 
									
										
										
										
											2017-05-03 02:50:29 +10:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-06-13 21:31:46 +02:00
										 |  |  |   *r_fb = gpu_offscreen_fb_get(ofs); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   *r_color = ofs->color; | 
					
						
							|  |  |  |   *r_depth = ofs->depth; | 
					
						
							| 
									
										
										
										
											2017-05-09 15:09:39 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-06-26 15:17:31 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-02 09:58:26 +10:00
										 |  |  | /** \} */ |