| 
									
										
										
										
											2014-07-23 15:24:07 +02:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU General Public License | 
					
						
							|  |  |  |  * as published by the Free Software Foundation; either version 2 | 
					
						
							|  |  |  |  * of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  * GNU General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  * along with this program; if not, write to the Free Software Foundation, | 
					
						
							|  |  |  |  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The Original Code is Copyright (C) 2014 Blender Foundation. | 
					
						
							|  |  |  |  * All rights reserved. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-18 08:08:12 +11:00
										 |  |  | /** \file
 | 
					
						
							|  |  |  |  * \ingroup gpu | 
					
						
							| 
									
										
										
										
											2014-07-23 15:24:07 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2019-01-16 15:18:44 +01:00
										 |  |  |  * Interface for accessing gpu-related methods for selection. The semantics are | 
					
						
							|  |  |  |  * similar to glRenderMode(GL_SELECT) from older OpenGL versions. | 
					
						
							| 
									
										
										
										
											2014-07-23 15:24:07 +02:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | #include <stdlib.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include <string.h>
 | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-07 15:46:19 -05:00
										 |  |  | #include "GPU_glew.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "GPU_select.h"
 | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-23 15:24:07 +02:00
										 |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-28 15:59:51 -03:00
										 |  |  | #include "BLI_rect.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-23 15:24:07 +02:00
										 |  |  | #include "DNA_userdef_types.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-29 19:12:33 +01:00
										 |  |  | #include "BLI_utildefines.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | #include "gpu_select_private.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Internal algorithm used */ | 
					
						
							|  |  |  | enum { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /** glBegin/EndQuery(GL_SAMPLES_PASSED... ), `gpu_select_query.c`
 | 
					
						
							|  |  |  |    * Only sets 4th component (ID) correctly. */ | 
					
						
							|  |  |  |   ALGO_GL_QUERY = 1, | 
					
						
							|  |  |  |   /** Read depth buffer for every drawing pass and extract depths, `gpu_select_pick.c`
 | 
					
						
							|  |  |  |    * Only sets 4th component (ID) correctly. */ | 
					
						
							|  |  |  |   ALGO_GL_PICK = 2, | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct GPUSelectState { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* To ignore selection id calls when not initialized */ | 
					
						
							|  |  |  |   bool select_is_active; | 
					
						
							|  |  |  |   /* mode of operation */ | 
					
						
							|  |  |  |   char mode; | 
					
						
							|  |  |  |   /* internal algorithm for selection */ | 
					
						
							|  |  |  |   char algorithm; | 
					
						
							|  |  |  |   /* allow GPU_select_begin/end without drawing */ | 
					
						
							|  |  |  |   bool use_cache; | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | } GPUSelectState; | 
					
						
							| 
									
										
										
										
											2014-07-23 15:24:07 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | static GPUSelectState g_select_state = {0}; | 
					
						
							| 
									
										
										
										
											2014-07-23 15:24:07 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-10 07:12:10 +11:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * initialize and provide buffer for results | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2018-02-28 11:37:39 +11:00
										 |  |  | void GPU_select_begin(uint *buffer, uint bufsize, const rcti *input, char mode, int oldhits) | 
					
						
							| 
									
										
										
										
											2014-07-23 15:24:07 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (mode == GPU_SELECT_NEAREST_SECOND_PASS) { | 
					
						
							| 
									
										
										
										
											2019-04-22 01:45:43 +10:00
										 |  |  |     /* In the case hits was '-1',
 | 
					
						
							|  |  |  |      * don't start the second pass since it's not going to give useful results. | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |      * As well as buffer overflow in 'gpu_select_query_load_id'. */ | 
					
						
							|  |  |  |     BLI_assert(oldhits != -1); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   g_select_state.select_is_active = true; | 
					
						
							|  |  |  |   g_select_state.mode = mode; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (ELEM(g_select_state.mode, GPU_SELECT_PICK_ALL, GPU_SELECT_PICK_NEAREST)) { | 
					
						
							|  |  |  |     g_select_state.algorithm = ALGO_GL_PICK; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     g_select_state.algorithm = ALGO_GL_QUERY; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   switch (g_select_state.algorithm) { | 
					
						
							|  |  |  |     case ALGO_GL_QUERY: { | 
					
						
							|  |  |  |       g_select_state.use_cache = false; | 
					
						
							|  |  |  |       gpu_select_query_begin((uint(*)[4])buffer, bufsize / 4, input, mode, oldhits); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     default: /* ALGO_GL_PICK */ | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       gpu_select_pick_begin((uint(*)[4])buffer, bufsize / 4, input, mode); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2014-07-23 15:24:07 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-10 07:12:10 +11:00
										 |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2019-04-22 01:45:43 +10:00
										 |  |  |  * loads a new selection id and ends previous query, if any. | 
					
						
							|  |  |  |  * In second pass of selection it also returns | 
					
						
							| 
									
										
										
										
											2016-01-10 07:12:10 +11:00
										 |  |  |  * if id has been hit on the first pass already. | 
					
						
							|  |  |  |  * Thus we can skip drawing un-hit objects. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \warning We rely on the order of object rendering on passes to be the same for this to work. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2018-02-28 11:37:39 +11:00
										 |  |  | bool GPU_select_load_id(uint id) | 
					
						
							| 
									
										
										
										
											2014-07-23 15:24:07 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* if no selection mode active, ignore */ | 
					
						
							| 
									
										
										
										
											2019-04-22 09:32:37 +10:00
										 |  |  |   if (!g_select_state.select_is_active) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     return true; | 
					
						
							| 
									
										
										
										
											2019-04-22 09:32:37 +10:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   switch (g_select_state.algorithm) { | 
					
						
							|  |  |  |     case ALGO_GL_QUERY: { | 
					
						
							|  |  |  |       return gpu_select_query_load_id(id); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     default: /* ALGO_GL_PICK */ | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-04-27 20:37:11 +02:00
										 |  |  |       return gpu_select_pick_load_id(id, false); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2014-07-23 15:24:07 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-10 07:12:10 +11:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Cleanup and flush selection results to buffer. | 
					
						
							|  |  |  |  * Return number of hits and hits in buffer. | 
					
						
							|  |  |  |  * if \a dopass is true, we will do a second pass with occlusion queries to get the closest hit. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2018-02-28 11:37:39 +11:00
										 |  |  | uint GPU_select_end(void) | 
					
						
							| 
									
										
										
										
											2014-07-23 15:24:07 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   uint hits = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   switch (g_select_state.algorithm) { | 
					
						
							|  |  |  |     case ALGO_GL_QUERY: { | 
					
						
							|  |  |  |       hits = gpu_select_query_end(); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     default: /* ALGO_GL_PICK */ | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       hits = gpu_select_pick_end(); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   g_select_state.select_is_active = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return hits; | 
					
						
							| 
									
										
										
										
											2014-07-23 15:24:07 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | /* ----------------------------------------------------------------------------
 | 
					
						
							|  |  |  |  * Caching | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Support multiple begin/end's as long as they are within the initial region. | 
					
						
							|  |  |  |  * Currently only used by ALGO_GL_PICK. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GPU_select_cache_begin(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* validate on GPU_select_begin, clear if not supported */ | 
					
						
							|  |  |  |   BLI_assert(g_select_state.use_cache == false); | 
					
						
							|  |  |  |   g_select_state.use_cache = true; | 
					
						
							|  |  |  |   if (g_select_state.algorithm == ALGO_GL_PICK) { | 
					
						
							|  |  |  |     gpu_select_pick_cache_begin(); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GPU_select_cache_load_id(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BLI_assert(g_select_state.use_cache == true); | 
					
						
							|  |  |  |   if (g_select_state.algorithm == ALGO_GL_PICK) { | 
					
						
							|  |  |  |     gpu_select_pick_cache_load_id(); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GPU_select_cache_end(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (g_select_state.algorithm == ALGO_GL_PICK) { | 
					
						
							|  |  |  |     gpu_select_pick_cache_end(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   g_select_state.use_cache = false; | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool GPU_select_is_cached(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return g_select_state.use_cache && gpu_select_pick_is_cached(); | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2017-07-19 20:12:24 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* ----------------------------------------------------------------------------
 | 
					
						
							|  |  |  |  * Utilities | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Helper function, nothing special but avoids doing inline since hit's aren't sorted by depth | 
					
						
							|  |  |  |  * and purpose of 4x buffer indices isn't so clear. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Note that comparing depth as uint is fine. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | const uint *GPU_select_buffer_near(const uint *buffer, int hits) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const uint *buffer_near = NULL; | 
					
						
							|  |  |  |   uint depth_min = (uint)-1; | 
					
						
							|  |  |  |   for (int i = 0; i < hits; i++) { | 
					
						
							|  |  |  |     if (buffer[1] < depth_min) { | 
					
						
							|  |  |  |       BLI_assert(buffer[3] != -1); | 
					
						
							|  |  |  |       depth_min = buffer[1]; | 
					
						
							|  |  |  |       buffer_near = buffer; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     buffer += 4; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return buffer_near; | 
					
						
							| 
									
										
										
										
											2017-07-26 02:34:14 +10:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-03-28 15:59:51 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-10 17:50:11 +10:00
										 |  |  | uint GPU_select_buffer_remove_by_id(uint *buffer, int hits, uint select_id) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   uint *buffer_src = buffer; | 
					
						
							|  |  |  |   uint *buffer_dst = buffer; | 
					
						
							|  |  |  |   int hits_final = 0; | 
					
						
							|  |  |  |   for (int i = 0; i < hits; i++) { | 
					
						
							|  |  |  |     if (buffer_src[3] != select_id) { | 
					
						
							|  |  |  |       if (buffer_dst != buffer_src) { | 
					
						
							|  |  |  |         memcpy(buffer_dst, buffer_src, sizeof(int[4])); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       buffer_dst += 4; | 
					
						
							|  |  |  |       hits_final += 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     buffer_src += 4; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return hits_final; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-28 15:59:51 -03:00
										 |  |  | /* Part of the solution copied from `rect_subregion_stride_calc`. */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | void GPU_select_buffer_stride_realign(const rcti *src, const rcti *dst, uint *r_buf) | 
					
						
							| 
									
										
										
										
											2019-03-28 15:59:51 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const int x = dst->xmin - src->xmin; | 
					
						
							|  |  |  |   const int y = dst->ymin - src->ymin; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   BLI_assert(src->xmin <= dst->xmin && src->ymin <= dst->ymin && src->xmax >= dst->xmax && | 
					
						
							|  |  |  |              src->ymax >= dst->ymax); | 
					
						
							|  |  |  |   BLI_assert(x >= 0 && y >= 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const int src_x = BLI_rcti_size_x(src); | 
					
						
							|  |  |  |   const int src_y = BLI_rcti_size_y(src); | 
					
						
							|  |  |  |   const int dst_x = BLI_rcti_size_x(dst); | 
					
						
							|  |  |  |   const int dst_y = BLI_rcti_size_y(dst); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   int last_px_id = src_x * (y + dst_y - 1) + (x + dst_x - 1); | 
					
						
							|  |  |  |   memset(&r_buf[last_px_id + 1], 0, (src_x * src_y - (last_px_id + 1)) * sizeof(*r_buf)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-21 20:56:27 -03:00
										 |  |  |   if (last_px_id < 0) { | 
					
						
							|  |  |  |     /* Nothing to write. */ | 
					
						
							|  |  |  |     BLI_assert(last_px_id == -1); | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   int last_px_written = dst_x * dst_y - 1; | 
					
						
							|  |  |  |   const int skip = src_x - dst_x; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   while (true) { | 
					
						
							|  |  |  |     for (int i = dst_x; i--;) { | 
					
						
							|  |  |  |       r_buf[last_px_id--] = r_buf[last_px_written--]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (last_px_written < 0) { | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     last_px_id -= skip; | 
					
						
							|  |  |  |     memset(&r_buf[last_px_id + 1], 0, skip * sizeof(*r_buf)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   memset(r_buf, 0, (last_px_id + 1) * sizeof(*r_buf)); | 
					
						
							| 
									
										
										
										
											2019-03-28 15:59:51 -03:00
										 |  |  | } |