| 
									
										
										
										
											2018-02-28 01:16:23 +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. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2019-01-23 11:29:18 +11:00
										 |  |  |  * Copyright 2016, Blender Foundation. | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-18 08:08:12 +11:00
										 |  |  | /** \file
 | 
					
						
							|  |  |  |  * \ingroup draw | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "draw_manager.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-05 20:45:32 +02:00
										 |  |  | #include "BLI_math_bits.h"
 | 
					
						
							| 
									
										
										
										
											2019-05-07 18:01:14 +02:00
										 |  |  | #include "BLI_memblock.h"
 | 
					
						
							| 
									
										
										
										
											2018-02-28 16:23:33 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | #include "BKE_global.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "GPU_draw.h"
 | 
					
						
							|  |  |  | #include "GPU_extensions.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-02 18:33:19 +02:00
										 |  |  | #include "intern/gpu_shader_private.h"
 | 
					
						
							| 
									
										
										
										
											2019-05-13 17:56:20 +02:00
										 |  |  | #include "intern/gpu_primitive_private.h"
 | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef USE_GPU_SELECT
 | 
					
						
							|  |  |  | #  include "GPU_select.h"
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef USE_GPU_SELECT
 | 
					
						
							| 
									
										
										
										
											2018-05-11 07:48:52 +02:00
										 |  |  | void DRW_select_load_id(uint id) | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BLI_assert(G.f & G_FLAG_PICKSEL); | 
					
						
							|  |  |  |   DST.select_id = id; | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-02 18:33:19 +02:00
										 |  |  | #define DEBUG_UBO_BINDING
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Draw State (DRW_state)
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void drw_state_set(DRWState state) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (DST.state == state) { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #define CHANGED_TO(f) \
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   ((DST.state_lock & (f)) ? \ | 
					
						
							|  |  |  |        0 : \ | 
					
						
							|  |  |  |        (((DST.state & (f)) ? ((state & (f)) ? 0 : -1) : ((state & (f)) ? 1 : 0)))) | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | #define CHANGED_ANY(f) (((DST.state & (f)) != (state & (f))) && ((DST.state_lock & (f)) == 0))
 | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #define CHANGED_ANY_STORE_VAR(f, enabled) \
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   (((DST.state & (f)) != (enabled = (state & (f)))) && (((DST.state_lock & (f)) == 0))) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Depth Write */ | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     int test; | 
					
						
							|  |  |  |     if ((test = CHANGED_TO(DRW_STATE_WRITE_DEPTH))) { | 
					
						
							|  |  |  |       if (test == 1) { | 
					
						
							|  |  |  |         glDepthMask(GL_TRUE); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         glDepthMask(GL_FALSE); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Color Write */ | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     int test; | 
					
						
							|  |  |  |     if ((test = CHANGED_TO(DRW_STATE_WRITE_COLOR))) { | 
					
						
							|  |  |  |       if (test == 1) { | 
					
						
							|  |  |  |         glColorMask(GL_TRUE, GL_TRUE, GL_TRUE, GL_TRUE); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         glColorMask(GL_FALSE, GL_FALSE, GL_FALSE, GL_FALSE); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Raster Discard */ | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     if (CHANGED_ANY(DRW_STATE_RASTERIZER_ENABLED)) { | 
					
						
							|  |  |  |       if ((state & DRW_STATE_RASTERIZER_ENABLED) != 0) { | 
					
						
							|  |  |  |         glDisable(GL_RASTERIZER_DISCARD); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         glEnable(GL_RASTERIZER_DISCARD); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Cull */ | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     DRWState test; | 
					
						
							|  |  |  |     if (CHANGED_ANY_STORE_VAR(DRW_STATE_CULL_BACK | DRW_STATE_CULL_FRONT, test)) { | 
					
						
							|  |  |  |       if (test) { | 
					
						
							|  |  |  |         glEnable(GL_CULL_FACE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ((state & DRW_STATE_CULL_BACK) != 0) { | 
					
						
							|  |  |  |           glCullFace(GL_BACK); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else if ((state & DRW_STATE_CULL_FRONT) != 0) { | 
					
						
							|  |  |  |           glCullFace(GL_FRONT); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |           BLI_assert(0); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         glDisable(GL_CULL_FACE); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Depth Test */ | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     DRWState test; | 
					
						
							|  |  |  |     if (CHANGED_ANY_STORE_VAR(DRW_STATE_DEPTH_LESS | DRW_STATE_DEPTH_LESS_EQUAL | | 
					
						
							|  |  |  |                                   DRW_STATE_DEPTH_EQUAL | DRW_STATE_DEPTH_GREATER | | 
					
						
							|  |  |  |                                   DRW_STATE_DEPTH_GREATER_EQUAL | DRW_STATE_DEPTH_ALWAYS, | 
					
						
							|  |  |  |                               test)) { | 
					
						
							|  |  |  |       if (test) { | 
					
						
							|  |  |  |         glEnable(GL_DEPTH_TEST); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (state & DRW_STATE_DEPTH_LESS) { | 
					
						
							|  |  |  |           glDepthFunc(GL_LESS); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else if (state & DRW_STATE_DEPTH_LESS_EQUAL) { | 
					
						
							|  |  |  |           glDepthFunc(GL_LEQUAL); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else if (state & DRW_STATE_DEPTH_EQUAL) { | 
					
						
							|  |  |  |           glDepthFunc(GL_EQUAL); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else if (state & DRW_STATE_DEPTH_GREATER) { | 
					
						
							|  |  |  |           glDepthFunc(GL_GREATER); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else if (state & DRW_STATE_DEPTH_GREATER_EQUAL) { | 
					
						
							|  |  |  |           glDepthFunc(GL_GEQUAL); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else if (state & DRW_STATE_DEPTH_ALWAYS) { | 
					
						
							|  |  |  |           glDepthFunc(GL_ALWAYS); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |           BLI_assert(0); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         glDisable(GL_DEPTH_TEST); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Wire Width */ | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     int test; | 
					
						
							| 
									
										
										
										
											2019-05-17 14:04:30 +02:00
										 |  |  |     if ((test = CHANGED_TO(DRW_STATE_WIRE_SMOOTH))) { | 
					
						
							|  |  |  |       if (test == 1) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         GPU_line_width(2.0f); | 
					
						
							|  |  |  |         GPU_line_smooth(true); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         GPU_line_width(1.0f); | 
					
						
							|  |  |  |         GPU_line_smooth(false); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Blending (all buffer) */ | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     int test; | 
					
						
							|  |  |  |     if (CHANGED_ANY_STORE_VAR(DRW_STATE_BLEND | DRW_STATE_BLEND_PREMUL | DRW_STATE_ADDITIVE | | 
					
						
							|  |  |  |                                   DRW_STATE_MULTIPLY | DRW_STATE_ADDITIVE_FULL | | 
					
						
							| 
									
										
										
										
											2019-05-01 10:48:13 +02:00
										 |  |  |                                   DRW_STATE_BLEND_OIT | DRW_STATE_BLEND_PREMUL_UNDER, | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |                               test)) { | 
					
						
							|  |  |  |       if (test) { | 
					
						
							|  |  |  |         glEnable(GL_BLEND); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ((state & DRW_STATE_BLEND) != 0) { | 
					
						
							|  |  |  |           glBlendFuncSeparate(GL_SRC_ALPHA, | 
					
						
							|  |  |  |                               GL_ONE_MINUS_SRC_ALPHA, /* RGB */ | 
					
						
							|  |  |  |                               GL_ONE, | 
					
						
							|  |  |  |                               GL_ONE_MINUS_SRC_ALPHA); /* Alpha */ | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-05-01 10:48:13 +02:00
										 |  |  |         else if ((state & DRW_STATE_BLEND_PREMUL_UNDER) != 0) { | 
					
						
							|  |  |  |           glBlendFunc(GL_ONE_MINUS_DST_ALPHA, GL_ONE); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         else if ((state & DRW_STATE_BLEND_PREMUL) != 0) { | 
					
						
							|  |  |  |           glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else if ((state & DRW_STATE_MULTIPLY) != 0) { | 
					
						
							|  |  |  |           glBlendFunc(GL_DST_COLOR, GL_ZERO); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else if ((state & DRW_STATE_BLEND_OIT) != 0) { | 
					
						
							|  |  |  |           glBlendFuncSeparate(GL_ONE, | 
					
						
							|  |  |  |                               GL_ONE, /* RGB */ | 
					
						
							|  |  |  |                               GL_ZERO, | 
					
						
							|  |  |  |                               GL_ONE_MINUS_SRC_ALPHA); /* Alpha */ | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else if ((state & DRW_STATE_ADDITIVE) != 0) { | 
					
						
							|  |  |  |           /* Do not let alpha accumulate but premult the source RGB by it. */ | 
					
						
							|  |  |  |           glBlendFuncSeparate(GL_SRC_ALPHA, | 
					
						
							|  |  |  |                               GL_ONE, /* RGB */ | 
					
						
							|  |  |  |                               GL_ZERO, | 
					
						
							|  |  |  |                               GL_ONE); /* Alpha */ | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else if ((state & DRW_STATE_ADDITIVE_FULL) != 0) { | 
					
						
							|  |  |  |           /* Let alpha accumulate. */ | 
					
						
							|  |  |  |           glBlendFunc(GL_ONE, GL_ONE); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |           BLI_assert(0); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         glDisable(GL_BLEND); | 
					
						
							|  |  |  |         glBlendFunc(GL_ONE, GL_ONE); /* Don't multiply incoming color by alpha. */ | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Clip Planes */ | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     int test; | 
					
						
							|  |  |  |     if ((test = CHANGED_TO(DRW_STATE_CLIP_PLANES))) { | 
					
						
							|  |  |  |       if (test == 1) { | 
					
						
							|  |  |  |         for (int i = 0; i < DST.clip_planes_len; ++i) { | 
					
						
							|  |  |  |           glEnable(GL_CLIP_DISTANCE0 + i); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         for (int i = 0; i < MAX_CLIP_PLANES; ++i) { | 
					
						
							|  |  |  |           glDisable(GL_CLIP_DISTANCE0 + i); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Stencil */ | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     DRWState test; | 
					
						
							|  |  |  |     if (CHANGED_ANY_STORE_VAR(DRW_STATE_WRITE_STENCIL | DRW_STATE_WRITE_STENCIL_SHADOW_PASS | | 
					
						
							|  |  |  |                                   DRW_STATE_WRITE_STENCIL_SHADOW_FAIL | DRW_STATE_STENCIL_EQUAL | | 
					
						
							|  |  |  |                                   DRW_STATE_STENCIL_NEQUAL, | 
					
						
							|  |  |  |                               test)) { | 
					
						
							|  |  |  |       if (test) { | 
					
						
							|  |  |  |         glEnable(GL_STENCIL_TEST); | 
					
						
							|  |  |  |         /* Stencil Write */ | 
					
						
							|  |  |  |         if ((state & DRW_STATE_WRITE_STENCIL) != 0) { | 
					
						
							|  |  |  |           glStencilMask(0xFF); | 
					
						
							|  |  |  |           glStencilOp(GL_KEEP, GL_KEEP, GL_REPLACE); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else if ((state & DRW_STATE_WRITE_STENCIL_SHADOW_PASS) != 0) { | 
					
						
							|  |  |  |           glStencilMask(0xFF); | 
					
						
							|  |  |  |           glStencilOpSeparate(GL_BACK, GL_KEEP, GL_KEEP, GL_INCR_WRAP); | 
					
						
							|  |  |  |           glStencilOpSeparate(GL_FRONT, GL_KEEP, GL_KEEP, GL_DECR_WRAP); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else if ((state & DRW_STATE_WRITE_STENCIL_SHADOW_FAIL) != 0) { | 
					
						
							|  |  |  |           glStencilMask(0xFF); | 
					
						
							|  |  |  |           glStencilOpSeparate(GL_BACK, GL_KEEP, GL_DECR_WRAP, GL_KEEP); | 
					
						
							|  |  |  |           glStencilOpSeparate(GL_FRONT, GL_KEEP, GL_INCR_WRAP, GL_KEEP); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         /* Stencil Test */ | 
					
						
							|  |  |  |         else if ((state & (DRW_STATE_STENCIL_EQUAL | DRW_STATE_STENCIL_NEQUAL)) != 0) { | 
					
						
							|  |  |  |           glStencilMask(0x00); /* disable write */ | 
					
						
							|  |  |  |           DST.stencil_mask = STENCIL_UNDEFINED; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |           BLI_assert(0); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         /* disable write & test */ | 
					
						
							|  |  |  |         DST.stencil_mask = 0; | 
					
						
							|  |  |  |         glStencilMask(0x00); | 
					
						
							|  |  |  |         glStencilFunc(GL_ALWAYS, 0, 0xFF); | 
					
						
							|  |  |  |         glDisable(GL_STENCIL_TEST); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Provoking Vertex */ | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     int test; | 
					
						
							|  |  |  |     if ((test = CHANGED_TO(DRW_STATE_FIRST_VERTEX_CONVENTION))) { | 
					
						
							|  |  |  |       if (test == 1) { | 
					
						
							|  |  |  |         glProvokingVertex(GL_FIRST_VERTEX_CONVENTION); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         glProvokingVertex(GL_LAST_VERTEX_CONVENTION); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Polygon Offset */ | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     int test; | 
					
						
							|  |  |  |     if (CHANGED_ANY_STORE_VAR(DRW_STATE_OFFSET_POSITIVE | DRW_STATE_OFFSET_NEGATIVE, test)) { | 
					
						
							|  |  |  |       if (test) { | 
					
						
							|  |  |  |         glEnable(GL_POLYGON_OFFSET_FILL); | 
					
						
							|  |  |  |         glEnable(GL_POLYGON_OFFSET_LINE); | 
					
						
							|  |  |  |         glEnable(GL_POLYGON_OFFSET_POINT); | 
					
						
							|  |  |  |         /* Stencil Write */ | 
					
						
							|  |  |  |         if ((state & DRW_STATE_OFFSET_POSITIVE) != 0) { | 
					
						
							|  |  |  |           glPolygonOffset(1.0f, 1.0f); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else if ((state & DRW_STATE_OFFSET_NEGATIVE) != 0) { | 
					
						
							|  |  |  |           glPolygonOffset(-1.0f, -1.0f); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |           BLI_assert(0); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         glDisable(GL_POLYGON_OFFSET_FILL); | 
					
						
							|  |  |  |         glDisable(GL_POLYGON_OFFSET_LINE); | 
					
						
							|  |  |  |         glDisable(GL_POLYGON_OFFSET_POINT); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-12-11 18:18:36 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | #undef CHANGED_TO
 | 
					
						
							|  |  |  | #undef CHANGED_ANY
 | 
					
						
							|  |  |  | #undef CHANGED_ANY_STORE_VAR
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   DST.state = state; | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-11 07:48:52 +02:00
										 |  |  | static void drw_stencil_set(uint mask) | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (DST.stencil_mask != mask) { | 
					
						
							|  |  |  |     DST.stencil_mask = mask; | 
					
						
							|  |  |  |     /* Stencil Write */ | 
					
						
							|  |  |  |     if ((DST.state & DRW_STATE_WRITE_STENCIL) != 0) { | 
					
						
							|  |  |  |       glStencilFunc(GL_ALWAYS, mask, 0xFF); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     /* Stencil Test */ | 
					
						
							|  |  |  |     else if ((DST.state & DRW_STATE_STENCIL_EQUAL) != 0) { | 
					
						
							|  |  |  |       glStencilFunc(GL_EQUAL, mask, 0xFF); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if ((DST.state & DRW_STATE_STENCIL_NEQUAL) != 0) { | 
					
						
							|  |  |  |       glStencilFunc(GL_NOTEQUAL, mask, 0xFF); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Reset state to not interfer with other UI drawcall */ | 
					
						
							|  |  |  | void DRW_state_reset_ex(DRWState state) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   DST.state = ~state; | 
					
						
							|  |  |  |   drw_state_set(state); | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-01 17:14:35 +11:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Use with care, intended so selection code can override passes depth settings, | 
					
						
							|  |  |  |  * which is important for selection to work properly. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Should be set in main draw loop, cleared afterwards | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void DRW_state_lock(DRWState state) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   DST.state_lock = state; | 
					
						
							| 
									
										
										
										
											2018-03-01 17:14:35 +11:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | void DRW_state_reset(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   DRW_state_reset_ex(DRW_STATE_DEFAULT); | 
					
						
							| 
									
										
										
										
											2018-05-07 18:29:37 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-17 14:04:30 +02:00
										 |  |  |   GPU_point_size(5); | 
					
						
							|  |  |  |   GPU_enable_program_point_size(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* Reset blending function */ | 
					
						
							|  |  |  |   glBlendFuncSeparate(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA, GL_ONE, GL_ONE_MINUS_SRC_ALPHA); | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * This only works if DRWPasses have been tagged with DRW_STATE_CLIP_PLANES, | 
					
						
							|  |  |  |  * and if the shaders have support for it (see usage of gl_ClipDistance). | 
					
						
							|  |  |  |  * Be sure to call DRW_state_clip_planes_reset() after you finish drawing. | 
					
						
							| 
									
										
										
										
											2019-03-19 15:17:46 +11:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2019-01-21 11:24:19 +11:00
										 |  |  | void DRW_state_clip_planes_len_set(uint plane_len) | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BLI_assert(plane_len <= MAX_CLIP_PLANES); | 
					
						
							|  |  |  |   DST.clip_planes_len = plane_len; | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DRW_state_clip_planes_reset(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   DST.clip_planes_len = 0; | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-21 23:51:34 +11:00
										 |  |  | void DRW_state_clip_planes_set_from_rv3d(RegionView3D *rv3d) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   int max_len = 6; | 
					
						
							|  |  |  |   int real_len = (rv3d->viewlock & RV3D_BOXCLIP) ? 4 : max_len; | 
					
						
							|  |  |  |   while (real_len < max_len) { | 
					
						
							|  |  |  |     /* Fill in dummy values that wont change results (6 is hard coded in shaders). */ | 
					
						
							|  |  |  |     copy_v4_v4(rv3d->clip[real_len], rv3d->clip[3]); | 
					
						
							|  |  |  |     real_len++; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   DRW_state_clip_planes_len_set(max_len); | 
					
						
							| 
									
										
										
										
											2019-01-21 23:51:34 +11:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							| 
									
										
										
										
											2018-03-01 03:52:54 +01:00
										 |  |  | /** \name Clipping (DRW_clipping)
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-16 10:40:46 -03:00
										 |  |  | /* Extract the 8 corners from a Projection Matrix.
 | 
					
						
							| 
									
										
										
										
											2018-04-02 12:21:09 -03:00
										 |  |  |  * Although less accurate, this solution can be simplified as follows: | 
					
						
							| 
									
										
										
										
											2019-05-01 10:35:46 +10:00
										 |  |  |  * BKE_boundbox_init_from_minmax(&bbox, (const float[3]){-1.0f, -1.0f, -1.0f}, (const | 
					
						
							|  |  |  |  * float[3]){1.0f, 1.0f, 1.0f}); for (int i = 0; i < 8; i++) {mul_project_m4_v3(projinv, | 
					
						
							|  |  |  |  * bbox.vec[i]);} | 
					
						
							| 
									
										
										
										
											2018-04-02 12:21:09 -03:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | static void draw_frustum_boundbox_calc(const float (*projmat)[4], BoundBox *r_bbox) | 
					
						
							| 
									
										
										
										
											2018-04-02 12:21:09 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   float left, right, bottom, top, near, far; | 
					
						
							|  |  |  |   bool is_persp = projmat[3][3] == 0.0f; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   projmat_dimensions(projmat, &left, &right, &bottom, &top, &near, &far); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (is_persp) { | 
					
						
							|  |  |  |     left *= near; | 
					
						
							|  |  |  |     right *= near; | 
					
						
							|  |  |  |     bottom *= near; | 
					
						
							|  |  |  |     top *= near; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   r_bbox->vec[0][2] = r_bbox->vec[3][2] = r_bbox->vec[7][2] = r_bbox->vec[4][2] = -near; | 
					
						
							|  |  |  |   r_bbox->vec[0][0] = r_bbox->vec[3][0] = left; | 
					
						
							|  |  |  |   r_bbox->vec[4][0] = r_bbox->vec[7][0] = right; | 
					
						
							|  |  |  |   r_bbox->vec[0][1] = r_bbox->vec[4][1] = bottom; | 
					
						
							|  |  |  |   r_bbox->vec[7][1] = r_bbox->vec[3][1] = top; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Get the coordinates of the far plane. */ | 
					
						
							|  |  |  |   if (is_persp) { | 
					
						
							|  |  |  |     float sca_far = far / near; | 
					
						
							|  |  |  |     left *= sca_far; | 
					
						
							|  |  |  |     right *= sca_far; | 
					
						
							|  |  |  |     bottom *= sca_far; | 
					
						
							|  |  |  |     top *= sca_far; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   r_bbox->vec[1][2] = r_bbox->vec[2][2] = r_bbox->vec[6][2] = r_bbox->vec[5][2] = -far; | 
					
						
							|  |  |  |   r_bbox->vec[1][0] = r_bbox->vec[2][0] = left; | 
					
						
							|  |  |  |   r_bbox->vec[6][0] = r_bbox->vec[5][0] = right; | 
					
						
							|  |  |  |   r_bbox->vec[1][1] = r_bbox->vec[5][1] = bottom; | 
					
						
							|  |  |  |   r_bbox->vec[2][1] = r_bbox->vec[6][1] = top; | 
					
						
							| 
									
										
										
										
											2018-04-02 12:21:09 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-01 03:52:54 +01:00
										 |  |  | static void draw_clipping_setup_from_view(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (DST.clipping.updated) { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-03-01 03:52:54 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   float(*viewinv)[4] = DST.view_data.matstate.mat[DRW_MAT_VIEWINV]; | 
					
						
							|  |  |  |   float(*projmat)[4] = DST.view_data.matstate.mat[DRW_MAT_WIN]; | 
					
						
							|  |  |  |   float(*projinv)[4] = DST.view_data.matstate.mat[DRW_MAT_WININV]; | 
					
						
							|  |  |  |   BoundSphere *bsphere = &DST.clipping.frustum_bsphere; | 
					
						
							| 
									
										
										
										
											2018-03-01 03:52:54 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* Extract Clipping Planes */ | 
					
						
							|  |  |  |   BoundBox bbox; | 
					
						
							| 
									
										
										
										
											2018-04-16 09:22:42 -03:00
										 |  |  | #if 0 /* It has accuracy problems. */
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:24:14 +02:00
										 |  |  |   BKE_boundbox_init_from_minmax( | 
					
						
							|  |  |  |       &bbox, (const float[3]){-1.0f, -1.0f, -1.0f}, (const float[3]){1.0f, 1.0f, 1.0f}); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   for (int i = 0; i < 8; i++) { | 
					
						
							|  |  |  |     mul_project_m4_v3(projinv, bbox.vec[i]); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-04-16 09:22:42 -03:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   draw_frustum_boundbox_calc(projmat, &bbox); | 
					
						
							| 
									
										
										
										
											2018-04-15 21:48:57 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* Transform into world space. */ | 
					
						
							|  |  |  |   for (int i = 0; i < 8; i++) { | 
					
						
							|  |  |  |     mul_m4_v3(viewinv, bbox.vec[i]); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   memcpy(&DST.clipping.frustum_corners, &bbox, sizeof(BoundBox)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Compute clip planes using the world space frustum corners. */ | 
					
						
							|  |  |  |   for (int p = 0; p < 6; p++) { | 
					
						
							|  |  |  |     int q, r, s; | 
					
						
							|  |  |  |     switch (p) { | 
					
						
							|  |  |  |       case 0: | 
					
						
							|  |  |  |         q = 1; | 
					
						
							|  |  |  |         r = 2; | 
					
						
							|  |  |  |         s = 3; | 
					
						
							|  |  |  |         break; /* -X */ | 
					
						
							|  |  |  |       case 1: | 
					
						
							|  |  |  |         q = 0; | 
					
						
							|  |  |  |         r = 4; | 
					
						
							|  |  |  |         s = 5; | 
					
						
							|  |  |  |         break; /* -Y */ | 
					
						
							|  |  |  |       case 2: | 
					
						
							|  |  |  |         q = 1; | 
					
						
							|  |  |  |         r = 5; | 
					
						
							|  |  |  |         s = 6; | 
					
						
							|  |  |  |         break; /* +Z (far) */ | 
					
						
							|  |  |  |       case 3: | 
					
						
							|  |  |  |         q = 2; | 
					
						
							|  |  |  |         r = 6; | 
					
						
							|  |  |  |         s = 7; | 
					
						
							|  |  |  |         break; /* +Y */ | 
					
						
							|  |  |  |       case 4: | 
					
						
							|  |  |  |         q = 0; | 
					
						
							|  |  |  |         r = 3; | 
					
						
							|  |  |  |         s = 7; | 
					
						
							|  |  |  |         break; /* -Z (near) */ | 
					
						
							|  |  |  |       default: | 
					
						
							|  |  |  |         q = 4; | 
					
						
							|  |  |  |         r = 7; | 
					
						
							|  |  |  |         s = 6; | 
					
						
							|  |  |  |         break; /* +X */ | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (DST.frontface == GL_CW) { | 
					
						
							|  |  |  |       SWAP(int, q, s); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     normal_quad_v3( | 
					
						
							|  |  |  |         DST.clipping.frustum_planes[p], bbox.vec[p], bbox.vec[q], bbox.vec[r], bbox.vec[s]); | 
					
						
							|  |  |  |     /* Increase precision and use the mean of all 4 corners. */ | 
					
						
							|  |  |  |     DST.clipping.frustum_planes[p][3] = -dot_v3v3(DST.clipping.frustum_planes[p], bbox.vec[p]); | 
					
						
							|  |  |  |     DST.clipping.frustum_planes[p][3] += -dot_v3v3(DST.clipping.frustum_planes[p], bbox.vec[q]); | 
					
						
							|  |  |  |     DST.clipping.frustum_planes[p][3] += -dot_v3v3(DST.clipping.frustum_planes[p], bbox.vec[r]); | 
					
						
							|  |  |  |     DST.clipping.frustum_planes[p][3] += -dot_v3v3(DST.clipping.frustum_planes[p], bbox.vec[s]); | 
					
						
							|  |  |  |     DST.clipping.frustum_planes[p][3] *= 0.25f; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Extract Bounding Sphere */ | 
					
						
							|  |  |  |   if (projmat[3][3] != 0.0f) { | 
					
						
							|  |  |  |     /* Orthographic */ | 
					
						
							|  |  |  |     /* The most extreme points on the near and far plane. (normalized device coords). */ | 
					
						
							|  |  |  |     float *nearpoint = bbox.vec[0]; | 
					
						
							|  |  |  |     float *farpoint = bbox.vec[6]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* just use median point */ | 
					
						
							|  |  |  |     mid_v3_v3v3(bsphere->center, farpoint, nearpoint); | 
					
						
							|  |  |  |     bsphere->radius = len_v3v3(bsphere->center, farpoint); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else if (projmat[2][0] == 0.0f && projmat[2][1] == 0.0f) { | 
					
						
							|  |  |  |     /* Perspective with symmetrical frustum. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* We obtain the center and radius of the circumscribed circle of the
 | 
					
						
							|  |  |  |      * isosceles trapezoid composed by the diagonals of the near and far clipping plane */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* center of each clipping plane */ | 
					
						
							|  |  |  |     float mid_min[3], mid_max[3]; | 
					
						
							|  |  |  |     mid_v3_v3v3(mid_min, bbox.vec[3], bbox.vec[4]); | 
					
						
							|  |  |  |     mid_v3_v3v3(mid_max, bbox.vec[2], bbox.vec[5]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* square length of the diagonals of each clipping plane */ | 
					
						
							|  |  |  |     float a_sq = len_squared_v3v3(bbox.vec[3], bbox.vec[4]); | 
					
						
							|  |  |  |     float b_sq = len_squared_v3v3(bbox.vec[2], bbox.vec[5]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* distance squared between clipping planes */ | 
					
						
							|  |  |  |     float h_sq = len_squared_v3v3(mid_min, mid_max); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     float fac = (4 * h_sq + b_sq - a_sq) / (8 * h_sq); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* The goal is to get the smallest sphere,
 | 
					
						
							|  |  |  |      * not the sphere that passes through each corner */ | 
					
						
							|  |  |  |     CLAMP(fac, 0.0f, 1.0f); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     interp_v3_v3v3(bsphere->center, mid_min, mid_max, fac); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* distance from the center to one of the points of the far plane (1, 2, 5, 6) */ | 
					
						
							|  |  |  |     bsphere->radius = len_v3v3(bsphere->center, bbox.vec[1]); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     /* Perspective with asymmetrical frustum. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* We put the sphere center on the line that goes from origin
 | 
					
						
							|  |  |  |      * to the center of the far clipping plane. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Detect which of the corner of the far clipping plane is the farthest to the origin */ | 
					
						
							|  |  |  |     float nfar[4];               /* most extreme far point in NDC space */ | 
					
						
							|  |  |  |     float farxy[2];              /* farpoint projection onto the near plane */ | 
					
						
							|  |  |  |     float farpoint[3] = {0.0f};  /* most extreme far point in camera coordinate */ | 
					
						
							|  |  |  |     float nearpoint[3];          /* most extreme near point in camera coordinate */ | 
					
						
							|  |  |  |     float farcenter[3] = {0.0f}; /* center of far cliping plane in camera coordinate */ | 
					
						
							|  |  |  |     float F = -1.0f, N;          /* square distance of far and near point to origin */ | 
					
						
							|  |  |  |     float f, n; /* distance of far and near point to z axis. f is always > 0 but n can be < 0 */ | 
					
						
							|  |  |  |     float e, s; /* far and near clipping distance (<0) */ | 
					
						
							| 
									
										
										
										
											2019-04-21 13:44:06 +10:00
										 |  |  |     float c;    /* slope of center line = distance of far clipping center
 | 
					
						
							|  |  |  |                  * to z axis / far clipping distance. */ | 
					
						
							|  |  |  |     float z;    /* projection of sphere center on z axis (<0) */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* Find farthest corner and center of far clip plane. */ | 
					
						
							|  |  |  |     float corner[3] = {1.0f, 1.0f, 1.0f}; /* in clip space */ | 
					
						
							|  |  |  |     for (int i = 0; i < 4; i++) { | 
					
						
							|  |  |  |       float point[3]; | 
					
						
							|  |  |  |       mul_v3_project_m4_v3(point, projinv, corner); | 
					
						
							|  |  |  |       float len = len_squared_v3(point); | 
					
						
							|  |  |  |       if (len > F) { | 
					
						
							|  |  |  |         copy_v3_v3(nfar, corner); | 
					
						
							|  |  |  |         copy_v3_v3(farpoint, point); | 
					
						
							|  |  |  |         F = len; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       add_v3_v3(farcenter, point); | 
					
						
							|  |  |  |       /* rotate by 90 degree to walk through the 4 points of the far clip plane */ | 
					
						
							|  |  |  |       float tmp = corner[0]; | 
					
						
							|  |  |  |       corner[0] = -corner[1]; | 
					
						
							|  |  |  |       corner[1] = tmp; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* the far center is the average of the far clipping points */ | 
					
						
							|  |  |  |     mul_v3_fl(farcenter, 0.25f); | 
					
						
							|  |  |  |     /* the extreme near point is the opposite point on the near clipping plane */ | 
					
						
							|  |  |  |     copy_v3_fl3(nfar, -nfar[0], -nfar[1], -1.0f); | 
					
						
							|  |  |  |     mul_v3_project_m4_v3(nearpoint, projinv, nfar); | 
					
						
							|  |  |  |     /* this is a frustum projection */ | 
					
						
							|  |  |  |     N = len_squared_v3(nearpoint); | 
					
						
							|  |  |  |     e = farpoint[2]; | 
					
						
							|  |  |  |     s = nearpoint[2]; | 
					
						
							|  |  |  |     /* distance to view Z axis */ | 
					
						
							|  |  |  |     f = len_v2(farpoint); | 
					
						
							|  |  |  |     /* get corresponding point on the near plane */ | 
					
						
							|  |  |  |     mul_v2_v2fl(farxy, farpoint, s / e); | 
					
						
							|  |  |  |     /* this formula preserve the sign of n */ | 
					
						
							|  |  |  |     sub_v2_v2(nearpoint, farxy); | 
					
						
							|  |  |  |     n = f * s / e - len_v2(nearpoint); | 
					
						
							|  |  |  |     c = len_v2(farcenter) / e; | 
					
						
							|  |  |  |     /* the big formula, it simplifies to (F-N)/(2(e-s)) for the symmetric case */ | 
					
						
							|  |  |  |     z = (F - N) / (2.0f * (e - s + c * (f - n))); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bsphere->center[0] = farcenter[0] * z / e; | 
					
						
							|  |  |  |     bsphere->center[1] = farcenter[1] * z / e; | 
					
						
							|  |  |  |     bsphere->center[2] = z; | 
					
						
							|  |  |  |     bsphere->radius = len_v3v3(bsphere->center, farpoint); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Transform to world space. */ | 
					
						
							|  |  |  |     mul_m4_v3(viewinv, bsphere->center); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   DST.clipping.updated = true; | 
					
						
							| 
									
										
										
										
											2018-03-01 03:52:54 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Return True if the given BoundSphere intersect the current view frustum */ | 
					
						
							| 
									
										
										
										
											2018-03-08 17:54:14 +01:00
										 |  |  | bool DRW_culling_sphere_test(BoundSphere *bsphere) | 
					
						
							| 
									
										
										
										
											2018-03-01 03:52:54 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   draw_clipping_setup_from_view(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Bypass test if radius is negative. */ | 
					
						
							|  |  |  |   if (bsphere->radius < 0.0f) { | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Do a rough test first: Sphere VS Sphere intersect. */ | 
					
						
							|  |  |  |   BoundSphere *frustum_bsphere = &DST.clipping.frustum_bsphere; | 
					
						
							| 
									
										
										
										
											2019-05-04 01:41:39 +02:00
										 |  |  |   float center_dist_sq = len_squared_v3v3(bsphere->center, frustum_bsphere->center); | 
					
						
							|  |  |  |   float radius_sum = bsphere->radius + frustum_bsphere->radius; | 
					
						
							|  |  |  |   if (center_dist_sq > SQUARE(radius_sum)) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-05-04 01:41:39 +02:00
										 |  |  |   /* TODO we could test against the inscribed sphere of the frustum to early out positively. */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Test against the 6 frustum planes. */ | 
					
						
							| 
									
										
										
										
											2019-05-17 14:04:30 +02:00
										 |  |  |   /* TODO order planes with sides first then far then near clip. Should be better culling
 | 
					
						
							|  |  |  |    * heuristic when sculpting. */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   for (int p = 0; p < 6; p++) { | 
					
						
							|  |  |  |     float dist = plane_point_side_v3(DST.clipping.frustum_planes[p], bsphere->center); | 
					
						
							|  |  |  |     if (dist < -bsphere->radius) { | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2018-03-01 03:52:54 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-08 17:54:14 +01:00
										 |  |  | /* Return True if the given BoundBox intersect the current view frustum.
 | 
					
						
							|  |  |  |  * bbox must be in world space. */ | 
					
						
							|  |  |  | bool DRW_culling_box_test(BoundBox *bbox) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   draw_clipping_setup_from_view(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* 6 view frustum planes */ | 
					
						
							|  |  |  |   for (int p = 0; p < 6; p++) { | 
					
						
							|  |  |  |     /* 8 box vertices. */ | 
					
						
							|  |  |  |     for (int v = 0; v < 8; v++) { | 
					
						
							|  |  |  |       float dist = plane_point_side_v3(DST.clipping.frustum_planes[p], bbox->vec[v]); | 
					
						
							|  |  |  |       if (dist > 0.0f) { | 
					
						
							|  |  |  |         /* At least one point in front of this plane.
 | 
					
						
							|  |  |  |          * Go to next plane. */ | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else if (v == 7) { | 
					
						
							|  |  |  |         /* 8 points behind this plane. */ | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2018-03-08 17:54:14 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-03 15:43:42 +02:00
										 |  |  | /* Return True if the current view frustum is inside or intersect the given plane */ | 
					
						
							|  |  |  | bool DRW_culling_plane_test(float plane[4]) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   draw_clipping_setup_from_view(); | 
					
						
							| 
									
										
										
										
											2018-05-03 15:43:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* Test against the 8 frustum corners. */ | 
					
						
							|  |  |  |   for (int c = 0; c < 8; c++) { | 
					
						
							|  |  |  |     float dist = plane_point_side_v3(plane, DST.clipping.frustum_corners.vec[c]); | 
					
						
							|  |  |  |     if (dist < 0.0f) { | 
					
						
							|  |  |  |       return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-05-03 15:43:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return false; | 
					
						
							| 
									
										
										
										
											2018-05-03 15:43:42 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-26 22:28:35 +02:00
										 |  |  | void DRW_culling_frustum_corners_get(BoundBox *corners) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   draw_clipping_setup_from_view(); | 
					
						
							|  |  |  |   memcpy(corners, &DST.clipping.frustum_corners, sizeof(BoundBox)); | 
					
						
							| 
									
										
										
										
											2018-05-26 22:28:35 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-04 17:33:25 +02:00
										 |  |  | /* See draw_clipping_setup_from_view() for the plane order. */ | 
					
						
							|  |  |  | void DRW_culling_frustum_planes_get(float planes[6][4]) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   draw_clipping_setup_from_view(); | 
					
						
							|  |  |  |   memcpy(planes, &DST.clipping.frustum_planes, sizeof(DST.clipping.frustum_planes)); | 
					
						
							| 
									
										
										
										
											2018-06-04 17:33:25 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-01 03:52:54 +01:00
										 |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | /** \name Draw (DRW_draw)
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-24 12:29:15 +02:00
										 |  |  | static void draw_visibility_eval(DRWCallState *st) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   bool culled = st->flag & DRW_CALL_CULLED; | 
					
						
							| 
									
										
										
										
											2018-04-24 12:29:15 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (st->cache_id != DST.state_cache_id) { | 
					
						
							|  |  |  |     /* Update culling result for this view. */ | 
					
						
							|  |  |  |     culled = !DRW_culling_sphere_test(&st->bsphere); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-04-24 12:29:15 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (st->visibility_cb) { | 
					
						
							|  |  |  |     culled = !st->visibility_cb(!culled, st->user_data); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-04-24 12:29:15 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   SET_FLAG_FROM_TEST(st->flag, culled, DRW_CALL_CULLED); | 
					
						
							| 
									
										
										
										
											2018-04-24 12:29:15 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | static void draw_matrices_model_prepare(DRWCallState *st) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (st->cache_id == DST.state_cache_id) { | 
					
						
							|  |  |  |     /* Values are already updated for this view. */ | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     st->cache_id = DST.state_cache_id; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* No need to go further the call will not be used. */ | 
					
						
							|  |  |  |   if ((st->flag & DRW_CALL_CULLED) != 0 && (st->flag & DRW_CALL_BYPASS_CULLING) == 0) { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-05-11 14:04:18 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (st->matflag & DRW_CALL_MODELVIEWPROJECTION) { | 
					
						
							|  |  |  |     mul_m4_m4m4(st->modelviewprojection, DST.view_data.matstate.mat[DRW_MAT_PERS], st->model); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-09 17:32:13 +02:00
										 |  |  | static void draw_geometry_prepare(DRWShadingGroup *shgroup, DRWCall *call) | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* step 1 : bind object dependent matrices */ | 
					
						
							|  |  |  |   if (call != NULL) { | 
					
						
							|  |  |  |     DRWCallState *state = call->state; | 
					
						
							| 
									
										
										
										
											2019-05-07 17:21:26 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (shgroup->model != -1) { | 
					
						
							|  |  |  |       GPU_shader_uniform_vector(shgroup->shader, shgroup->model, 16, 1, (float *)state->model); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (shgroup->modelinverse != -1) { | 
					
						
							|  |  |  |       GPU_shader_uniform_vector( | 
					
						
							|  |  |  |           shgroup->shader, shgroup->modelinverse, 16, 1, (float *)state->modelinverse); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (shgroup->modelviewprojection != -1) { | 
					
						
							|  |  |  |       GPU_shader_uniform_vector(shgroup->shader, | 
					
						
							|  |  |  |                                 shgroup->modelviewprojection, | 
					
						
							|  |  |  |                                 16, | 
					
						
							|  |  |  |                                 1, | 
					
						
							|  |  |  |                                 (float *)state->modelviewprojection); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (shgroup->objectinfo != -1) { | 
					
						
							| 
									
										
										
										
											2019-05-11 14:42:26 +02:00
										 |  |  |       float infos[4]; | 
					
						
							|  |  |  |       infos[0] = state->ob_index; | 
					
						
							|  |  |  |       // infos[1]; /* UNUSED. */
 | 
					
						
							|  |  |  |       infos[2] = state->ob_random; | 
					
						
							|  |  |  |       infos[3] = (state->flag & DRW_CALL_NEGSCALE) ? -1.0f : 1.0f; | 
					
						
							|  |  |  |       GPU_shader_uniform_vector(shgroup->shader, shgroup->objectinfo, 4, 1, (float *)infos); | 
					
						
							| 
									
										
										
										
											2019-05-07 17:21:26 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (shgroup->orcotexfac != -1) { | 
					
						
							|  |  |  |       GPU_shader_uniform_vector( | 
					
						
							|  |  |  |           shgroup->shader, shgroup->orcotexfac, 3, 2, (float *)state->orcotexfac); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     /* For instancing and batching. */ | 
					
						
							|  |  |  |     float unitmat[4][4]; | 
					
						
							|  |  |  |     unit_m4(unitmat); | 
					
						
							| 
									
										
										
										
											2019-05-07 17:21:26 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (shgroup->model != -1) { | 
					
						
							|  |  |  |       GPU_shader_uniform_vector(shgroup->shader, shgroup->model, 16, 1, (float *)unitmat); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (shgroup->modelinverse != -1) { | 
					
						
							|  |  |  |       GPU_shader_uniform_vector(shgroup->shader, shgroup->modelinverse, 16, 1, (float *)unitmat); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (shgroup->modelviewprojection != -1) { | 
					
						
							|  |  |  |       GPU_shader_uniform_vector(shgroup->shader, | 
					
						
							|  |  |  |                                 shgroup->modelviewprojection, | 
					
						
							|  |  |  |                                 16, | 
					
						
							|  |  |  |                                 1, | 
					
						
							|  |  |  |                                 (float *)DST.view_data.matstate.mat[DRW_MAT_PERS]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (shgroup->objectinfo != -1) { | 
					
						
							|  |  |  |       GPU_shader_uniform_vector(shgroup->shader, shgroup->objectinfo, 4, 1, (float *)unitmat); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (shgroup->orcotexfac != -1) { | 
					
						
							| 
									
										
										
										
											2019-05-13 17:56:20 +02:00
										 |  |  |       float orcofacs[2][3] = {{0.0f, 0.0f, 0.0f}, {1.0f, 1.0f, 1.0f}}; | 
					
						
							|  |  |  |       GPU_shader_uniform_vector(shgroup->shader, shgroup->orcotexfac, 3, 2, (float *)orcofacs); | 
					
						
							| 
									
										
										
										
											2019-05-07 17:21:26 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-11 17:45:20 +02:00
										 |  |  | static void draw_geometry_execute( | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     DRWShadingGroup *shgroup, GPUBatch *geom, uint start, uint count, bool draw_instance) | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* step 2 : bind vertex array & draw */ | 
					
						
							|  |  |  |   GPU_batch_program_set_no_use( | 
					
						
							|  |  |  |       geom, GPU_shader_get_program(shgroup->shader), GPU_shader_get_interface(shgroup->shader)); | 
					
						
							|  |  |  |   /* XXX hacking gawain. we don't want to call glUseProgram! (huge performance loss) */ | 
					
						
							|  |  |  |   geom->program_in_use = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   GPU_batch_draw_range_ex(geom, start, count, draw_instance); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   geom->program_in_use = false; /* XXX hacking gawain */ | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-16 08:43:52 +01:00
										 |  |  | enum { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BIND_NONE = 0, | 
					
						
							|  |  |  |   BIND_TEMP = 1,    /* Release slot after this shading group. */ | 
					
						
							|  |  |  |   BIND_PERSIST = 2, /* Release slot only after the next shader change. */ | 
					
						
							| 
									
										
										
										
											2018-03-16 08:43:52 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-05 20:45:32 +02:00
										 |  |  | static void set_bound_flags(uint64_t *slots, uint64_t *persist_slots, int slot_idx, char bind_type) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-05-16 14:16:47 +02:00
										 |  |  |   uint64_t slot = 1lu << (unsigned long)slot_idx; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   *slots |= slot; | 
					
						
							|  |  |  |   if (bind_type == BIND_PERSIST) { | 
					
						
							|  |  |  |     *persist_slots |= slot; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-05 20:45:32 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int get_empty_slot_index(uint64_t slots) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   uint64_t empty_slots = ~slots; | 
					
						
							|  |  |  |   /* Find first empty slot using bitscan. */ | 
					
						
							|  |  |  |   if (empty_slots != 0) { | 
					
						
							|  |  |  |     if ((empty_slots & 0xFFFFFFFFlu) != 0) { | 
					
						
							|  |  |  |       return (int)bitscan_forward_uint(empty_slots); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       return (int)bitscan_forward_uint(empty_slots >> 32) + 32; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     /* Greater than GPU_max_textures() */ | 
					
						
							|  |  |  |     return 99999; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-05 20:45:32 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-16 08:43:52 +01:00
										 |  |  | static void bind_texture(GPUTexture *tex, char bind_type) | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   int idx = GPU_texture_bound_number(tex); | 
					
						
							|  |  |  |   if (idx == -1) { | 
					
						
							|  |  |  |     /* Texture isn't bound yet. Find an empty slot and bind it. */ | 
					
						
							|  |  |  |     idx = get_empty_slot_index(DST.RST.bound_tex_slots); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (idx < GPU_max_textures()) { | 
					
						
							|  |  |  |       GPUTexture **gpu_tex_slot = &DST.RST.bound_texs[idx]; | 
					
						
							|  |  |  |       /* Unbind any previous texture. */ | 
					
						
							|  |  |  |       if (*gpu_tex_slot != NULL) { | 
					
						
							|  |  |  |         GPU_texture_unbind(*gpu_tex_slot); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       GPU_texture_bind(tex, idx); | 
					
						
							|  |  |  |       *gpu_tex_slot = tex; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       printf("Not enough texture slots! Reduce number of textures used by your shader.\n"); | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     /* This texture slot was released but the tex
 | 
					
						
							|  |  |  |      * is still bound. Just flag the slot again. */ | 
					
						
							|  |  |  |     BLI_assert(DST.RST.bound_texs[idx] == tex); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   set_bound_flags(&DST.RST.bound_tex_slots, &DST.RST.bound_tex_slots_persist, idx, bind_type); | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-16 08:43:52 +01:00
										 |  |  | static void bind_ubo(GPUUniformBuffer *ubo, char bind_type) | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   int idx = GPU_uniformbuffer_bindpoint(ubo); | 
					
						
							|  |  |  |   if (idx == -1) { | 
					
						
							|  |  |  |     /* UBO isn't bound yet. Find an empty slot and bind it. */ | 
					
						
							|  |  |  |     idx = get_empty_slot_index(DST.RST.bound_ubo_slots); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (idx < GPU_max_ubo_binds()) { | 
					
						
							|  |  |  |       GPUUniformBuffer **gpu_ubo_slot = &DST.RST.bound_ubos[idx]; | 
					
						
							|  |  |  |       /* Unbind any previous UBO. */ | 
					
						
							|  |  |  |       if (*gpu_ubo_slot != NULL) { | 
					
						
							|  |  |  |         GPU_uniformbuffer_unbind(*gpu_ubo_slot); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       GPU_uniformbuffer_bind(ubo, idx); | 
					
						
							|  |  |  |       *gpu_ubo_slot = ubo; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       /* printf so user can report bad behavior */ | 
					
						
							|  |  |  |       printf("Not enough ubo slots! This should not happen!\n"); | 
					
						
							|  |  |  |       /* This is not depending on user input.
 | 
					
						
							|  |  |  |        * It is our responsibility to make sure there is enough slots. */ | 
					
						
							|  |  |  |       BLI_assert(0); | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     /* This UBO slot was released but the UBO is
 | 
					
						
							|  |  |  |      * still bound here. Just flag the slot again. */ | 
					
						
							|  |  |  |     BLI_assert(DST.RST.bound_ubos[idx] == ubo); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   set_bound_flags(&DST.RST.bound_ubo_slots, &DST.RST.bound_ubo_slots_persist, idx, bind_type); | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-02 18:33:19 +02:00
										 |  |  | #ifndef NDEBUG
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Opengl specification is strict on buffer binding. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * " If any active uniform block is not backed by a | 
					
						
							|  |  |  |  * sufficiently large buffer object, the results of shader | 
					
						
							|  |  |  |  * execution are undefined, and may result in GL interruption or | 
					
						
							|  |  |  |  * termination. " - Opengl 3.3 Core Specification | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * For now we only check if the binding is correct. Not the size of | 
					
						
							|  |  |  |  * the bound ubo. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * See T55475. | 
					
						
							|  |  |  |  * */ | 
					
						
							|  |  |  | static bool ubo_bindings_validate(DRWShadingGroup *shgroup) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   bool valid = true; | 
					
						
							| 
									
										
										
										
											2018-08-02 18:33:19 +02:00
										 |  |  | #  ifdef DEBUG_UBO_BINDING
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* Check that all active uniform blocks have a non-zero buffer bound. */ | 
					
						
							|  |  |  |   GLint program = 0; | 
					
						
							|  |  |  |   GLint active_blocks = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   glGetIntegerv(GL_CURRENT_PROGRAM, &program); | 
					
						
							|  |  |  |   glGetProgramiv(program, GL_ACTIVE_UNIFORM_BLOCKS, &active_blocks); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (uint i = 0; i < active_blocks; ++i) { | 
					
						
							|  |  |  |     int binding = 0; | 
					
						
							|  |  |  |     int buffer = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     glGetActiveUniformBlockiv(program, i, GL_UNIFORM_BLOCK_BINDING, &binding); | 
					
						
							|  |  |  |     glGetIntegeri_v(GL_UNIFORM_BUFFER_BINDING, binding, &buffer); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (buffer == 0) { | 
					
						
							|  |  |  |       char blockname[64]; | 
					
						
							|  |  |  |       glGetActiveUniformBlockName(program, i, sizeof(blockname), NULL, blockname); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (valid) { | 
					
						
							|  |  |  |         printf("Trying to draw with missing UBO binding.\n"); | 
					
						
							|  |  |  |         valid = false; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       printf("Pass : %s, Shader : %s, Block : %s\n", | 
					
						
							|  |  |  |              shgroup->pass_parent->name, | 
					
						
							|  |  |  |              shgroup->shader->name, | 
					
						
							|  |  |  |              blockname); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-08-02 18:33:19 +02:00
										 |  |  | #  endif
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return valid; | 
					
						
							| 
									
										
										
										
											2018-08-02 18:33:19 +02:00
										 |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-16 08:43:52 +01:00
										 |  |  | static void release_texture_slots(bool with_persist) | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (with_persist) { | 
					
						
							|  |  |  |     DST.RST.bound_tex_slots = 0; | 
					
						
							|  |  |  |     DST.RST.bound_tex_slots_persist = 0; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     DST.RST.bound_tex_slots &= DST.RST.bound_tex_slots_persist; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-16 08:43:52 +01:00
										 |  |  | static void release_ubo_slots(bool with_persist) | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (with_persist) { | 
					
						
							|  |  |  |     DST.RST.bound_ubo_slots = 0; | 
					
						
							|  |  |  |     DST.RST.bound_ubo_slots_persist = 0; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     DST.RST.bound_ubo_slots &= DST.RST.bound_ubo_slots_persist; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-19 16:54:01 +02:00
										 |  |  | static void draw_update_uniforms(DRWShadingGroup *shgroup) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   for (DRWUniform *uni = shgroup->uniforms; uni; uni = uni->next) { | 
					
						
							|  |  |  |     GPUTexture *tex; | 
					
						
							|  |  |  |     GPUUniformBuffer *ubo; | 
					
						
							|  |  |  |     if (uni->location == -2) { | 
					
						
							|  |  |  |       uni->location = GPU_shader_get_uniform_ensure(shgroup->shader, | 
					
						
							|  |  |  |                                                     DST.uniform_names.buffer + uni->name_ofs); | 
					
						
							|  |  |  |       if (uni->location == -1) { | 
					
						
							|  |  |  |         continue; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     const void *data = uni->pvalue; | 
					
						
							|  |  |  |     if (ELEM(uni->type, DRW_UNIFORM_INT_COPY, DRW_UNIFORM_FLOAT_COPY)) { | 
					
						
							|  |  |  |       data = uni->fvalue; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     switch (uni->type) { | 
					
						
							|  |  |  |       case DRW_UNIFORM_INT_COPY: | 
					
						
							|  |  |  |       case DRW_UNIFORM_INT: | 
					
						
							|  |  |  |         GPU_shader_uniform_vector_int( | 
					
						
							|  |  |  |             shgroup->shader, uni->location, uni->length, uni->arraysize, data); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case DRW_UNIFORM_FLOAT_COPY: | 
					
						
							|  |  |  |       case DRW_UNIFORM_FLOAT: | 
					
						
							|  |  |  |         GPU_shader_uniform_vector( | 
					
						
							|  |  |  |             shgroup->shader, uni->location, uni->length, uni->arraysize, data); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case DRW_UNIFORM_TEXTURE: | 
					
						
							|  |  |  |         tex = (GPUTexture *)uni->pvalue; | 
					
						
							|  |  |  |         BLI_assert(tex); | 
					
						
							|  |  |  |         bind_texture(tex, BIND_TEMP); | 
					
						
							|  |  |  |         GPU_shader_uniform_texture(shgroup->shader, uni->location, tex); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case DRW_UNIFORM_TEXTURE_PERSIST: | 
					
						
							|  |  |  |         tex = (GPUTexture *)uni->pvalue; | 
					
						
							|  |  |  |         BLI_assert(tex); | 
					
						
							|  |  |  |         bind_texture(tex, BIND_PERSIST); | 
					
						
							|  |  |  |         GPU_shader_uniform_texture(shgroup->shader, uni->location, tex); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case DRW_UNIFORM_TEXTURE_REF: | 
					
						
							|  |  |  |         tex = *((GPUTexture **)uni->pvalue); | 
					
						
							|  |  |  |         BLI_assert(tex); | 
					
						
							|  |  |  |         bind_texture(tex, BIND_TEMP); | 
					
						
							|  |  |  |         GPU_shader_uniform_texture(shgroup->shader, uni->location, tex); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case DRW_UNIFORM_BLOCK: | 
					
						
							|  |  |  |         ubo = (GPUUniformBuffer *)uni->pvalue; | 
					
						
							|  |  |  |         bind_ubo(ubo, BIND_TEMP); | 
					
						
							|  |  |  |         GPU_shader_uniform_buffer(shgroup->shader, uni->location, ubo); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case DRW_UNIFORM_BLOCK_PERSIST: | 
					
						
							|  |  |  |         ubo = (GPUUniformBuffer *)uni->pvalue; | 
					
						
							|  |  |  |         bind_ubo(ubo, BIND_PERSIST); | 
					
						
							|  |  |  |         GPU_shader_uniform_buffer(shgroup->shader, uni->location, ubo); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   BLI_assert(ubo_bindings_validate(shgroup)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-13 17:56:20 +02:00
										 |  |  | BLI_INLINE bool draw_select_do_call(DRWShadingGroup *shgroup, DRWCall *call) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #ifdef USE_GPU_SELECT
 | 
					
						
							|  |  |  |   if ((G.f & G_FLAG_PICKSEL) == 0) { | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (call->inst_selectid != NULL) { | 
					
						
							|  |  |  |     const bool is_instancing = (call->inst_count != 0); | 
					
						
							|  |  |  |     uint start = 0; | 
					
						
							|  |  |  |     uint count = 1; | 
					
						
							|  |  |  |     uint tot = is_instancing ? call->inst_count : call->vert_count; | 
					
						
							|  |  |  |     /* Hack : get vbo data without actually drawing. */ | 
					
						
							|  |  |  |     GPUVertBufRaw raw; | 
					
						
							|  |  |  |     GPU_vertbuf_attr_get_raw_data(call->inst_selectid, 0, &raw); | 
					
						
							|  |  |  |     int *select_id = GPU_vertbuf_raw_step(&raw); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Batching */ | 
					
						
							|  |  |  |     if (!is_instancing) { | 
					
						
							|  |  |  |       /* FIXME: Meh a bit nasty. */ | 
					
						
							|  |  |  |       if (call->batch->gl_prim_type == convert_prim_type_to_gl(GPU_PRIM_TRIS)) { | 
					
						
							|  |  |  |         count = 3; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else if (call->batch->gl_prim_type == convert_prim_type_to_gl(GPU_PRIM_LINES)) { | 
					
						
							|  |  |  |         count = 2; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while (start < tot) { | 
					
						
							|  |  |  |       GPU_select_load_id(select_id[start]); | 
					
						
							|  |  |  |       draw_geometry_execute(shgroup, call->batch, start, count, is_instancing); | 
					
						
							|  |  |  |       start += count; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     GPU_select_load_id(call->select_id); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |   return false; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | static void draw_shgroup(DRWShadingGroup *shgroup, DRWState pass_state) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BLI_assert(shgroup->shader); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const bool shader_changed = (DST.shader != shgroup->shader); | 
					
						
							|  |  |  |   bool use_tfeedback = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (shader_changed) { | 
					
						
							|  |  |  |     if (DST.shader) { | 
					
						
							|  |  |  |       GPU_shader_unbind(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     GPU_shader_bind(shgroup->shader); | 
					
						
							|  |  |  |     DST.shader = shgroup->shader; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-17 14:04:30 +02:00
										 |  |  |   if (shgroup->tfeedback_target != NULL) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     use_tfeedback = GPU_shader_transform_feedback_enable(shgroup->shader, | 
					
						
							|  |  |  |                                                          shgroup->tfeedback_target->vbo_id); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   release_ubo_slots(shader_changed); | 
					
						
							|  |  |  |   release_texture_slots(shader_changed); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   drw_state_set((pass_state & shgroup->state_extra_disable) | shgroup->state_extra); | 
					
						
							|  |  |  |   drw_stencil_set(shgroup->stencil_mask); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-19 16:54:01 +02:00
										 |  |  |   draw_update_uniforms(shgroup); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Rendering Calls */ | 
					
						
							| 
									
										
										
										
											2019-05-13 17:56:20 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     bool prev_neg_scale = false; | 
					
						
							|  |  |  |     int callid = 0; | 
					
						
							|  |  |  |     for (DRWCall *call = shgroup->calls.first; call; call = call->next) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* OPTI/IDEA(clem): Do this preparation in another thread. */ | 
					
						
							|  |  |  |       draw_visibility_eval(call->state); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-19 16:54:01 +02:00
										 |  |  |       if ((call->state->flag & DRW_CALL_CULLED) != 0) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         continue; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* XXX small exception/optimisation for outline rendering. */ | 
					
						
							|  |  |  |       if (shgroup->callid != -1) { | 
					
						
							|  |  |  |         GPU_shader_uniform_vector_int(shgroup->shader, shgroup->callid, 1, 1, &callid); | 
					
						
							|  |  |  |         callid += 1; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* Negative scale objects */ | 
					
						
							|  |  |  |       bool neg_scale = call->state->flag & DRW_CALL_NEGSCALE; | 
					
						
							|  |  |  |       if (neg_scale != prev_neg_scale) { | 
					
						
							| 
									
										
										
										
											2019-05-19 16:54:01 +02:00
										 |  |  |         glFrontFace((neg_scale) ? GL_CW : GL_CCW); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         prev_neg_scale = neg_scale; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       draw_geometry_prepare(shgroup, call); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-13 17:56:20 +02:00
										 |  |  |       if (draw_select_do_call(shgroup, call)) { | 
					
						
							|  |  |  |         continue; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-11 19:16:46 +02:00
										 |  |  |       /* TODO revisit when DRW_SHG_INSTANCE and the like is gone. */ | 
					
						
							|  |  |  |       if (call->inst_count == 0) { | 
					
						
							|  |  |  |         draw_geometry_execute(shgroup, call->batch, call->vert_first, call->vert_count, false); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         draw_geometry_execute(shgroup, call->batch, 0, call->inst_count, true); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     /* Reset state */ | 
					
						
							| 
									
										
										
										
											2019-05-19 16:54:01 +02:00
										 |  |  |     glFrontFace(GL_CCW); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (use_tfeedback) { | 
					
						
							|  |  |  |     GPU_shader_transform_feedback_disable(shgroup->shader); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-09 19:52:37 +01:00
										 |  |  | static void drw_update_view(void) | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (DST.dirty_mat) { | 
					
						
							|  |  |  |     DST.state_cache_id++; | 
					
						
							|  |  |  |     DST.dirty_mat = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     DRW_uniformbuffer_update(G_draw.view_ubo, &DST.view_data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Catch integer wrap around. */ | 
					
						
							|  |  |  |     if (UNLIKELY(DST.state_cache_id == 0)) { | 
					
						
							|  |  |  |       DST.state_cache_id = 1; | 
					
						
							|  |  |  |       /* We must reset all CallStates to ensure that not
 | 
					
						
							|  |  |  |        * a single one stayed with cache_id equal to 1. */ | 
					
						
							| 
									
										
										
										
											2019-05-07 18:01:14 +02:00
										 |  |  |       BLI_memblock_iter iter; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       DRWCallState *state; | 
					
						
							| 
									
										
										
										
											2019-05-07 18:01:14 +02:00
										 |  |  |       BLI_memblock_iternew(DST.vmempool->states, &iter); | 
					
						
							|  |  |  |       while ((state = BLI_memblock_iterstep(&iter))) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         state->cache_id = 0; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* TODO dispatch threads to compute matrices/culling */ | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   draw_clipping_setup_from_view(); | 
					
						
							| 
									
										
										
										
											2018-03-09 19:52:37 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | static void drw_draw_pass_ex(DRWPass *pass, | 
					
						
							|  |  |  |                              DRWShadingGroup *start_group, | 
					
						
							|  |  |  |                              DRWShadingGroup *end_group) | 
					
						
							| 
									
										
										
										
											2018-03-09 19:52:37 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (start_group == NULL) { | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   DST.shader = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   BLI_assert(DST.buffer_finish_called && | 
					
						
							|  |  |  |              "DRW_render_instance_buffer_finish had not been called before drawing"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   drw_update_view(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* GPU_framebuffer_clear calls can change the state outside the DRW module.
 | 
					
						
							|  |  |  |    * Force reset the affected states to avoid problems later. */ | 
					
						
							|  |  |  |   drw_state_set(DST.state | DRW_STATE_WRITE_DEPTH | DRW_STATE_WRITE_COLOR); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   drw_state_set(pass->state); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   DRW_stats_query_start(pass->name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (DRWShadingGroup *shgroup = start_group; shgroup; shgroup = shgroup->next) { | 
					
						
							|  |  |  |     draw_shgroup(shgroup, pass->state); | 
					
						
							|  |  |  |     /* break if upper limit */ | 
					
						
							|  |  |  |     if (shgroup == end_group) { | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Clear Bound textures */ | 
					
						
							|  |  |  |   for (int i = 0; i < DST_MAX_SLOTS; i++) { | 
					
						
							|  |  |  |     if (DST.RST.bound_texs[i] != NULL) { | 
					
						
							|  |  |  |       GPU_texture_unbind(DST.RST.bound_texs[i]); | 
					
						
							|  |  |  |       DST.RST.bound_texs[i] = NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Clear Bound Ubos */ | 
					
						
							|  |  |  |   for (int i = 0; i < DST_MAX_SLOTS; i++) { | 
					
						
							|  |  |  |     if (DST.RST.bound_ubos[i] != NULL) { | 
					
						
							|  |  |  |       GPU_uniformbuffer_unbind(DST.RST.bound_ubos[i]); | 
					
						
							|  |  |  |       DST.RST.bound_ubos[i] = NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (DST.shader) { | 
					
						
							|  |  |  |     GPU_shader_unbind(); | 
					
						
							|  |  |  |     DST.shader = NULL; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* HACK: Rasterized discard can affect clear commands which are not
 | 
					
						
							|  |  |  |    * part of a DRWPass (as of now). So disable rasterized discard here | 
					
						
							|  |  |  |    * if it has been enabled. */ | 
					
						
							|  |  |  |   if ((DST.state & DRW_STATE_RASTERIZER_ENABLED) == 0) { | 
					
						
							|  |  |  |     drw_state_set((DST.state & ~DRW_STATE_RASTERIZER_ENABLED) | DRW_STATE_DEFAULT); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   DRW_stats_query_end(); | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DRW_draw_pass(DRWPass *pass) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   drw_draw_pass_ex(pass, pass->shgroups.first, pass->shgroups.last); | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Draw only a subset of shgroups. Used in special situations as grease pencil strokes */ | 
					
						
							|  |  |  | void DRW_draw_pass_subset(DRWPass *pass, DRWShadingGroup *start_group, DRWShadingGroup *end_group) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   drw_draw_pass_ex(pass, start_group, end_group); | 
					
						
							| 
									
										
										
										
											2018-02-28 01:16:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \} */ |