| 
									
										
										
										
											2017-03-09 05:17:55 +11: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) 2017 Blender Foundation. | 
					
						
							|  |  |  |  * All rights reserved. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-18 08:08:12 +11:00
										 |  |  | /** \file
 | 
					
						
							|  |  |  |  * \ingroup gpu | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Custom select code for picking small regions (not efficient for large regions). | 
					
						
							|  |  |  |  * `gpu_select_pick_*` API. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | #include <float.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-20 16:57:38 +02:00
										 |  |  | #include "GPU_framebuffer.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "GPU_glew.h"
 | 
					
						
							|  |  |  | #include "GPU_immediate.h"
 | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | #include "GPU_select.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-11 14:52:57 +11:00
										 |  |  | #include "GPU_state.h"
 | 
					
						
							| 
									
										
										
										
											2018-06-04 09:09:12 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "BLI_listbase.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-19 09:33:03 +01:00
										 |  |  | #include "BLI_rect.h"
 | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | #include "BLI_utildefines.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "gpu_select_private.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 21:47:43 +11:00
										 |  |  | #include "BLI_strict_flags.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | /* #define DEBUG_PRINT */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Alloc number for depths */ | 
					
						
							|  |  |  | #define ALLOC_DEPTHS 200
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Z-depth of cleared depth buffer */ | 
					
						
							|  |  |  | #define DEPTH_MAX 0xffffffff
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ----------------------------------------------------------------------------
 | 
					
						
							|  |  |  |  * SubRectStride | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* For looping over a sub-region of a rect, could be moved into 'rct.c'*/ | 
					
						
							|  |  |  | typedef struct SubRectStride { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   uint start;    /* start here */ | 
					
						
							|  |  |  |   uint span;     /* read these */ | 
					
						
							|  |  |  |   uint span_len; /* len times (read span 'len' times). */ | 
					
						
							|  |  |  |   uint skip;     /* skip those */ | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | } SubRectStride; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* we may want to change back to float if uint isn't well supported */ | 
					
						
							| 
									
										
										
										
											2018-02-28 11:37:39 +11:00
										 |  |  | typedef uint depth_t; | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Calculate values needed for looping over a sub-region (smaller buffer within a larger buffer). | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 'src' must be bigger than 'dst'. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static void rect_subregion_stride_calc(const rcti *src, const rcti *dst, SubRectStride *r_sub) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   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); | 
					
						
							|  |  |  |   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); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   r_sub->start = (uint)((src_x * y) + x); | 
					
						
							|  |  |  |   r_sub->span = (uint)dst_x; | 
					
						
							|  |  |  |   r_sub->span_len = (uint)dst_y; | 
					
						
							|  |  |  |   r_sub->skip = (uint)(src_x - dst_x); | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 03:15:58 +11:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Ignore depth clearing as a change, | 
					
						
							|  |  |  |  * only check if its been changed _and_ filled in (ignore clearing since XRAY does this). | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | BLI_INLINE bool depth_is_filled(const depth_t *prev, const depth_t *curr) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return (*prev != *curr) && (*curr != DEPTH_MAX); | 
					
						
							| 
									
										
										
										
											2017-03-10 03:15:58 +11:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | /* ----------------------------------------------------------------------------
 | 
					
						
							|  |  |  |  * DepthBufCache | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Result of reading glReadPixels, | 
					
						
							|  |  |  |  * use for both cache and non-cached storage. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* store result of glReadPixels */ | 
					
						
							|  |  |  | typedef struct DepthBufCache { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   struct DepthBufCache *next, *prev; | 
					
						
							|  |  |  |   uint id; | 
					
						
							|  |  |  |   depth_t buf[0]; | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | } DepthBufCache; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-28 11:37:39 +11:00
										 |  |  | static DepthBufCache *depth_buf_malloc(uint rect_len) | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   DepthBufCache *rect = MEM_mallocN(sizeof(DepthBufCache) + sizeof(depth_t) * rect_len, __func__); | 
					
						
							|  |  |  |   rect->id = SELECT_ID_NONE; | 
					
						
							|  |  |  |   return rect; | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | static bool depth_buf_rect_depth_any(const DepthBufCache *rect_depth, uint rect_len) | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const depth_t *curr = rect_depth->buf; | 
					
						
							|  |  |  |   for (uint i = 0; i < rect_len; i++, curr++) { | 
					
						
							|  |  |  |     if (*curr != DEPTH_MAX) { | 
					
						
							|  |  |  |       return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return false; | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | static bool depth_buf_subrect_depth_any(const DepthBufCache *rect_depth, | 
					
						
							|  |  |  |                                         const SubRectStride *sub_rect) | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const depth_t *curr = rect_depth->buf + sub_rect->start; | 
					
						
							|  |  |  |   for (uint i = 0; i < sub_rect->span_len; i++) { | 
					
						
							|  |  |  |     const depth_t *curr_end = curr + sub_rect->span; | 
					
						
							|  |  |  |     for (; curr < curr_end; curr++, curr++) { | 
					
						
							|  |  |  |       if (*curr != DEPTH_MAX) { | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     curr += sub_rect->skip; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return false; | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | static bool depth_buf_rect_depth_any_filled(const DepthBufCache *rect_prev, | 
					
						
							|  |  |  |                                             const DepthBufCache *rect_curr, | 
					
						
							|  |  |  |                                             uint rect_len) | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-03-10 03:15:58 +11:00
										 |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return memcmp(rect_depth_a->buf, rect_depth_b->buf, rect_len * sizeof(depth_t)) != 0; | 
					
						
							| 
									
										
										
										
											2017-03-10 03:15:58 +11:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const depth_t *prev = rect_prev->buf; | 
					
						
							|  |  |  |   const depth_t *curr = rect_curr->buf; | 
					
						
							|  |  |  |   for (uint i = 0; i < rect_len; i++, curr++, prev++) { | 
					
						
							|  |  |  |     if (depth_is_filled(prev, curr)) { | 
					
						
							|  |  |  |       return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return false; | 
					
						
							| 
									
										
										
										
											2017-03-10 03:15:58 +11:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Both buffers are the same size, just check if the sub-rect contains any differences. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | static bool depth_buf_subrect_depth_any_filled(const DepthBufCache *rect_src, | 
					
						
							|  |  |  |                                                const DepthBufCache *rect_dst, | 
					
						
							|  |  |  |                                                const SubRectStride *sub_rect) | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* same as above but different rect sizes */ | 
					
						
							|  |  |  |   const depth_t *prev = rect_src->buf + sub_rect->start; | 
					
						
							|  |  |  |   const depth_t *curr = rect_dst->buf + sub_rect->start; | 
					
						
							|  |  |  |   for (uint i = 0; i < sub_rect->span_len; i++) { | 
					
						
							|  |  |  |     const depth_t *curr_end = curr + sub_rect->span; | 
					
						
							|  |  |  |     for (; curr < curr_end; prev++, curr++) { | 
					
						
							|  |  |  |       if (depth_is_filled(prev, curr)) { | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     prev += sub_rect->skip; | 
					
						
							|  |  |  |     curr += sub_rect->skip; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return false; | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ----------------------------------------------------------------------------
 | 
					
						
							|  |  |  |  * DepthID | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Internal structure for storing hits. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct DepthID { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   uint id; | 
					
						
							|  |  |  |   depth_t depth; | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | } DepthID; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int depth_id_cmp(const void *v1, const void *v2) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const DepthID *d1 = v1, *d2 = v2; | 
					
						
							|  |  |  |   if (d1->id < d2->id) { | 
					
						
							|  |  |  |     return -1; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-08-07 12:39:35 +02:00
										 |  |  |   if (d1->id > d2->id) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     return 1; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-08-07 12:39:35 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return 0; | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int depth_cmp(const void *v1, const void *v2) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const DepthID *d1 = v1, *d2 = v2; | 
					
						
							|  |  |  |   if (d1->depth < d2->depth) { | 
					
						
							|  |  |  |     return -1; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-08-07 12:39:35 +02:00
										 |  |  |   if (d1->depth > d2->depth) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     return 1; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-08-07 12:39:35 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return 0; | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* depth sorting */ | 
					
						
							|  |  |  | typedef struct GPUPickState { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* cache on initialization */ | 
					
						
							|  |  |  |   uint (*buffer)[4]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* buffer size (stores number of integers, for actual size multiply by sizeof integer)*/ | 
					
						
							|  |  |  |   uint bufsize; | 
					
						
							|  |  |  |   /* mode of operation */ | 
					
						
							|  |  |  |   char mode; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* OpenGL drawing, never use when (is_cached == true). */ | 
					
						
							|  |  |  |   struct { | 
					
						
							|  |  |  |     /* The current depth, accumulated as we draw */ | 
					
						
							|  |  |  |     DepthBufCache *rect_depth; | 
					
						
							|  |  |  |     /* Scratch buffer, avoid allocs every time (when not caching) */ | 
					
						
							|  |  |  |     DepthBufCache *rect_depth_test; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Pass to glReadPixels (x, y, w, h) */ | 
					
						
							|  |  |  |     int clip_readpixels[4]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Set after first draw */ | 
					
						
							|  |  |  |     bool is_init; | 
					
						
							|  |  |  |     uint prev_id; | 
					
						
							|  |  |  |   } gl; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* src: data stored in 'cache' and 'gl',
 | 
					
						
							|  |  |  |    * dst: use when cached region is smaller (where src -> dst isn't 1:1) */ | 
					
						
							|  |  |  |   struct { | 
					
						
							|  |  |  |     rcti clip_rect; | 
					
						
							|  |  |  |     uint rect_len; | 
					
						
							|  |  |  |   } src, dst; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Store cache between `GPU_select_cache_begin/end` */ | 
					
						
							|  |  |  |   bool use_cache; | 
					
						
							|  |  |  |   bool is_cached; | 
					
						
							|  |  |  |   struct { | 
					
						
							|  |  |  |     /* Cleanup used for iterating over both source and destination buffers:
 | 
					
						
							|  |  |  |      * src.clip_rect -> dst.clip_rect */ | 
					
						
							|  |  |  |     SubRectStride sub_rect; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* List of DepthBufCache, sized of 'src.clip_rect' */ | 
					
						
							|  |  |  |     ListBase bufs; | 
					
						
							|  |  |  |   } cache; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Pickign methods */ | 
					
						
							|  |  |  |   union { | 
					
						
							|  |  |  |     /* GPU_SELECT_PICK_ALL */ | 
					
						
							|  |  |  |     struct { | 
					
						
							|  |  |  |       DepthID *hits; | 
					
						
							|  |  |  |       uint hits_len; | 
					
						
							|  |  |  |       uint hits_len_alloc; | 
					
						
							|  |  |  |     } all; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* GPU_SELECT_PICK_NEAREST */ | 
					
						
							|  |  |  |     struct { | 
					
						
							|  |  |  |       uint *rect_id; | 
					
						
							|  |  |  |     } nearest; | 
					
						
							|  |  |  |   }; | 
					
						
							| 
									
										
										
										
											2020-08-16 23:20:59 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Previous state to restore after drawing. */ | 
					
						
							|  |  |  |   int viewport[4]; | 
					
						
							|  |  |  |   int scissor[4]; | 
					
						
							|  |  |  |   eGPUWriteMask write_mask; | 
					
						
							| 
									
										
										
										
											2020-08-20 16:38:34 +02:00
										 |  |  |   eGPUDepthTest depth_test; | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | } GPUPickState; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static GPUPickState g_pick_state = {0}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | void gpu_select_pick_begin(uint (*buffer)[4], uint bufsize, const rcti *input, char mode) | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   GPUPickState *ps = &g_pick_state; | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef DEBUG_PRINT
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   printf("%s: mode=%d, use_cache=%d, is_cache=%d\n", __func__, mode, ps->use_cache, ps->is_cached); | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   ps->bufsize = bufsize; | 
					
						
							|  |  |  |   ps->buffer = buffer; | 
					
						
							|  |  |  |   ps->mode = mode; | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const uint rect_len = (uint)(BLI_rcti_size_x(input) * BLI_rcti_size_y(input)); | 
					
						
							|  |  |  |   ps->dst.clip_rect = *input; | 
					
						
							|  |  |  |   ps->dst.rect_len = rect_len; | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* Restrict OpenGL operations for when we don't have cache */ | 
					
						
							|  |  |  |   if (ps->is_cached == false) { | 
					
						
							| 
									
										
										
										
											2020-08-16 23:20:59 +02:00
										 |  |  |     ps->write_mask = GPU_write_mask_get(); | 
					
						
							| 
									
										
										
										
											2020-08-20 16:38:34 +02:00
										 |  |  |     ps->depth_test = GPU_depth_test_get(); | 
					
						
							| 
									
										
										
										
											2020-08-16 23:20:59 +02:00
										 |  |  |     GPU_scissor_get(ps->scissor); | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* disable writing to the framebuffer */ | 
					
						
							| 
									
										
										
										
											2020-07-16 04:16:10 +02:00
										 |  |  |     GPU_color_mask(false, false, false, false); | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-20 16:57:38 +02:00
										 |  |  |     GPU_depth_mask(true); | 
					
						
							| 
									
										
										
										
											2020-05-14 13:43:09 +02:00
										 |  |  |     /* Always use #GL_LEQUAL even though GPU_SELECT_PICK_ALL always clears the buffer. This is
 | 
					
						
							|  |  |  |      * because individual objects themselves might have sections that overlap and we need these | 
					
						
							|  |  |  |      * to have the correct distance information. */ | 
					
						
							| 
									
										
										
										
											2020-08-20 16:57:38 +02:00
										 |  |  |     GPU_depth_test(GPU_DEPTH_LESS_EQUAL); | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     float viewport[4]; | 
					
						
							| 
									
										
										
										
											2020-07-17 19:03:30 +02:00
										 |  |  |     GPU_viewport_size_get_f(viewport); | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     ps->src.clip_rect = *input; | 
					
						
							|  |  |  |     ps->src.rect_len = rect_len; | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     ps->gl.clip_readpixels[0] = (int)viewport[0]; | 
					
						
							|  |  |  |     ps->gl.clip_readpixels[1] = (int)viewport[1]; | 
					
						
							|  |  |  |     ps->gl.clip_readpixels[2] = BLI_rcti_size_x(&ps->src.clip_rect); | 
					
						
							|  |  |  |     ps->gl.clip_readpixels[3] = BLI_rcti_size_y(&ps->src.clip_rect); | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-17 19:03:30 +02:00
										 |  |  |     GPU_viewport(UNPACK4(ps->gl.clip_readpixels)); | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* It's possible we don't want to clear depth buffer,
 | 
					
						
							|  |  |  |      * so existing elements are masked by current z-buffer. */ | 
					
						
							| 
									
										
										
										
											2020-08-23 11:11:27 +02:00
										 |  |  |     GPU_clear_depth(1.0f); | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* scratch buffer (read new values here) */ | 
					
						
							|  |  |  |     ps->gl.rect_depth_test = depth_buf_malloc(rect_len); | 
					
						
							|  |  |  |     ps->gl.rect_depth = depth_buf_malloc(rect_len); | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     /* set initial 'far' value */ | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:24:14 +02:00
										 |  |  |     glReadPixels(UNPACK4(ps->gl.clip_readpixels), | 
					
						
							|  |  |  |                  GL_DEPTH_COMPONENT, | 
					
						
							|  |  |  |                  GL_UNSIGNED_INT, | 
					
						
							|  |  |  |                  ps->gl.rect_depth->buf); | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     for (uint i = 0; i < rect_len; i++) { | 
					
						
							|  |  |  |       ps->gl.rect_depth->buf[i] = DEPTH_MAX; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     ps->gl.is_init = false; | 
					
						
							|  |  |  |     ps->gl.prev_id = 0; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     /* Using cache (ps->is_cached == true) */ | 
					
						
							|  |  |  |     /* src.clip_rect -> dst.clip_rect */ | 
					
						
							|  |  |  |     rect_subregion_stride_calc(&ps->src.clip_rect, &ps->dst.clip_rect, &ps->cache.sub_rect); | 
					
						
							|  |  |  |     BLI_assert(ps->gl.rect_depth == NULL); | 
					
						
							|  |  |  |     BLI_assert(ps->gl.rect_depth_test == NULL); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (mode == GPU_SELECT_PICK_ALL) { | 
					
						
							|  |  |  |     ps->all.hits = MEM_mallocN(sizeof(*ps->all.hits) * ALLOC_DEPTHS, __func__); | 
					
						
							|  |  |  |     ps->all.hits_len = 0; | 
					
						
							|  |  |  |     ps->all.hits_len_alloc = ALLOC_DEPTHS; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     /* Set to 0xff for SELECT_ID_NONE */ | 
					
						
							|  |  |  |     ps->nearest.rect_id = MEM_mallocN(sizeof(uint) * ps->dst.rect_len, __func__); | 
					
						
							|  |  |  |     memset(ps->nearest.rect_id, 0xff, sizeof(uint) * ps->dst.rect_len); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Given 2x depths, we know are different - update the depth information | 
					
						
							|  |  |  |  * use for both cached/uncached depth buffers. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static void gpu_select_load_id_pass_all(const DepthBufCache *rect_curr) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   GPUPickState *ps = &g_pick_state; | 
					
						
							|  |  |  |   const uint id = rect_curr->id; | 
					
						
							|  |  |  |   /* find the best depth for this pass and store in 'all.hits' */ | 
					
						
							|  |  |  |   depth_t depth_best = DEPTH_MAX; | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | 
 | 
					
						
							|  |  |  | #define EVAL_TEST() \
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (depth_best > *curr) { \ | 
					
						
							|  |  |  |     depth_best = *curr; \ | 
					
						
							|  |  |  |   } \ | 
					
						
							|  |  |  |   ((void)0) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (ps->is_cached == false) { | 
					
						
							|  |  |  |     const depth_t *curr = rect_curr->buf; | 
					
						
							|  |  |  |     BLI_assert(ps->src.rect_len == ps->dst.rect_len); | 
					
						
							|  |  |  |     const uint rect_len = ps->src.rect_len; | 
					
						
							|  |  |  |     for (uint i = 0; i < rect_len; i++, curr++) { | 
					
						
							|  |  |  |       EVAL_TEST(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     /* same as above but different rect sizes */ | 
					
						
							|  |  |  |     const depth_t *curr = rect_curr->buf + ps->cache.sub_rect.start; | 
					
						
							|  |  |  |     for (uint i = 0; i < ps->cache.sub_rect.span_len; i++) { | 
					
						
							|  |  |  |       const depth_t *curr_end = curr + ps->cache.sub_rect.span; | 
					
						
							|  |  |  |       for (; curr < curr_end; curr++) { | 
					
						
							|  |  |  |         EVAL_TEST(); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       curr += ps->cache.sub_rect.skip; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | 
 | 
					
						
							|  |  |  | #undef EVAL_TEST
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* ensure enough space */ | 
					
						
							|  |  |  |   if (UNLIKELY(ps->all.hits_len == ps->all.hits_len_alloc)) { | 
					
						
							|  |  |  |     ps->all.hits_len_alloc += ALLOC_DEPTHS; | 
					
						
							|  |  |  |     ps->all.hits = MEM_reallocN(ps->all.hits, ps->all.hits_len_alloc * sizeof(*ps->all.hits)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   DepthID *d = &ps->all.hits[ps->all.hits_len++]; | 
					
						
							|  |  |  |   d->id = id; | 
					
						
							|  |  |  |   d->depth = depth_best; | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | static void gpu_select_load_id_pass_nearest(const DepthBufCache *rect_prev, | 
					
						
							|  |  |  |                                             const DepthBufCache *rect_curr) | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   GPUPickState *ps = &g_pick_state; | 
					
						
							|  |  |  |   const uint id = rect_curr->id; | 
					
						
							|  |  |  |   /* keep track each pixels ID in 'nearest.rect_id' */ | 
					
						
							|  |  |  |   if (id != SELECT_ID_NONE) { | 
					
						
							|  |  |  |     uint *id_ptr = ps->nearest.rect_id; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Check against DEPTH_MAX because XRAY will clear the buffer,
 | 
					
						
							|  |  |  |      * so previously set values will become unset. | 
					
						
							|  |  |  |      * In this case just leave those id's left as-is. */ | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | #define EVAL_TEST() \
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (depth_is_filled(prev, curr)) { \ | 
					
						
							|  |  |  |     *id_ptr = id; \ | 
					
						
							|  |  |  |   } \ | 
					
						
							|  |  |  |   ((void)0) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (ps->is_cached == false) { | 
					
						
							|  |  |  |       const depth_t *prev = rect_prev->buf; | 
					
						
							|  |  |  |       const depth_t *curr = rect_curr->buf; | 
					
						
							|  |  |  |       BLI_assert(ps->src.rect_len == ps->dst.rect_len); | 
					
						
							|  |  |  |       const uint rect_len = ps->src.rect_len; | 
					
						
							|  |  |  |       for (uint i = 0; i < rect_len; i++, curr++, prev++, id_ptr++) { | 
					
						
							|  |  |  |         EVAL_TEST(); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       /* same as above but different rect sizes */ | 
					
						
							|  |  |  |       const depth_t *prev = rect_prev->buf + ps->cache.sub_rect.start; | 
					
						
							|  |  |  |       const depth_t *curr = rect_curr->buf + ps->cache.sub_rect.start; | 
					
						
							|  |  |  |       for (uint i = 0; i < ps->cache.sub_rect.span_len; i++) { | 
					
						
							|  |  |  |         const depth_t *curr_end = curr + ps->cache.sub_rect.span; | 
					
						
							|  |  |  |         for (; curr < curr_end; prev++, curr++, id_ptr++) { | 
					
						
							|  |  |  |           EVAL_TEST(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         prev += ps->cache.sub_rect.skip; | 
					
						
							|  |  |  |         curr += ps->cache.sub_rect.skip; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | 
 | 
					
						
							|  |  |  | #undef EVAL_TEST
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-27 20:37:11 +02:00
										 |  |  | bool gpu_select_pick_load_id(uint id, bool end) | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   GPUPickState *ps = &g_pick_state; | 
					
						
							| 
									
										
										
										
											2019-04-26 15:33:16 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (ps->gl.is_init) { | 
					
						
							| 
									
										
										
										
											2019-04-27 20:37:11 +02:00
										 |  |  |     if (id == ps->gl.prev_id && !end) { | 
					
						
							| 
									
										
										
										
											2019-04-26 15:33:16 +02:00
										 |  |  |       /* No need to read if we are still drawing for the same id since
 | 
					
						
							| 
									
										
										
										
											2019-04-29 14:14:14 +10:00
										 |  |  |        * all these depths will be merged / de-duplicated in the end. */ | 
					
						
							| 
									
										
										
										
											2019-04-26 15:33:16 +02:00
										 |  |  |       return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     const uint rect_len = ps->src.rect_len; | 
					
						
							|  |  |  |     glReadPixels(UNPACK4(ps->gl.clip_readpixels), | 
					
						
							|  |  |  |                  GL_DEPTH_COMPONENT, | 
					
						
							|  |  |  |                  GL_UNSIGNED_INT, | 
					
						
							|  |  |  |                  ps->gl.rect_depth_test->buf); | 
					
						
							|  |  |  |     /* perform initial check since most cases the array remains unchanged  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool do_pass = false; | 
					
						
							|  |  |  |     if (g_pick_state.mode == GPU_SELECT_PICK_ALL) { | 
					
						
							|  |  |  |       if (depth_buf_rect_depth_any(ps->gl.rect_depth_test, rect_len)) { | 
					
						
							|  |  |  |         ps->gl.rect_depth_test->id = ps->gl.prev_id; | 
					
						
							|  |  |  |         gpu_select_load_id_pass_all(ps->gl.rect_depth_test); | 
					
						
							|  |  |  |         do_pass = true; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       if (depth_buf_rect_depth_any_filled(ps->gl.rect_depth, ps->gl.rect_depth_test, rect_len)) { | 
					
						
							|  |  |  |         ps->gl.rect_depth_test->id = ps->gl.prev_id; | 
					
						
							|  |  |  |         gpu_select_load_id_pass_nearest(ps->gl.rect_depth, ps->gl.rect_depth_test); | 
					
						
							|  |  |  |         do_pass = true; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (do_pass) { | 
					
						
							|  |  |  |       /* Store depth in cache */ | 
					
						
							|  |  |  |       if (ps->use_cache) { | 
					
						
							|  |  |  |         BLI_addtail(&ps->cache.bufs, ps->gl.rect_depth); | 
					
						
							|  |  |  |         ps->gl.rect_depth = depth_buf_malloc(ps->src.rect_len); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       SWAP(DepthBufCache *, ps->gl.rect_depth, ps->gl.rect_depth_test); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (g_pick_state.mode == GPU_SELECT_PICK_ALL) { | 
					
						
							| 
									
										
										
										
											2020-08-25 12:15:15 +02:00
										 |  |  |         /* (fclem) This is to be on the safe side. I don't know if this is required. */ | 
					
						
							|  |  |  |         bool prev_depth_mask = GPU_depth_mask_get(); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         /* we want new depths every time */ | 
					
						
							| 
									
										
										
										
											2020-08-25 12:15:15 +02:00
										 |  |  |         GPU_depth_mask(true); | 
					
						
							| 
									
										
										
										
											2020-08-23 11:11:27 +02:00
										 |  |  |         GPU_clear_depth(1.0f); | 
					
						
							| 
									
										
										
										
											2020-08-25 12:15:15 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         GPU_depth_mask(prev_depth_mask); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ps->gl.is_init = true; | 
					
						
							|  |  |  |   ps->gl.prev_id = id; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-28 11:37:39 +11:00
										 |  |  | uint gpu_select_pick_end(void) | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   GPUPickState *ps = &g_pick_state; | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef DEBUG_PRINT
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   printf("%s\n", __func__); | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (ps->is_cached == false) { | 
					
						
							|  |  |  |     if (ps->gl.is_init) { | 
					
						
							|  |  |  |       /* force finishing last pass */ | 
					
						
							| 
									
										
										
										
											2019-04-27 20:37:11 +02:00
										 |  |  |       gpu_select_pick_load_id(ps->gl.prev_id, true); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-08-16 23:20:59 +02:00
										 |  |  |     GPU_write_mask(ps->write_mask); | 
					
						
							|  |  |  |     GPU_depth_test(ps->depth_test); | 
					
						
							|  |  |  |     GPU_viewport(UNPACK4(ps->viewport)); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* assign but never free directly since it may be in cache */ | 
					
						
							|  |  |  |   DepthBufCache *rect_depth_final; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Store depth in cache */ | 
					
						
							|  |  |  |   if (ps->use_cache && !ps->is_cached) { | 
					
						
							|  |  |  |     BLI_addtail(&ps->cache.bufs, ps->gl.rect_depth); | 
					
						
							|  |  |  |     ps->gl.rect_depth = NULL; | 
					
						
							|  |  |  |     rect_depth_final = ps->cache.bufs.last; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else if (ps->is_cached) { | 
					
						
							|  |  |  |     rect_depth_final = ps->cache.bufs.last; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     /* common case, no cache */ | 
					
						
							|  |  |  |     rect_depth_final = ps->gl.rect_depth; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   uint maxhits = g_pick_state.bufsize; | 
					
						
							|  |  |  |   DepthID *depth_data; | 
					
						
							|  |  |  |   uint depth_data_len = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (g_pick_state.mode == GPU_SELECT_PICK_ALL) { | 
					
						
							|  |  |  |     depth_data = ps->all.hits; | 
					
						
							|  |  |  |     depth_data_len = ps->all.hits_len; | 
					
						
							|  |  |  |     /* move ownership */ | 
					
						
							|  |  |  |     ps->all.hits = NULL; | 
					
						
							|  |  |  |     ps->all.hits_len = 0; | 
					
						
							|  |  |  |     ps->all.hits_len_alloc = 0; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     /* GPU_SELECT_PICK_NEAREST */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Over alloc (unlikely we have as many depths as pixels) */ | 
					
						
							|  |  |  |     uint depth_data_len_first_pass = 0; | 
					
						
							|  |  |  |     depth_data = MEM_mallocN(ps->dst.rect_len * sizeof(*depth_data), __func__); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Partially de-duplicating copy,
 | 
					
						
							|  |  |  |      * when contiguous ID's are found - update their closest depth. | 
					
						
							|  |  |  |      * This isn't essential but means there is less data to sort. */ | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | 
 | 
					
						
							|  |  |  | #define EVAL_TEST(i_src, i_dst) \
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   { \ | 
					
						
							|  |  |  |     const uint id = ps->nearest.rect_id[i_dst]; \ | 
					
						
							|  |  |  |     if (id != SELECT_ID_NONE) { \ | 
					
						
							|  |  |  |       const depth_t depth = rect_depth_final->buf[i_src]; \ | 
					
						
							|  |  |  |       if (depth_last == NULL || depth_last->id != id) { \ | 
					
						
							|  |  |  |         DepthID *d = &depth_data[depth_data_len_first_pass++]; \ | 
					
						
							|  |  |  |         d->id = id; \ | 
					
						
							|  |  |  |         d->depth = depth; \ | 
					
						
							|  |  |  |       } \ | 
					
						
							|  |  |  |       else if (depth_last->depth > depth) { \ | 
					
						
							|  |  |  |         depth_last->depth = depth; \ | 
					
						
							|  |  |  |       } \ | 
					
						
							|  |  |  |     } \ | 
					
						
							|  |  |  |   } \ | 
					
						
							|  |  |  |   ((void)0) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       DepthID *depth_last = NULL; | 
					
						
							|  |  |  |       if (ps->is_cached == false) { | 
					
						
							|  |  |  |         for (uint i = 0; i < ps->src.rect_len; i++) { | 
					
						
							|  |  |  |           EVAL_TEST(i, i); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         /* same as above but different rect sizes */ | 
					
						
							|  |  |  |         uint i_src = ps->cache.sub_rect.start, i_dst = 0; | 
					
						
							|  |  |  |         for (uint j = 0; j < ps->cache.sub_rect.span_len; j++) { | 
					
						
							|  |  |  |           const uint i_src_end = i_src + ps->cache.sub_rect.span; | 
					
						
							|  |  |  |           for (; i_src < i_src_end; i_src++, i_dst++) { | 
					
						
							|  |  |  |             EVAL_TEST(i_src, i_dst); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           i_src += ps->cache.sub_rect.skip; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | 
 | 
					
						
							|  |  |  | #undef EVAL_TEST
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     qsort(depth_data, depth_data_len_first_pass, sizeof(DepthID), depth_id_cmp); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Sort by ID's then keep the best depth for each ID */ | 
					
						
							|  |  |  |     depth_data_len = 0; | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       DepthID *depth_last = NULL; | 
					
						
							|  |  |  |       for (uint i = 0; i < depth_data_len_first_pass; i++) { | 
					
						
							|  |  |  |         if (depth_last == NULL || depth_last->id != depth_data[i].id) { | 
					
						
							|  |  |  |           depth_last = &depth_data[depth_data_len++]; | 
					
						
							|  |  |  |           *depth_last = depth_data[i]; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else if (depth_last->depth > depth_data[i].depth) { | 
					
						
							|  |  |  |           depth_last->depth = depth_data[i].depth; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Finally sort each unique (id, depth) pair by depth
 | 
					
						
							|  |  |  |    * so the final hit-list is sorted by depth (nearest first) */ | 
					
						
							|  |  |  |   uint hits = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (depth_data_len > maxhits) { | 
					
						
							|  |  |  |     hits = (uint)-1; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     /* leave sorting up to the caller */ | 
					
						
							|  |  |  |     qsort(depth_data, depth_data_len, sizeof(DepthID), depth_cmp); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (uint i = 0; i < depth_data_len; i++) { | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | #ifdef DEBUG_PRINT
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       printf("  hit: %u: depth %u\n", depth_data[i].id, depth_data[i].depth); | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       /* first 3 are dummy values */ | 
					
						
							|  |  |  |       g_pick_state.buffer[hits][0] = 1; | 
					
						
							|  |  |  |       g_pick_state.buffer[hits][1] = 0x0; /* depth_data[i].depth; */ /* unused */ | 
					
						
							|  |  |  |       g_pick_state.buffer[hits][2] = 0x0; /* z-far is currently never used. */ | 
					
						
							|  |  |  |       g_pick_state.buffer[hits][3] = depth_data[i].id; | 
					
						
							|  |  |  |       hits++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     BLI_assert(hits < maxhits); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   MEM_freeN(depth_data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   MEM_SAFE_FREE(ps->gl.rect_depth); | 
					
						
							|  |  |  |   MEM_SAFE_FREE(ps->gl.rect_depth_test); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (g_pick_state.mode == GPU_SELECT_PICK_ALL) { | 
					
						
							|  |  |  |     /* 'hits' already freed as 'depth_data' */ | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     MEM_freeN(ps->nearest.rect_id); | 
					
						
							|  |  |  |     ps->nearest.rect_id = NULL; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (ps->use_cache) { | 
					
						
							|  |  |  |     ps->is_cached = true; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return hits; | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ----------------------------------------------------------------------------
 | 
					
						
							|  |  |  |  * Caching | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Support multiple begin/end's reusing depth buffers. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void gpu_select_pick_cache_begin(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BLI_assert(g_pick_state.use_cache == false); | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | #ifdef DEBUG_PRINT
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   printf("%s\n", __func__); | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   g_pick_state.use_cache = true; | 
					
						
							|  |  |  |   g_pick_state.is_cached = false; | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void gpu_select_pick_cache_end(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #ifdef DEBUG_PRINT
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   printf("%s: with %d buffers\n", __func__, BLI_listbase_count(&g_pick_state.cache.bufs)); | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   g_pick_state.use_cache = false; | 
					
						
							|  |  |  |   g_pick_state.is_cached = false; | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BLI_freelistN(&g_pick_state.cache.bufs); | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* is drawing needed? */ | 
					
						
							|  |  |  | bool gpu_select_pick_is_cached(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return g_pick_state.is_cached; | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void gpu_select_pick_cache_load_id(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BLI_assert(g_pick_state.is_cached == true); | 
					
						
							|  |  |  |   GPUPickState *ps = &g_pick_state; | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | #ifdef DEBUG_PRINT
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   printf("%s (building depth from cache)\n", __func__); | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-04-03 19:15:01 +02:00
										 |  |  |   LISTBASE_FOREACH (DepthBufCache *, rect_depth, &ps->cache.bufs) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     if (rect_depth->next != NULL) { | 
					
						
							|  |  |  |       /* we know the buffers differ, but this sub-region may not.
 | 
					
						
							|  |  |  |        * double check before adding an id-pass */ | 
					
						
							|  |  |  |       if (g_pick_state.mode == GPU_SELECT_PICK_ALL) { | 
					
						
							|  |  |  |         if (depth_buf_subrect_depth_any(rect_depth->next, &ps->cache.sub_rect)) { | 
					
						
							|  |  |  |           gpu_select_load_id_pass_all(rect_depth->next); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         if (depth_buf_subrect_depth_any_filled( | 
					
						
							|  |  |  |                 rect_depth, rect_depth->next, &ps->cache.sub_rect)) { | 
					
						
							|  |  |  |           gpu_select_load_id_pass_nearest(rect_depth, rect_depth->next); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-03-09 05:17:55 +11:00
										 |  |  | } |