| 
									
										
										
										
											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. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "BLI_blenlib.h"
 | 
					
						
							|  |  |  | #include "BLI_utildefines.h"
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | #include "BLI_math_base.h"
 | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-22 19:57:58 +01:00
										 |  |  | #include "GPU_batch.h"
 | 
					
						
							| 
									
										
										
										
											2017-04-04 20:33:23 +02:00
										 |  |  | #include "GPU_draw.h"
 | 
					
						
							| 
									
										
										
										
											2017-10-10 17:18:29 +02:00
										 |  |  | #include "GPU_extensions.h"
 | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | #include "GPU_framebuffer.h"
 | 
					
						
							| 
									
										
										
										
											2017-02-22 19:57:58 +01:00
										 |  |  | #include "GPU_matrix.h"
 | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | #include "GPU_shader.h"
 | 
					
						
							|  |  |  | #include "GPU_texture.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-19 15:48:13 +02:00
										 |  |  | #include "gpu_private.h"
 | 
					
						
							|  |  |  | #include "gpu_context_private.h"
 | 
					
						
							| 
									
										
										
										
											2018-06-11 20:50:14 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | typedef enum { | 
					
						
							|  |  |  | 	GPU_FB_DEPTH_ATTACHMENT = 0, | 
					
						
							|  |  |  | 	GPU_FB_DEPTH_STENCIL_ATTACHMENT, | 
					
						
							|  |  |  | 	GPU_FB_COLOR_ATTACHMENT0, | 
					
						
							|  |  |  | 	GPU_FB_COLOR_ATTACHMENT1, | 
					
						
							|  |  |  | 	GPU_FB_COLOR_ATTACHMENT2, | 
					
						
							|  |  |  | 	GPU_FB_COLOR_ATTACHMENT3, | 
					
						
							|  |  |  | 	GPU_FB_COLOR_ATTACHMENT4, | 
					
						
							|  |  |  | 	/* Number of maximum output slots.
 | 
					
						
							|  |  |  | 	 * We support 5 outputs for now (usually we wouldn't need more to preserve fill rate). */ | 
					
						
							|  |  |  | 	/* Keep in mind that GL max is GL_MAX_DRAW_BUFFERS and is at least 8, corresponding to
 | 
					
						
							|  |  |  | 	 * the maximum number of COLOR attachments specified by glDrawBuffers. */ | 
					
						
							|  |  |  | 	GPU_FB_MAX_ATTACHEMENT | 
					
						
							|  |  |  | } GPUAttachmentType; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define GPU_FB_MAX_COLOR_ATTACHMENT (GPU_FB_MAX_ATTACHEMENT - GPU_FB_COLOR_ATTACHMENT0)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define GPU_FB_DIRTY_DRAWBUFFER (1 << 15)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define GPU_FB_ATTACHEMENT_IS_DIRTY(flag, type) ((flag & (1 << type)) != 0)
 | 
					
						
							|  |  |  | #define GPU_FB_ATTACHEMENT_SET_DIRTY(flag, type) (flag |= (1 << type))
 | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | struct GPUFrameBuffer { | 
					
						
							| 
									
										
										
										
											2018-07-19 15:48:13 +02:00
										 |  |  | 	GPUContext *ctx; | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 	GLuint object; | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	GPUAttachment attachments[GPU_FB_MAX_ATTACHEMENT]; | 
					
						
							|  |  |  | 	uint16_t dirty_flag; | 
					
						
							|  |  |  | 	int width, height; | 
					
						
							|  |  |  | 	bool multisample; | 
					
						
							|  |  |  | 	/* TODO Check that we always use the right context when binding
 | 
					
						
							| 
									
										
										
										
											2018-09-24 18:46:51 +02:00
										 |  |  | 	 * (FBOs are not shared across ogl contexts). */ | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	// void *ctx;
 | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | static GLenum convert_attachment_type_to_gl(GPUAttachmentType type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	static const GLenum table[] = { | 
					
						
							|  |  |  | 		[GPU_FB_DEPTH_ATTACHMENT] = GL_DEPTH_ATTACHMENT, | 
					
						
							|  |  |  | 		[GPU_FB_DEPTH_STENCIL_ATTACHMENT] = GL_DEPTH_STENCIL_ATTACHMENT, | 
					
						
							|  |  |  | 		[GPU_FB_COLOR_ATTACHMENT0] = GL_COLOR_ATTACHMENT0, | 
					
						
							|  |  |  | 		[GPU_FB_COLOR_ATTACHMENT1] = GL_COLOR_ATTACHMENT1, | 
					
						
							|  |  |  | 		[GPU_FB_COLOR_ATTACHMENT2] = GL_COLOR_ATTACHMENT2, | 
					
						
							|  |  |  | 		[GPU_FB_COLOR_ATTACHMENT3] = GL_COLOR_ATTACHMENT3, | 
					
						
							| 
									
										
										
										
											2019-02-03 14:01:45 +11:00
										 |  |  | 		[GPU_FB_COLOR_ATTACHMENT4] = GL_COLOR_ATTACHMENT4, | 
					
						
							| 
									
										
										
										
											2018-04-16 17:26:55 +02:00
										 |  |  | 	}; | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	return table[type]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static GPUAttachmentType attachment_type_from_tex(GPUTexture *tex, int slot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	switch (GPU_texture_format(tex)) { | 
					
						
							|  |  |  | 		case GPU_DEPTH_COMPONENT32F: | 
					
						
							|  |  |  | 		case GPU_DEPTH_COMPONENT24: | 
					
						
							|  |  |  | 		case GPU_DEPTH_COMPONENT16: | 
					
						
							|  |  |  | 			return GPU_FB_DEPTH_ATTACHMENT; | 
					
						
							|  |  |  | 		case GPU_DEPTH24_STENCIL8: | 
					
						
							| 
									
										
										
										
											2018-10-25 19:08:35 +02:00
										 |  |  | 		case GPU_DEPTH32F_STENCIL8: | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 			return GPU_FB_DEPTH_STENCIL_ATTACHMENT; | 
					
						
							|  |  |  | 		default: | 
					
						
							|  |  |  | 			return GPU_FB_COLOR_ATTACHMENT0 + slot; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-23 14:15:43 +11:00
										 |  |  | static GLenum convert_buffer_bits_to_gl(eGPUFrameBufferBits bits) | 
					
						
							| 
									
										
										
										
											2018-03-25 14:18:39 +02:00
										 |  |  | { | 
					
						
							|  |  |  | 	GLbitfield mask = 0; | 
					
						
							|  |  |  | 	mask |= (bits & GPU_DEPTH_BIT) ? GL_DEPTH_BUFFER_BIT : 0; | 
					
						
							|  |  |  | 	mask |= (bits & GPU_STENCIL_BIT) ? GL_STENCIL_BUFFER_BIT : 0; | 
					
						
							|  |  |  | 	mask |= (bits & GPU_COLOR_BIT) ? GL_COLOR_BUFFER_BIT : 0; | 
					
						
							|  |  |  | 	return mask; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | static GPUTexture *framebuffer_get_depth_tex(GPUFrameBuffer *fb) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (fb->attachments[GPU_FB_DEPTH_ATTACHMENT].tex) | 
					
						
							|  |  |  | 		return fb->attachments[GPU_FB_DEPTH_ATTACHMENT].tex; | 
					
						
							|  |  |  | 	else | 
					
						
							| 
									
										
										
										
											2018-07-18 23:09:31 +10:00
										 |  |  | 		return fb->attachments[GPU_FB_DEPTH_STENCIL_ATTACHMENT].tex; | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static GPUTexture *framebuffer_get_color_tex(GPUFrameBuffer *fb, int slot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return fb->attachments[GPU_FB_COLOR_ATTACHMENT0 + slot].tex; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-12 20:18:44 +10:00
										 |  |  | static void gpu_print_framebuffer_error(GLenum status, char err_out[256]) | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-02-03 16:01:32 +01:00
										 |  |  | 	const char *format = "GPUFrameBuffer: framebuffer status %s\n"; | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 	const char *err = "unknown"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-08 18:02:40 -04:00
										 |  |  | #define format_status(X) \
 | 
					
						
							|  |  |  | 	case GL_FRAMEBUFFER_##X: err = "GL_FRAMEBUFFER_"#X; \ | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 	switch (status) { | 
					
						
							| 
									
										
										
										
											2016-08-08 18:02:40 -04:00
										 |  |  | 		/* success */ | 
					
						
							|  |  |  | 		format_status(COMPLETE) | 
					
						
							|  |  |  | 		/* errors shared by OpenGL desktop & ES */ | 
					
						
							|  |  |  | 		format_status(INCOMPLETE_ATTACHMENT) | 
					
						
							|  |  |  | 		format_status(INCOMPLETE_MISSING_ATTACHMENT) | 
					
						
							|  |  |  | 		format_status(UNSUPPORTED) | 
					
						
							|  |  |  | #if 0 /* for OpenGL ES only */
 | 
					
						
							|  |  |  | 		format_status(INCOMPLETE_DIMENSIONS) | 
					
						
							|  |  |  | #else /* for desktop GL only */
 | 
					
						
							|  |  |  | 		format_status(INCOMPLETE_DRAW_BUFFER) | 
					
						
							|  |  |  | 		format_status(INCOMPLETE_READ_BUFFER) | 
					
						
							|  |  |  | 		format_status(INCOMPLETE_MULTISAMPLE) | 
					
						
							|  |  |  | 		format_status(UNDEFINED) | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-08 18:02:40 -04:00
										 |  |  | #undef format_status
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 	if (err_out) { | 
					
						
							| 
									
										
										
										
											2016-08-08 18:02:40 -04:00
										 |  |  | 		BLI_snprintf(err_out, 256, format, err); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2016-08-08 18:02:40 -04:00
										 |  |  | 		fprintf(stderr, format, err); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-11 20:50:14 +02:00
										 |  |  | void gpu_framebuffer_module_init(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void gpu_framebuffer_module_exit(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-31 18:16:08 +02:00
										 |  |  | GPUFrameBuffer *GPU_framebuffer_active_get(void) | 
					
						
							| 
									
										
										
										
											2018-06-11 20:50:14 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-07-31 18:16:08 +02:00
										 |  |  | 	GPUContext *ctx = GPU_context_active_get(); | 
					
						
							|  |  |  | 	if (ctx) { | 
					
						
							|  |  |  | 		return gpu_context_active_framebuffer_get(ctx); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-06-11 20:50:14 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-31 18:16:08 +02:00
										 |  |  | static void gpu_framebuffer_current_set(GPUFrameBuffer *fb) | 
					
						
							| 
									
										
										
										
											2018-06-11 20:50:14 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-07-31 18:16:08 +02:00
										 |  |  | 	GPUContext *ctx = GPU_context_active_get(); | 
					
						
							|  |  |  | 	if (ctx) { | 
					
						
							|  |  |  | 		gpu_context_active_framebuffer_set(ctx, fb); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-06-11 20:50:14 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | /* GPUFrameBuffer */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GPUFrameBuffer *GPU_framebuffer_create(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	/* We generate the FB object later at first use in order to
 | 
					
						
							|  |  |  | 	 * create the framebuffer in the right opengl context. */ | 
					
						
							| 
									
										
										
										
											2018-09-20 14:47:10 +00:00
										 |  |  | 	return MEM_callocN(sizeof(GPUFrameBuffer), "GPUFrameBuffer"); | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | static void gpu_framebuffer_init(GPUFrameBuffer *fb) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-07-19 15:48:13 +02:00
										 |  |  | 	fb->object = GPU_fbo_alloc(); | 
					
						
							|  |  |  | 	fb->ctx = GPU_context_active_get(); | 
					
						
							|  |  |  | 	gpu_context_add_framebuffer(fb->ctx, fb); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | void GPU_framebuffer_free(GPUFrameBuffer *fb) | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	for (GPUAttachmentType type = 0; type < GPU_FB_MAX_ATTACHEMENT; type++) { | 
					
						
							|  |  |  | 		if (fb->attachments[type].tex != NULL) { | 
					
						
							|  |  |  | 			GPU_framebuffer_texture_detach(fb, fb->attachments[type].tex); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-19 15:48:13 +02:00
										 |  |  | 	if (fb->object != 0) { | 
					
						
							|  |  |  | 		/* This restores the framebuffer if it was bound */ | 
					
						
							|  |  |  | 		GPU_fbo_free(fb->object, fb->ctx); | 
					
						
							|  |  |  | 		gpu_context_remove_framebuffer(fb->ctx, fb); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-31 18:16:08 +02:00
										 |  |  | 	if (GPU_framebuffer_active_get() == fb) { | 
					
						
							|  |  |  | 		gpu_framebuffer_current_set(NULL); | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	MEM_freeN(fb); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | /* ---------- Attach ----------- */ | 
					
						
							| 
									
										
										
										
											2017-06-08 20:12:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | static void gpu_framebuffer_texture_attach_ex(GPUFrameBuffer *fb, GPUTexture *tex, int slot, int layer, int mip) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (slot >= GPU_FB_MAX_COLOR_ATTACHMENT) { | 
					
						
							| 
									
										
										
										
											2017-06-08 20:12:58 +02:00
										 |  |  | 		fprintf(stderr, | 
					
						
							|  |  |  | 		        "Attaching to index %d framebuffer slot unsupported. " | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 		        "Use at most %d\n", slot, GPU_FB_MAX_COLOR_ATTACHMENT); | 
					
						
							|  |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2017-06-08 20:12:58 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	GPUAttachmentType type = attachment_type_from_tex(tex, slot); | 
					
						
							|  |  |  | 	GPUAttachment *attachment = &fb->attachments[type]; | 
					
						
							| 
									
										
										
										
											2017-06-08 20:12:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	if ((attachment->tex == tex) && | 
					
						
							| 
									
										
										
										
											2018-04-19 07:47:03 +02:00
										 |  |  | 	    (attachment->mip == mip) && | 
					
						
							|  |  |  | 	    (attachment->layer == layer)) | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 		return; /* Exact same texture already bound here. */ | 
					
						
							| 
									
										
										
										
											2017-06-16 13:25:22 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	else if (attachment->tex != NULL) { | 
					
						
							|  |  |  | 		GPU_framebuffer_texture_detach(fb, attachment->tex); | 
					
						
							| 
									
										
										
										
											2017-06-16 13:25:22 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-06-08 20:12:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	if (attachment->tex == NULL) { | 
					
						
							|  |  |  | 		GPU_texture_attach_framebuffer(tex, fb, type); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-06-08 20:12:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	attachment->tex = tex; | 
					
						
							|  |  |  | 	attachment->mip = mip; | 
					
						
							|  |  |  | 	attachment->layer = layer; | 
					
						
							|  |  |  | 	GPU_FB_ATTACHEMENT_SET_DIRTY(fb->dirty_flag, type); | 
					
						
							| 
									
										
										
										
											2017-06-08 20:12:58 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | void GPU_framebuffer_texture_attach(GPUFrameBuffer *fb, GPUTexture *tex, int slot, int mip) | 
					
						
							| 
									
										
										
										
											2017-06-16 13:25:22 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-26 17:54:35 +02:00
										 |  |  | 	gpu_framebuffer_texture_attach_ex(fb, tex, slot, -1, mip); | 
					
						
							| 
									
										
										
										
											2017-06-16 13:25:22 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | void GPU_framebuffer_texture_layer_attach(GPUFrameBuffer *fb, GPUTexture *tex, int slot, int layer, int mip) | 
					
						
							| 
									
										
										
										
											2017-06-16 13:25:22 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	/* NOTE: We could support 1D ARRAY texture. */ | 
					
						
							|  |  |  | 	BLI_assert(GPU_texture_target(tex) == GL_TEXTURE_2D_ARRAY); | 
					
						
							| 
									
										
										
										
											2018-03-26 17:54:35 +02:00
										 |  |  | 	gpu_framebuffer_texture_attach_ex(fb, tex, slot, layer, mip); | 
					
						
							| 
									
										
										
										
											2017-06-16 13:25:22 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | void GPU_framebuffer_texture_cubeface_attach(GPUFrameBuffer *fb, GPUTexture *tex, int slot, int face, int mip) | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	BLI_assert(GPU_texture_cube(tex)); | 
					
						
							| 
									
										
										
										
											2018-03-26 17:54:35 +02:00
										 |  |  | 	gpu_framebuffer_texture_attach_ex(fb, tex, slot, face, mip); | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | /* ---------- Detach ----------- */ | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | void GPU_framebuffer_texture_detach_slot(GPUFrameBuffer *fb, GPUTexture *tex, int type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	GPUAttachment *attachment = &fb->attachments[type]; | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	if (attachment->tex != tex) { | 
					
						
							|  |  |  | 		fprintf(stderr, | 
					
						
							|  |  |  | 		        "Warning, attempting to detach Texture %p from framebuffer %p " | 
					
						
							|  |  |  | 		        "but texture is not attached.\n", tex, fb); | 
					
						
							|  |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	attachment->tex = NULL; | 
					
						
							|  |  |  | 	GPU_FB_ATTACHEMENT_SET_DIRTY(fb->dirty_flag, type); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | void GPU_framebuffer_texture_detach(GPUFrameBuffer *fb, GPUTexture *tex) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	GPUAttachmentType type = GPU_texture_detach_framebuffer(tex, fb); | 
					
						
							|  |  |  | 	GPU_framebuffer_texture_detach_slot(fb, tex, type); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | /* ---------- Config (Attach & Detach) ----------- */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * 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. | 
					
						
							|  |  |  |  **/ | 
					
						
							| 
									
										
										
										
											2018-07-08 13:14:49 +02:00
										 |  |  | void GPU_framebuffer_config_array(GPUFrameBuffer *fb, const GPUAttachment *config, int config_len) | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	if (config[0].tex) { | 
					
						
							|  |  |  | 		BLI_assert(GPU_texture_depth(config[0].tex)); | 
					
						
							|  |  |  | 		gpu_framebuffer_texture_attach_ex(fb, config[0].tex, 0, config[0].layer, config[0].mip); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else if (config[0].mip == -1) { | 
					
						
							|  |  |  | 		/* Leave texture attached */ | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else if (fb->attachments[GPU_FB_DEPTH_ATTACHMENT].tex != NULL) { | 
					
						
							|  |  |  | 		GPU_framebuffer_texture_detach(fb, fb->attachments[GPU_FB_DEPTH_ATTACHMENT].tex); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else if (fb->attachments[GPU_FB_DEPTH_STENCIL_ATTACHMENT].tex != NULL) { | 
					
						
							|  |  |  | 		GPU_framebuffer_texture_detach(fb, fb->attachments[GPU_FB_DEPTH_STENCIL_ATTACHMENT].tex); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	int slot = 0; | 
					
						
							| 
									
										
										
										
											2018-07-08 13:14:49 +02:00
										 |  |  | 	for (int i = 1; i < config_len; ++i, ++slot) { | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 		if (config[i].tex != NULL) { | 
					
						
							|  |  |  | 			BLI_assert(GPU_texture_depth(config[i].tex) == false); | 
					
						
							|  |  |  | 			gpu_framebuffer_texture_attach_ex(fb, config[i].tex, slot, config[i].layer, config[i].mip); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else if (config[i].mip != -1) { | 
					
						
							|  |  |  | 			GPUTexture *tex = framebuffer_get_color_tex(fb, slot); | 
					
						
							|  |  |  | 			if (tex != NULL) { | 
					
						
							|  |  |  | 				GPU_framebuffer_texture_detach(fb, tex); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | /* ---------- Bind / Restore ----------- */ | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | static void gpu_framebuffer_attachment_attach(GPUAttachment *attach, GPUAttachmentType attach_type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int tex_bind = GPU_texture_opengl_bindcode(attach->tex); | 
					
						
							|  |  |  | 	GLenum gl_attachment = convert_attachment_type_to_gl(attach_type); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	if (attach->layer > -1) { | 
					
						
							|  |  |  | 		if (GPU_texture_cube(attach->tex)) { | 
					
						
							| 
									
										
										
										
											2018-07-18 23:09:31 +10:00
										 |  |  | 			glFramebufferTexture2D( | 
					
						
							|  |  |  | 			        GL_FRAMEBUFFER, gl_attachment, GL_TEXTURE_CUBE_MAP_POSITIVE_X + attach->layer, | 
					
						
							|  |  |  | 			        tex_bind, attach->mip); | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							|  |  |  | 			glFramebufferTextureLayer(GL_FRAMEBUFFER, gl_attachment, tex_bind, attach->mip, attach->layer); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 		glFramebufferTexture(GL_FRAMEBUFFER, gl_attachment, tex_bind, attach->mip); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | static void gpu_framebuffer_attachment_detach(GPUAttachment *UNUSED(attachment), GPUAttachmentType attach_type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	GLenum gl_attachment = convert_attachment_type_to_gl(attach_type); | 
					
						
							|  |  |  | 	glFramebufferTexture(GL_FRAMEBUFFER, gl_attachment, 0, 0); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | static void gpu_framebuffer_update_attachments(GPUFrameBuffer *fb) | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	GLenum gl_attachments[GPU_FB_MAX_COLOR_ATTACHMENT]; | 
					
						
							|  |  |  | 	int numslots = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-31 18:16:08 +02:00
										 |  |  | 	BLI_assert(GPU_framebuffer_active_get() == fb); | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* Update attachments */ | 
					
						
							|  |  |  | 	for (GPUAttachmentType type = 0; type < GPU_FB_MAX_ATTACHEMENT; ++type) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (type >= GPU_FB_COLOR_ATTACHMENT0) { | 
					
						
							|  |  |  | 			if (fb->attachments[type].tex) { | 
					
						
							|  |  |  | 				gl_attachments[numslots] = convert_attachment_type_to_gl(type); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else { | 
					
						
							|  |  |  | 				gl_attachments[numslots] = GL_NONE; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 			numslots++; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 		if (GPU_FB_ATTACHEMENT_IS_DIRTY(fb->dirty_flag, type) == false) { | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else if (fb->attachments[type].tex != NULL) { | 
					
						
							|  |  |  | 			gpu_framebuffer_attachment_attach(&fb->attachments[type], type); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 			fb->multisample = (GPU_texture_samples(fb->attachments[type].tex) > 0); | 
					
						
							|  |  |  | 			fb->width = GPU_texture_width(fb->attachments[type].tex); | 
					
						
							|  |  |  | 			fb->height = GPU_texture_height(fb->attachments[type].tex); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							|  |  |  | 			gpu_framebuffer_attachment_detach(&fb->attachments[type], type); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	fb->dirty_flag = 0; | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	/* Update draw buffers (color targets)
 | 
					
						
							|  |  |  | 	 * This state is saved in the FBO */ | 
					
						
							|  |  |  | 	if (numslots) | 
					
						
							|  |  |  | 		glDrawBuffers(numslots, gl_attachments); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		glDrawBuffer(GL_NONE); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-05 20:59:22 -02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Hack to solve the problem of some bugged AMD GPUs (see `GPU_unused_fb_slot_workaround`). | 
					
						
							|  |  |  |  * If there is an empty color slot between the color slots, | 
					
						
							|  |  |  |  * all textures after this slot are apparently skipped/discarded. | 
					
						
							|  |  |  |  **/ | 
					
						
							|  |  |  | static void gpu_framebuffer_update_attachments_and_fill_empty_slots(GPUFrameBuffer *fb) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	GLenum gl_attachments[GPU_FB_MAX_COLOR_ATTACHMENT]; | 
					
						
							|  |  |  | 	int dummy_tex = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	BLI_assert(GPU_framebuffer_active_get() == fb); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Update attachments */ | 
					
						
							|  |  |  | 	for (GPUAttachmentType type = GPU_FB_MAX_ATTACHEMENT; type--;) { | 
					
						
							|  |  |  | 		GPUTexture *tex = fb->attachments[type].tex; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (type >= GPU_FB_COLOR_ATTACHMENT0) { | 
					
						
							|  |  |  | 			int slot = type - GPU_FB_COLOR_ATTACHMENT0; | 
					
						
							| 
									
										
										
										
											2018-12-06 11:47:37 -02:00
										 |  |  | 			if (tex != NULL || (dummy_tex != 0)) { | 
					
						
							| 
									
										
										
										
											2018-12-05 20:59:22 -02:00
										 |  |  | 				gl_attachments[slot] = convert_attachment_type_to_gl(type); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-06 11:47:37 -02:00
										 |  |  | 				if (dummy_tex == 0) { | 
					
						
							| 
									
										
										
										
											2018-12-05 20:59:22 -02:00
										 |  |  | 					dummy_tex = GPU_texture_opengl_bindcode(tex); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else { | 
					
						
							|  |  |  | 				gl_attachments[slot] = GL_NONE; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							|  |  |  | 			dummy_tex = 0; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-06 11:47:37 -02:00
										 |  |  | 		if ((dummy_tex != 0) && tex == NULL) { | 
					
						
							|  |  |  | 			/* Fill empty slot */ | 
					
						
							|  |  |  | 			glFramebufferTexture(GL_FRAMEBUFFER, convert_attachment_type_to_gl(type), dummy_tex, 0); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else if (GPU_FB_ATTACHEMENT_IS_DIRTY(fb->dirty_flag, type)) { | 
					
						
							| 
									
										
										
										
											2018-12-05 20:59:22 -02:00
										 |  |  | 			if (tex != NULL) { | 
					
						
							|  |  |  | 				gpu_framebuffer_attachment_attach(&fb->attachments[type], type); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				fb->multisample = (GPU_texture_samples(tex) > 0); | 
					
						
							|  |  |  | 				fb->width = GPU_texture_width(tex); | 
					
						
							|  |  |  | 				fb->height = GPU_texture_height(tex); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else { | 
					
						
							| 
									
										
										
										
											2018-12-06 11:47:37 -02:00
										 |  |  | 				gpu_framebuffer_attachment_detach(&fb->attachments[type], type); | 
					
						
							| 
									
										
										
										
											2018-12-05 20:59:22 -02:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	fb->dirty_flag = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Update draw buffers (color targets)
 | 
					
						
							|  |  |  | 	 * This state is saved in the FBO */ | 
					
						
							|  |  |  | 	glDrawBuffers(GPU_FB_MAX_COLOR_ATTACHMENT, gl_attachments); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-06 15:24:13 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #define FRAMEBUFFER_STACK_DEPTH 16
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct { | 
					
						
							|  |  |  | 	GPUFrameBuffer *framebuffers[FRAMEBUFFER_STACK_DEPTH]; | 
					
						
							|  |  |  | 	uint top; | 
					
						
							| 
									
										
										
										
											2018-12-30 15:14:00 +11:00
										 |  |  | } FrameBufferStack = {{0}}; | 
					
						
							| 
									
										
										
										
											2018-11-06 15:24:13 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | static void gpuPushFrameBuffer(GPUFrameBuffer *fbo) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	BLI_assert(FrameBufferStack.top < FRAMEBUFFER_STACK_DEPTH); | 
					
						
							|  |  |  | 	FrameBufferStack.framebuffers[FrameBufferStack.top] = fbo; | 
					
						
							|  |  |  | 	FrameBufferStack.top++; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static GPUFrameBuffer *gpuPopFrameBuffer(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	BLI_assert(FrameBufferStack.top > 0); | 
					
						
							|  |  |  | 	FrameBufferStack.top--; | 
					
						
							|  |  |  | 	return FrameBufferStack.framebuffers[FrameBufferStack.top]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #undef FRAMEBUFFER_STACK_DEPTH
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-07 11:20:15 +01:00
										 |  |  | void GPU_framebuffer_bind(GPUFrameBuffer *fb) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	if (fb->object == 0) | 
					
						
							|  |  |  | 		gpu_framebuffer_init(fb); | 
					
						
							| 
									
										
										
										
											2017-02-07 11:20:15 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-31 18:16:08 +02:00
										 |  |  | 	if (GPU_framebuffer_active_get() != fb) | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 		glBindFramebuffer(GL_FRAMEBUFFER, fb->object); | 
					
						
							| 
									
										
										
										
											2017-02-07 11:20:15 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-31 18:16:08 +02:00
										 |  |  | 	gpu_framebuffer_current_set(fb); | 
					
						
							| 
									
										
										
										
											2017-02-07 11:20:15 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-05 20:59:22 -02:00
										 |  |  | 	if (fb->dirty_flag != 0) { | 
					
						
							|  |  |  | 		if (GPU_unused_fb_slot_workaround()) { | 
					
						
							|  |  |  | 			/* XXX: Please AMD, fix this. */ | 
					
						
							|  |  |  | 			gpu_framebuffer_update_attachments_and_fill_empty_slots(fb); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							|  |  |  | 			gpu_framebuffer_update_attachments(fb); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-02-07 11:20:15 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	/* TODO manually check for errors? */ | 
					
						
							|  |  |  | #if 0
 | 
					
						
							|  |  |  | 	char err_out[256]; | 
					
						
							|  |  |  | 	if (!GPU_framebuffer_check_valid(fb, err_out)) { | 
					
						
							|  |  |  | 		printf("Invalid %s\n", err_out); | 
					
						
							| 
									
										
										
										
											2017-02-07 11:20:15 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2017-02-07 11:20:15 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	if (fb->multisample) | 
					
						
							| 
									
										
										
										
											2017-09-26 14:01:50 +02:00
										 |  |  | 		glEnable(GL_MULTISAMPLE); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	glViewport(0, 0, fb->width, fb->height); | 
					
						
							| 
									
										
										
										
											2017-02-07 11:20:15 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | void GPU_framebuffer_restore(void) | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-07-31 18:16:08 +02:00
										 |  |  | 	if (GPU_framebuffer_active_get() != NULL) { | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 		glBindFramebuffer(GL_FRAMEBUFFER, 0); | 
					
						
							| 
									
										
										
										
											2018-07-31 18:16:08 +02:00
										 |  |  | 		gpu_framebuffer_current_set(NULL); | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | bool GPU_framebuffer_bound(GPUFrameBuffer *fb) | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-07-31 18:16:08 +02:00
										 |  |  | 	return (fb == GPU_framebuffer_active_get()) && (fb->object != 0); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool GPU_framebuffer_check_valid(GPUFrameBuffer *fb, char err_out[256]) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-04-03 09:35:22 +02:00
										 |  |  | 	if (!GPU_framebuffer_bound(fb)) | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 		GPU_framebuffer_bind(fb); | 
					
						
							| 
									
										
										
										
											2018-02-17 14:05:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-19 00:52:52 -04:00
										 |  |  | 	GLenum status = glCheckFramebufferStatus(GL_FRAMEBUFFER); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-04 21:11:45 -04:00
										 |  |  | 	if (status != GL_FRAMEBUFFER_COMPLETE) { | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 		GPU_framebuffer_restore(); | 
					
						
							| 
									
										
										
										
											2017-04-12 20:18:44 +10:00
										 |  |  | 		gpu_print_framebuffer_error(status, err_out); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 		return false; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-08-19 00:52:52 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 	return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | /* ---------- Framebuffer Operations ----------- */ | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 14:18:39 +02:00
										 |  |  | #define CHECK_FRAMEBUFFER_IS_BOUND(_fb) \
 | 
					
						
							|  |  |  | 	BLI_assert(GPU_framebuffer_bound(_fb)); \ | 
					
						
							|  |  |  | 	UNUSED_VARS_NDEBUG(_fb); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Needs to be done after binding. */ | 
					
						
							|  |  |  | void GPU_framebuffer_viewport_set(GPUFrameBuffer *fb, int x, int y, int w, int h) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	CHECK_FRAMEBUFFER_IS_BOUND(fb); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	glViewport(x, y, w, h); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GPU_framebuffer_clear( | 
					
						
							| 
									
										
										
										
											2019-01-23 14:15:43 +11:00
										 |  |  |         GPUFrameBuffer *fb, eGPUFrameBufferBits buffers, | 
					
						
							| 
									
										
										
										
											2018-09-12 12:18:35 +10:00
										 |  |  |         const float clear_col[4], float clear_depth, uint clear_stencil) | 
					
						
							| 
									
										
										
										
											2018-03-25 14:18:39 +02:00
										 |  |  | { | 
					
						
							|  |  |  | 	CHECK_FRAMEBUFFER_IS_BOUND(fb); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (buffers & GPU_COLOR_BIT) { | 
					
						
							|  |  |  | 		glColorMask(GL_TRUE, GL_TRUE, GL_TRUE, GL_TRUE); | 
					
						
							|  |  |  | 		glClearColor(clear_col[0], clear_col[1], clear_col[2], clear_col[3]); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (buffers & GPU_DEPTH_BIT) { | 
					
						
							|  |  |  | 		glDepthMask(GL_TRUE); | 
					
						
							|  |  |  | 		glClearDepth(clear_depth); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (buffers & GPU_STENCIL_BIT) { | 
					
						
							| 
									
										
										
										
											2018-08-15 17:41:36 +02:00
										 |  |  | 		glStencilMask(0xFF); | 
					
						
							|  |  |  | 		glClearStencil(clear_stencil); | 
					
						
							| 
									
										
										
										
											2018-03-25 14:18:39 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	GLbitfield mask = convert_buffer_bits_to_gl(buffers); | 
					
						
							|  |  |  | 	glClear(mask); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | void GPU_framebuffer_read_depth(GPUFrameBuffer *fb, int x, int y, int w, int h, float *data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	CHECK_FRAMEBUFFER_IS_BOUND(fb); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	GLenum type = GL_DEPTH_COMPONENT; | 
					
						
							|  |  |  | 	glReadBuffer(GL_COLOR_ATTACHMENT0); /* This is OK! */ | 
					
						
							|  |  |  | 	glReadPixels(x, y, w, h, type, GL_FLOAT, data); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GPU_framebuffer_read_color( | 
					
						
							|  |  |  |         GPUFrameBuffer *fb, int x, int y, int w, int h, int channels, int slot, float *data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	CHECK_FRAMEBUFFER_IS_BOUND(fb); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	GLenum type; | 
					
						
							|  |  |  | 	switch (channels) { | 
					
						
							|  |  |  | 		case 1: type = GL_RED; break; | 
					
						
							|  |  |  | 		case 2: type = GL_RG; break; | 
					
						
							|  |  |  | 		case 3: type = GL_RGB; break; | 
					
						
							| 
									
										
										
										
											2018-07-18 23:09:31 +10:00
										 |  |  | 		case 4: type = GL_RGBA; break; | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 		default: | 
					
						
							|  |  |  | 			BLI_assert(false && "wrong number of read channels"); | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	glReadBuffer(GL_COLOR_ATTACHMENT0 + slot); | 
					
						
							|  |  |  | 	glReadPixels(x, y, w, h, type, GL_FLOAT, data); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* read_slot and write_slot are only used for color buffers. */ | 
					
						
							| 
									
										
										
										
											2017-11-10 23:36:05 +01:00
										 |  |  | void GPU_framebuffer_blit( | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  |         GPUFrameBuffer *fb_read, int read_slot, | 
					
						
							|  |  |  |         GPUFrameBuffer *fb_write, int write_slot, | 
					
						
							| 
									
										
										
										
											2019-01-23 14:15:43 +11:00
										 |  |  |         eGPUFrameBufferBits blit_buffers) | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | { | 
					
						
							|  |  |  | 	BLI_assert(blit_buffers != 0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-31 18:16:08 +02:00
										 |  |  | 	GPUFrameBuffer *prev_fb = GPU_framebuffer_active_get(); | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* Framebuffers must be up to date. This simplify this function. */ | 
					
						
							|  |  |  | 	if (fb_read->dirty_flag != 0 || fb_read->object == 0) { | 
					
						
							|  |  |  | 		GPU_framebuffer_bind(fb_read); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (fb_write->dirty_flag != 0 || fb_write->object == 0) { | 
					
						
							|  |  |  | 		GPU_framebuffer_bind(fb_write); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const bool do_color = (blit_buffers & GPU_COLOR_BIT); | 
					
						
							|  |  |  | 	const bool do_depth = (blit_buffers & GPU_DEPTH_BIT); | 
					
						
							|  |  |  | 	const bool do_stencil = (blit_buffers & GPU_STENCIL_BIT); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 23:09:31 +10:00
										 |  |  | 	GPUTexture *read_tex = ( | 
					
						
							|  |  |  | 	        (do_depth || do_stencil) ? | 
					
						
							|  |  |  | 	        framebuffer_get_depth_tex(fb_read) : | 
					
						
							|  |  |  | 	        framebuffer_get_color_tex(fb_read, read_slot)); | 
					
						
							|  |  |  | 	GPUTexture *write_tex = ( | 
					
						
							|  |  |  | 	        (do_depth || do_stencil) ? | 
					
						
							|  |  |  | 	        framebuffer_get_depth_tex(fb_write) : | 
					
						
							|  |  |  | 	        framebuffer_get_color_tex(fb_write, read_slot)); | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (do_depth) { | 
					
						
							| 
									
										
										
										
											2017-11-10 23:36:05 +01:00
										 |  |  | 		BLI_assert(GPU_texture_depth(read_tex) && GPU_texture_depth(write_tex)); | 
					
						
							| 
									
										
										
										
											2017-11-14 20:49:13 +01:00
										 |  |  | 		BLI_assert(GPU_texture_format(read_tex) == GPU_texture_format(write_tex)); | 
					
						
							| 
									
										
										
										
											2017-11-10 23:36:05 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	if (do_stencil) { | 
					
						
							| 
									
										
										
										
											2017-11-10 23:36:05 +01:00
										 |  |  | 		BLI_assert(GPU_texture_stencil(read_tex) && GPU_texture_stencil(write_tex)); | 
					
						
							| 
									
										
										
										
											2017-11-14 20:49:13 +01:00
										 |  |  | 		BLI_assert(GPU_texture_format(read_tex) == GPU_texture_format(write_tex)); | 
					
						
							| 
									
										
										
										
											2017-11-10 23:36:05 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	if (GPU_texture_samples(write_tex) != 0 || | 
					
						
							| 
									
										
										
										
											2018-04-19 07:47:03 +02:00
										 |  |  | 	    GPU_texture_samples(read_tex) != 0) | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 		/* Can only blit multisample textures to another texture of the same size. */ | 
					
						
							|  |  |  | 		BLI_assert((fb_read->width == fb_write->width) && | 
					
						
							|  |  |  | 		           (fb_read->height == fb_write->height)); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-11-10 23:36:05 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 15:15:42 +01:00
										 |  |  | 	glBindFramebuffer(GL_READ_FRAMEBUFFER, fb_read->object); | 
					
						
							|  |  |  | 	glBindFramebuffer(GL_DRAW_FRAMEBUFFER, fb_write->object); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	if (do_color) { | 
					
						
							|  |  |  | 		glReadBuffer(GL_COLOR_ATTACHMENT0 + read_slot); | 
					
						
							|  |  |  | 		glDrawBuffer(GL_COLOR_ATTACHMENT0 + write_slot); | 
					
						
							|  |  |  | 		/* XXX we messed with the glDrawBuffer, this will reset the
 | 
					
						
							|  |  |  | 		 * glDrawBuffers the next time we bind fb_write. */ | 
					
						
							|  |  |  | 		fb_write->dirty_flag = GPU_FB_DIRTY_DRAWBUFFER; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	GLbitfield mask = convert_buffer_bits_to_gl(blit_buffers); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 23:09:31 +10:00
										 |  |  | 	glBlitFramebuffer( | 
					
						
							|  |  |  | 	        0, 0, fb_read->width, fb_read->height, | 
					
						
							|  |  |  | 	        0, 0, fb_write->width, fb_write->height, | 
					
						
							|  |  |  | 	        mask, GL_NEAREST); | 
					
						
							| 
									
										
										
										
											2017-02-15 15:15:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* Restore previous framebuffer */ | 
					
						
							| 
									
										
										
										
											2018-07-31 18:16:08 +02:00
										 |  |  | 	if (fb_write == prev_fb) { | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 		GPU_framebuffer_bind(fb_write); /* To update drawbuffers */ | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-10-22 22:49:33 +02:00
										 |  |  | 	else if (prev_fb) { | 
					
						
							| 
									
										
										
										
											2018-07-31 18:16:08 +02:00
										 |  |  | 		glBindFramebuffer(GL_FRAMEBUFFER, prev_fb->object); | 
					
						
							| 
									
										
										
										
											2018-06-11 20:50:14 +02:00
										 |  |  | 		gpu_framebuffer_current_set(prev_fb); | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-10-22 22:49:33 +02:00
										 |  |  | 	else { | 
					
						
							|  |  |  | 		glBindFramebuffer(GL_FRAMEBUFFER, 0); | 
					
						
							|  |  |  | 		gpu_framebuffer_current_set(NULL); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-02-15 15:15:42 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-22 02:01:58 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Use this if you need to custom downsample your texture and use the previous mip level as input. | 
					
						
							|  |  |  |  * This function only takes care of the correct texture handling. It execute the callback for each texture level. | 
					
						
							|  |  |  |  **/ | 
					
						
							|  |  |  | void GPU_framebuffer_recursive_downsample( | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  |         GPUFrameBuffer *fb, int max_lvl, | 
					
						
							|  |  |  |         void (*callback)(void *userData, int level), void *userData) | 
					
						
							| 
									
										
										
										
											2017-06-22 02:01:58 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	/* Framebuffer must be up to date and bound. This simplify this function. */ | 
					
						
							| 
									
										
										
										
											2018-07-31 18:16:08 +02:00
										 |  |  | 	if (GPU_framebuffer_active_get() != fb || fb->dirty_flag != 0 || fb->object == 0) { | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 		GPU_framebuffer_bind(fb); | 
					
						
							| 
									
										
										
										
											2017-07-21 14:27:16 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	/* HACK: We make the framebuffer appear not bound in order to
 | 
					
						
							|  |  |  | 	 * not trigger any error in GPU_texture_bind().  */ | 
					
						
							| 
									
										
										
										
											2018-07-31 18:16:08 +02:00
										 |  |  | 	GPUFrameBuffer *prev_fb = GPU_framebuffer_active_get(); | 
					
						
							|  |  |  | 	gpu_framebuffer_current_set(NULL); | 
					
						
							| 
									
										
										
										
											2017-06-22 02:01:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-16 21:22:55 +02:00
										 |  |  | 	int levels = floor(log2(max_ii(fb->width, fb->height))); | 
					
						
							|  |  |  | 	max_lvl = min_ii(max_lvl, levels); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	int i; | 
					
						
							|  |  |  | 	int current_dim[2] = {fb->width, fb->height}; | 
					
						
							|  |  |  | 	for (i = 1; i < max_lvl + 1; i++) { | 
					
						
							| 
									
										
										
										
											2017-10-17 03:06:04 +02:00
										 |  |  | 		/* calculate next viewport size */ | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 		current_dim[0] = max_ii(current_dim[0] / 2, 1); | 
					
						
							|  |  |  | 		current_dim[1] = max_ii(current_dim[1] / 2, 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		for (GPUAttachmentType type = 0; type < GPU_FB_MAX_ATTACHEMENT; ++type) { | 
					
						
							|  |  |  | 			if (fb->attachments[type].tex != NULL) { | 
					
						
							| 
									
										
										
										
											2018-10-22 12:58:11 +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! */ | 
					
						
							|  |  |  | 				int next_lvl = (GPU_mip_render_workaround()) ? i : i - 1; | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 				/* bind next level for rendering but first restrict fetches only to previous level */ | 
					
						
							|  |  |  | 				GPUTexture *tex = fb->attachments[type].tex; | 
					
						
							|  |  |  | 				GPU_texture_bind(tex, 0); | 
					
						
							|  |  |  | 				glTexParameteri(GPU_texture_target(tex), GL_TEXTURE_BASE_LEVEL, i - 1); | 
					
						
							| 
									
										
										
										
											2018-10-22 12:58:11 +02:00
										 |  |  | 				glTexParameteri(GPU_texture_target(tex), GL_TEXTURE_MAX_LEVEL, next_lvl); | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 				GPU_texture_unbind(tex); | 
					
						
							|  |  |  | 				/* copy attachment and replace miplevel. */ | 
					
						
							|  |  |  | 				GPUAttachment attachment = fb->attachments[type]; | 
					
						
							|  |  |  | 				attachment.mip = i; | 
					
						
							|  |  |  | 				gpu_framebuffer_attachment_attach(&attachment, type); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2017-10-10 17:18:29 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-06-22 02:01:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-30 20:15:03 +02:00
										 |  |  | 		BLI_assert(GL_FRAMEBUFFER_COMPLETE == glCheckFramebufferStatus(GL_FRAMEBUFFER)); | 
					
						
							| 
									
										
										
										
											2018-03-29 13:43:23 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-22 02:01:58 +02:00
										 |  |  | 		glViewport(0, 0, current_dim[0], current_dim[1]); | 
					
						
							|  |  |  | 		callback(userData, i); | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		if (current_dim[0] == 1 && current_dim[1] == 1) | 
					
						
							|  |  |  | 			break; | 
					
						
							| 
									
										
										
										
											2017-06-22 02:01:58 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	for (GPUAttachmentType type = 0; type < GPU_FB_MAX_ATTACHEMENT; ++type) { | 
					
						
							|  |  |  | 		if (fb->attachments[type].tex != NULL) { | 
					
						
							|  |  |  | 			/* reset mipmap level range */ | 
					
						
							|  |  |  | 			GPUTexture *tex = fb->attachments[type].tex; | 
					
						
							|  |  |  | 			GPU_texture_bind(tex, 0); | 
					
						
							|  |  |  | 			glTexParameteri(GPU_texture_target(tex), GL_TEXTURE_BASE_LEVEL, 0); | 
					
						
							|  |  |  | 			glTexParameteri(GPU_texture_target(tex), GL_TEXTURE_MAX_LEVEL, i - 1); | 
					
						
							|  |  |  | 			GPU_texture_unbind(tex); | 
					
						
							|  |  |  | 			/* Reattach original level */ | 
					
						
							|  |  |  | 			/* NOTE: This is not necessary but this makes the FBO config
 | 
					
						
							|  |  |  | 			 *       remain in sync with the GPUFrameBuffer config. */ | 
					
						
							|  |  |  | 			gpu_framebuffer_attachment_attach(&fb->attachments[type], type); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-06-22 02:01:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-11 20:50:14 +02:00
										 |  |  | 	gpu_framebuffer_current_set(prev_fb); | 
					
						
							| 
									
										
										
										
											2017-06-22 02:01:58 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | /* GPUOffScreen */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct GPUOffScreen { | 
					
						
							|  |  |  | 	GPUFrameBuffer *fb; | 
					
						
							|  |  |  | 	GPUTexture *color; | 
					
						
							|  |  |  | 	GPUTexture *depth; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-13 18:18:04 +01:00
										 |  |  | GPUOffScreen *GPU_offscreen_create(int width, int height, int samples, bool depth, bool high_bitdepth, char err_out[256]) | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | { | 
					
						
							|  |  |  | 	GPUOffScreen *ofs; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ofs = MEM_callocN(sizeof(GPUOffScreen), "GPUOffScreen"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-22 16:30:17 +01: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); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 23:09:31 +10:00
										 |  |  | 	ofs->color = GPU_texture_create_2D_multisample( | 
					
						
							|  |  |  | 	        width, height, | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	        (high_bitdepth) ? GPU_RGBA16F : GPU_RGBA8, NULL, samples, err_out); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-13 18:18:04 +01:00
										 |  |  | 	if (depth) { | 
					
						
							| 
									
										
										
										
											2018-04-30 16:02:24 +02:00
										 |  |  | 		ofs->depth = GPU_texture_create_2D_multisample(width, height, GPU_DEPTH24_STENCIL8, NULL, samples, err_out); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-20 20:51:54 +02:00
										 |  |  | 	if ((depth && !ofs->depth) || !ofs->color) { | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 		GPU_offscreen_free(ofs); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-04-20 20:51:54 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-29 07:46:25 +11:00
										 |  |  | 	gpuPushAttr(GPU_VIEWPORT_BIT); | 
					
						
							| 
									
										
										
										
											2018-04-20 20:51:54 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	GPU_framebuffer_ensure_config(&ofs->fb, { | 
					
						
							|  |  |  | 		GPU_ATTACHMENT_TEXTURE(ofs->depth), | 
					
						
							|  |  |  | 		GPU_ATTACHMENT_TEXTURE(ofs->color) | 
					
						
							|  |  |  | 	}); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 	/* check validity at the very end! */ | 
					
						
							|  |  |  | 	if (!GPU_framebuffer_check_valid(ofs->fb, err_out)) { | 
					
						
							|  |  |  | 		GPU_offscreen_free(ofs); | 
					
						
							| 
									
										
										
										
											2019-01-29 07:46:25 +11:00
										 |  |  | 		gpuPopAttr(); | 
					
						
							| 
									
										
										
										
											2018-06-04 09:09:12 +02:00
										 |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	GPU_framebuffer_restore(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-29 07:46:25 +11:00
										 |  |  | 	gpuPopAttr(); | 
					
						
							| 
									
										
										
										
											2018-04-20 20:51:54 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 	return ofs; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GPU_offscreen_free(GPUOffScreen *ofs) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (ofs->fb) | 
					
						
							|  |  |  | 		GPU_framebuffer_free(ofs->fb); | 
					
						
							|  |  |  | 	if (ofs->color) | 
					
						
							|  |  |  | 		GPU_texture_free(ofs->color); | 
					
						
							|  |  |  | 	if (ofs->depth) | 
					
						
							|  |  |  | 		GPU_texture_free(ofs->depth); | 
					
						
							| 
									
										
										
										
											2018-06-04 09:09:12 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 	MEM_freeN(ofs); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GPU_offscreen_bind(GPUOffScreen *ofs, bool save) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	if (save) { | 
					
						
							| 
									
										
										
										
											2019-01-29 07:46:25 +11:00
										 |  |  | 		gpuPushAttr(GPU_SCISSOR_BIT | GPU_VIEWPORT_BIT); | 
					
						
							| 
									
										
										
										
											2018-11-06 15:24:13 +01:00
										 |  |  | 		GPUFrameBuffer *fb = GPU_framebuffer_active_get(); | 
					
						
							|  |  |  | 		gpuPushFrameBuffer(fb); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	glDisable(GL_SCISSOR_TEST); | 
					
						
							|  |  |  | 	GPU_framebuffer_bind(ofs->fb); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | void GPU_offscreen_unbind(GPUOffScreen *UNUSED(ofs), bool restore) | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-11-06 15:24:13 +01:00
										 |  |  | 	GPUFrameBuffer *fb = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	if (restore) { | 
					
						
							| 
									
										
										
										
											2019-01-29 07:46:25 +11:00
										 |  |  | 		gpuPopAttr(); | 
					
						
							| 
									
										
										
										
											2018-11-06 15:24:13 +01:00
										 |  |  | 		fb = gpuPopFrameBuffer(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (fb) { | 
					
						
							|  |  |  | 		GPU_framebuffer_bind(fb); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		GPU_framebuffer_restore(); | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											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) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	const int w = GPU_texture_width(ofs->color); | 
					
						
							|  |  |  | 	const int h = GPU_texture_height(ofs->color); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	glBindFramebuffer(GL_READ_FRAMEBUFFER, ofs->fb->object); | 
					
						
							|  |  |  | 	GLenum status = glCheckFramebufferStatus(GL_READ_FRAMEBUFFER); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (status == GL_FRAMEBUFFER_COMPLETE) { | 
					
						
							|  |  |  | 		glBlitFramebuffer(0, 0, w, h, x, y, x + w, y + h, GL_COLOR_BUFFER_BIT, GL_NEAREST); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		gpu_print_framebuffer_error(status, NULL); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	glBindFramebuffer(GL_READ_FRAMEBUFFER, 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | void GPU_offscreen_read_pixels(GPUOffScreen *ofs, int type, void *pixels) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	const int w = GPU_texture_width(ofs->color); | 
					
						
							|  |  |  | 	const int h = GPU_texture_height(ofs->color); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 	BLI_assert(type == GL_UNSIGNED_BYTE || type == GL_FLOAT); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 	if (GPU_texture_target(ofs->color) == GL_TEXTURE_2D_MULTISAMPLE) { | 
					
						
							|  |  |  | 		/* For a multi-sample texture,
 | 
					
						
							|  |  |  | 		 * we need to create an intermediate buffer to blit to, | 
					
						
							|  |  |  | 		 * before its copied using 'glReadPixels' */ | 
					
						
							|  |  |  | 		GLuint fbo_blit = 0; | 
					
						
							|  |  |  | 		GLuint tex_blit = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* create texture for new 'fbo_blit' */ | 
					
						
							|  |  |  | 		glGenTextures(1, &tex_blit); | 
					
						
							|  |  |  | 		glBindTexture(GL_TEXTURE_2D, tex_blit); | 
					
						
							| 
									
										
										
										
											2018-07-18 23:09:31 +10:00
										 |  |  | 		glTexImage2D( | 
					
						
							|  |  |  | 		        GL_TEXTURE_2D, 0, (type == GL_FLOAT) ? GL_RGBA16F : GL_RGBA8, | 
					
						
							|  |  |  | 		        w, h, 0, GL_RGBA, type, 0); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		/* write into new single-sample buffer */ | 
					
						
							| 
									
										
										
										
											2016-08-04 21:11:45 -04:00
										 |  |  | 		glGenFramebuffers(1, &fbo_blit); | 
					
						
							|  |  |  | 		glBindFramebuffer(GL_DRAW_FRAMEBUFFER, fbo_blit); | 
					
						
							| 
									
										
										
										
											2018-07-18 23:09:31 +10:00
										 |  |  | 		glFramebufferTexture2D( | 
					
						
							|  |  |  | 		        GL_DRAW_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, | 
					
						
							|  |  |  | 		        GL_TEXTURE_2D, tex_blit, 0); | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		GLenum status = glCheckFramebufferStatus(GL_DRAW_FRAMEBUFFER); | 
					
						
							| 
									
										
										
										
											2016-08-04 21:11:45 -04:00
										 |  |  | 		if (status != GL_FRAMEBUFFER_COMPLETE) { | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 			goto finally; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* perform the copy */ | 
					
						
							| 
									
										
										
										
											2016-08-04 21:11:45 -04:00
										 |  |  | 		glBlitFramebuffer(0, 0, w, h, 0, 0, w, h, GL_COLOR_BUFFER_BIT, GL_NEAREST); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		/* read the results */ | 
					
						
							| 
									
										
										
										
											2016-08-04 21:11:45 -04:00
										 |  |  | 		glBindFramebuffer(GL_READ_FRAMEBUFFER, fbo_blit); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 		glReadPixels(0, 0, w, h, GL_RGBA, type, pixels); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* restore the original frame-bufer */ | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 		glBindFramebuffer(GL_FRAMEBUFFER, ofs->fb->object); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | finally: | 
					
						
							|  |  |  | 		/* cleanup */ | 
					
						
							| 
									
										
										
										
											2018-03-25 17:46:48 +02:00
										 |  |  | 		glDeleteTextures(1, &tex_blit); | 
					
						
							|  |  |  | 		glDeleteFramebuffers(1, &fbo_blit); | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		glReadPixels(0, 0, w, h, GL_RGBA, type, pixels); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int GPU_offscreen_width(const GPUOffScreen *ofs) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return GPU_texture_width(ofs->color); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int GPU_offscreen_height(const GPUOffScreen *ofs) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return GPU_texture_height(ofs->color); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-13 18:18:04 +01:00
										 |  |  | GPUTexture *GPU_offscreen_color_texture(const GPUOffScreen *ofs) | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-02-13 18:18:04 +01:00
										 |  |  | 	return ofs->color; | 
					
						
							| 
									
										
										
										
											2015-12-06 21:20:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-03 02:50:29 +10:00
										 |  |  | /* only to be used by viewport code! */ | 
					
						
							|  |  |  | void GPU_offscreen_viewport_data_get( | 
					
						
							|  |  |  |         GPUOffScreen *ofs, | 
					
						
							|  |  |  |         GPUFrameBuffer **r_fb, GPUTexture **r_color, GPUTexture **r_depth) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	*r_fb = ofs->fb; | 
					
						
							|  |  |  | 	*r_color = ofs->color; | 
					
						
							|  |  |  | 	*r_depth = ofs->depth; | 
					
						
							| 
									
										
										
										
											2017-05-09 15:09:39 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-06-26 15:17:31 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | void GPU_clear_color(float red, float green, float blue, float alpha) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	glClearColor(red, green, blue, alpha); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-23 14:15:43 +11:00
										 |  |  | void GPU_clear(eGPUFrameBufferBits flags) | 
					
						
							| 
									
										
										
										
											2018-06-26 15:17:31 -06:00
										 |  |  | { | 
					
						
							|  |  |  | 	glClear(convert_buffer_bits_to_gl(flags)); | 
					
						
							|  |  |  | } |