| 
									
										
										
										
											2011-02-23 10:52:22 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  |  * ***** BEGIN GPL LICENSE BLOCK ***** | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 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, | 
					
						
							| 
									
										
										
										
											2010-02-12 13:34:04 +00:00
										 |  |  |  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * ***** END GPL LICENSE BLOCK ***** | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-20 13:29:58 +00:00
										 |  |  | /** \file blender/blenkernel/intern/pbvh.c
 | 
					
						
							| 
									
										
										
										
											2011-02-27 20:37:56 +00:00
										 |  |  |  *  \ingroup bli | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | #include "DNA_meshdata_types.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-16 05:46:10 +00:00
										 |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-22 23:57:31 +00:00
										 |  |  | #include "BLI_bitmap.h"
 | 
					
						
							| 
									
										
										
										
											2009-11-11 10:44:46 +00:00
										 |  |  | #include "BLI_math.h"
 | 
					
						
							| 
									
										
										
										
											2011-01-07 18:36:47 +00:00
										 |  |  | #include "BLI_utildefines.h"
 | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | #include "BLI_ghash.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-15 15:59:25 +00:00
										 |  |  | #include "BKE_pbvh.h"
 | 
					
						
							| 
									
										
										
										
											2012-05-10 20:33:09 +00:00
										 |  |  | #include "BKE_ccg.h"
 | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | #include "BKE_DerivedMesh.h"
 | 
					
						
							| 
									
										
										
										
											2012-05-05 21:28:12 +00:00
										 |  |  | #include "BKE_mesh.h" /* for BKE_mesh_calc_normals */
 | 
					
						
							|  |  |  | #include "BKE_global.h" /* for BKE_mesh_calc_normals */
 | 
					
						
							| 
									
										
										
										
											2012-03-14 06:32:25 +00:00
										 |  |  | #include "BKE_paint.h"
 | 
					
						
							|  |  |  | #include "BKE_subsurf.h"
 | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-14 10:46:12 +00:00
										 |  |  | #include "GPU_buffers.h"
 | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-30 02:16:22 +00:00
										 |  |  | #include "bmesh.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:25:36 +00:00
										 |  |  | #include "pbvh_intern.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | #define LEAF_LIMIT 10000
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //#define PERFCNTRS
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | #define STACK_FIXED_DEPTH   100
 | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | typedef struct PBVHStack { | 
					
						
							|  |  |  | 	PBVHNode *node; | 
					
						
							|  |  |  | 	int revisiting; | 
					
						
							|  |  |  | } PBVHStack; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct PBVHIter { | 
					
						
							|  |  |  | 	PBVH *bvh; | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | 	BKE_pbvh_SearchCallback scb; | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 	void *search_data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	PBVHStack *stack; | 
					
						
							|  |  |  | 	int stacksize; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	PBVHStack stackfixed[STACK_FIXED_DEPTH]; | 
					
						
							|  |  |  | 	int stackspace; | 
					
						
							|  |  |  | } PBVHIter; | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:28:36 +00:00
										 |  |  | void BB_reset(BB *bb) | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	bb->bmin[0] = bb->bmin[1] = bb->bmin[2] = FLT_MAX; | 
					
						
							|  |  |  | 	bb->bmax[0] = bb->bmax[1] = bb->bmax[2] = -FLT_MAX; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Expand the bounding box to include a new coordinate */ | 
					
						
							| 
									
										
										
										
											2012-12-30 18:28:36 +00:00
										 |  |  | void BB_expand(BB *bb, const float co[3]) | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 	int i; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	for (i = 0; i < 3; ++i) { | 
					
						
							| 
									
										
										
										
											2012-10-23 13:28:22 +00:00
										 |  |  | 		bb->bmin[i] = min_ff(bb->bmin[i], co[i]); | 
					
						
							|  |  |  | 		bb->bmax[i] = max_ff(bb->bmax[i], co[i]); | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Expand the bounding box to include another bounding box */ | 
					
						
							| 
									
										
										
										
											2012-12-30 18:28:36 +00:00
										 |  |  | void BB_expand_with_bb(BB *bb, BB *bb2) | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	int i; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	for (i = 0; i < 3; ++i) { | 
					
						
							| 
									
										
										
										
											2012-10-23 13:28:22 +00:00
										 |  |  | 		bb->bmin[i] = min_ff(bb->bmin[i], bb2->bmin[i]); | 
					
						
							|  |  |  | 		bb->bmax[i] = max_ff(bb->bmax[i], bb2->bmax[i]); | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Return 0, 1, or 2 to indicate the widest axis of the bounding box */ | 
					
						
							| 
									
										
										
										
											2012-12-30 18:28:36 +00:00
										 |  |  | int BB_widest_axis(const BB *bb) | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	float dim[3]; | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	for (i = 0; i < 3; ++i) | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 		dim[i] = bb->bmax[i] - bb->bmin[i]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (dim[0] > dim[1]) { | 
					
						
							|  |  |  | 		if (dim[0] > dim[2]) | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 			return 0; | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			return 2; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (dim[1] > dim[2]) | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 			return 1; | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			return 2; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:28:36 +00:00
										 |  |  | void BBC_update_centroid(BBC *bbc) | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	int i; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	for (i = 0; i < 3; ++i) | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 		bbc->bcentroid[i] = (bbc->bmin[i] + bbc->bmax[i]) * 0.5f; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Not recursive */ | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | static void update_node_vb(PBVH *bvh, PBVHNode *node) | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 	BB vb; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	BB_reset(&vb); | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (node->flag & PBVH_Leaf) { | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 		PBVHVertexIter vd; | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | 		BKE_pbvh_vertex_iter_begin(bvh, node, vd, PBVH_ITER_ALL) | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 			BB_expand(&vb, vd.co); | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | 		BKE_pbvh_vertex_iter_end; | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 		BB_expand_with_bb(&vb, | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		                  &bvh->nodes[node->children_offset].vb); | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 		BB_expand_with_bb(&vb, | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		                  &bvh->nodes[node->children_offset + 1].vb); | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	node->vb = vb; | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | //void BKE_pbvh_node_BB_reset(PBVHNode *node)
 | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | //{
 | 
					
						
							|  |  |  | //	BB_reset(&node->vb);
 | 
					
						
							|  |  |  | //}
 | 
					
						
							|  |  |  | //
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | //void BKE_pbvh_node_BB_expand(PBVHNode *node, float co[3])
 | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | //{
 | 
					
						
							|  |  |  | //	BB_expand(&node->vb, co);
 | 
					
						
							|  |  |  | //}
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-06 02:40:08 +00:00
										 |  |  | static int face_materials_match(const MFace *f1, const MFace *f2) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return ((f1->flag & ME_SMOOTH) == (f2->flag & ME_SMOOTH) && | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	        (f1->mat_nr == f2->mat_nr)); | 
					
						
							| 
									
										
										
										
											2012-03-06 02:40:08 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int grid_materials_match(const DMFlagMat *f1, const DMFlagMat *f2) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return ((f1->flag & ME_SMOOTH) == (f2->flag & ME_SMOOTH) && | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	        (f1->mat_nr == f2->mat_nr)); | 
					
						
							| 
									
										
										
										
											2012-03-06 02:40:08 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | /* Adapted from BLI_kdopbvh.c */ | 
					
						
							|  |  |  | /* Returns the index of the first element on the right of the partition */ | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | static int partition_indices(int *prim_indices, int lo, int hi, int axis, | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  |                              float mid, BBC *prim_bbc) | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	int i = lo, j = hi; | 
					
						
							|  |  |  | 	for (;; ) { | 
					
						
							|  |  |  | 		for (; prim_bbc[prim_indices[i]].bcentroid[axis] < mid; i++) ; | 
					
						
							|  |  |  | 		for (; mid < prim_bbc[prim_indices[j]].bcentroid[axis]; j--) ; | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (!(i < j)) | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 			return i; | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 		SWAP(int, prim_indices[i], prim_indices[j]); | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 		i++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-06 02:40:08 +00:00
										 |  |  | /* Returns the index of the first element on the right of the partition */ | 
					
						
							|  |  |  | static int partition_indices_material(PBVH *bvh, int lo, int hi) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	const MFace *faces = bvh->faces; | 
					
						
							|  |  |  | 	const DMFlagMat *flagmats = bvh->grid_flag_mats; | 
					
						
							|  |  |  | 	const int *indices = bvh->prim_indices; | 
					
						
							|  |  |  | 	const void *first; | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	int i = lo, j = hi; | 
					
						
							| 
									
										
										
										
											2012-03-06 02:40:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (bvh->faces) | 
					
						
							| 
									
										
										
										
											2012-03-06 02:40:08 +00:00
										 |  |  | 		first = &faces[bvh->prim_indices[lo]]; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		first = &flagmats[bvh->prim_indices[lo]]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	for (;; ) { | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (bvh->faces) { | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 			for (; face_materials_match(first, &faces[indices[i]]); i++) ; | 
					
						
							|  |  |  | 			for (; !face_materials_match(first, &faces[indices[j]]); j--) ; | 
					
						
							| 
									
										
										
										
											2012-03-06 02:40:08 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 			for (; grid_materials_match(first, &flagmats[indices[i]]); i++) ; | 
					
						
							|  |  |  | 			for (; !grid_materials_match(first, &flagmats[indices[j]]); j--) ; | 
					
						
							| 
									
										
										
										
											2012-03-06 02:40:08 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (!(i < j)) | 
					
						
							| 
									
										
										
										
											2012-03-06 02:40:08 +00:00
										 |  |  | 			return i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		SWAP(int, bvh->prim_indices[i], bvh->prim_indices[j]); | 
					
						
							|  |  |  | 		i++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:28:36 +00:00
										 |  |  | void pbvh_grow_nodes(PBVH *bvh, int totnode) | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (totnode > bvh->node_mem_count) { | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 		PBVHNode *prev = bvh->nodes; | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 		bvh->node_mem_count *= 1.33; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (bvh->node_mem_count < totnode) | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 			bvh->node_mem_count = totnode; | 
					
						
							| 
									
										
										
										
											2013-08-21 15:21:56 +00:00
										 |  |  | 		bvh->nodes = MEM_mallocN(sizeof(PBVHNode) * bvh->node_mem_count, | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		                         "bvh nodes"); | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 		memcpy(bvh->nodes, prev, bvh->totnode * sizeof(PBVHNode)); | 
					
						
							| 
									
										
										
										
											2013-08-21 15:21:56 +00:00
										 |  |  | 		memset(bvh->nodes + bvh->totnode, 0, (bvh->node_mem_count - bvh->totnode) * sizeof(PBVHNode)); | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 		MEM_freeN(prev); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	bvh->totnode = totnode; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Add a vertex to the map, with a positive value for unique vertices and
 | 
					
						
							| 
									
										
										
										
											2012-03-03 20:19:11 +00:00
										 |  |  |  * a negative value for additional vertices */ | 
					
						
							| 
									
										
										
										
											2009-11-06 16:46:35 +00:00
										 |  |  | static int map_insert_vert(PBVH *bvh, GHash *map, | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  |                            unsigned int *face_verts, | 
					
						
							|  |  |  |                            unsigned int *uniq_verts, int vertex) | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-08-24 15:31:47 +00:00
										 |  |  | 	void *key, **value_p; | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-24 15:31:47 +00:00
										 |  |  | 	key = SET_INT_IN_POINTER(vertex); | 
					
						
							|  |  |  | 	value_p = BLI_ghash_lookup_p(map, key); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (value_p == NULL) { | 
					
						
							|  |  |  | 		void *value; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (BLI_BITMAP_GET(bvh->vert_bitmap, vertex)) { | 
					
						
							| 
									
										
										
										
											2011-09-25 12:33:51 +00:00
										 |  |  | 			value = SET_INT_IN_POINTER(~(*face_verts)); | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 			++(*face_verts); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							| 
									
										
										
										
											2012-02-22 23:57:31 +00:00
										 |  |  | 			BLI_BITMAP_SET(bvh->vert_bitmap, vertex); | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 			value = SET_INT_IN_POINTER(*uniq_verts); | 
					
						
							|  |  |  | 			++(*uniq_verts); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		BLI_ghash_insert(map, key, value); | 
					
						
							| 
									
										
										
										
											2009-11-06 16:46:35 +00:00
										 |  |  | 		return GET_INT_FROM_POINTER(value); | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-08-24 15:31:47 +00:00
										 |  |  | 	else { | 
					
						
							|  |  |  | 		return GET_INT_FROM_POINTER(*value_p); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Find vertices used by the faces in this node and update the draw buffers */ | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | static void build_mesh_leaf_node(PBVH *bvh, PBVHNode *node) | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	GHashIterator *iter; | 
					
						
							|  |  |  | 	GHash *map; | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 	int i, j, totface; | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	node->uniq_verts = node->face_verts = 0; | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	totface = node->totprim; | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-24 17:33:47 +00:00
										 |  |  | 	/* reserve size is rough guess */ | 
					
						
							|  |  |  | 	map = BLI_ghash_int_new_ex("build_mesh_leaf_node gh", 2 * totface); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	node->face_vert_indices = MEM_callocN(sizeof(int) * 4 * totface, | 
					
						
							|  |  |  | 	                                      "bvh node face vert indices"); | 
					
						
							| 
									
										
										
										
											2009-11-06 16:46:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	for (i = 0; i < totface; ++i) { | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 		MFace *f = bvh->faces + node->prim_indices[i]; | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 		int sides = f->v4 ? 4 : 3; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		for (j = 0; j < sides; ++j) { | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 			node->face_vert_indices[i][j] = | 
					
						
							|  |  |  | 			        map_insert_vert(bvh, map, &node->face_verts, | 
					
						
							|  |  |  | 			                        &node->uniq_verts, (&f->v1)[j]); | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	node->vert_indices = MEM_callocN(sizeof(int) * | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	                                 (node->uniq_verts + node->face_verts), | 
					
						
							|  |  |  | 	                                 "bvh node vert indices"); | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* Build the vertex list, unique verts first */ | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	for (iter = BLI_ghashIterator_new(map), i = 0; | 
					
						
							| 
									
										
										
										
											2013-05-08 12:58:11 +00:00
										 |  |  | 	     BLI_ghashIterator_done(iter) == false; | 
					
						
							| 
									
										
										
										
											2012-05-20 19:49:27 +00:00
										 |  |  | 	     BLI_ghashIterator_step(iter), ++i) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 		void *value = BLI_ghashIterator_getValue(iter); | 
					
						
							|  |  |  | 		int ndx = GET_INT_FROM_POINTER(value); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (ndx < 0) | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 			ndx = -ndx + node->uniq_verts - 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		node->vert_indices[ndx] = | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		    GET_INT_FROM_POINTER(BLI_ghashIterator_getKey(iter)); | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-01 11:36:22 +00:00
										 |  |  | 	BLI_ghashIterator_free(iter); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	for (i = 0; i < totface; ++i) { | 
					
						
							| 
									
										
										
										
											2011-07-31 02:34:53 +00:00
										 |  |  | 		MFace *f = bvh->faces + node->prim_indices[i]; | 
					
						
							|  |  |  | 		int sides = f->v4 ? 4 : 3; | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		for (j = 0; j < sides; ++j) { | 
					
						
							|  |  |  | 			if (node->face_vert_indices[i][j] < 0) | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 				node->face_vert_indices[i][j] = | 
					
						
							|  |  |  | 				        -node->face_vert_indices[i][j] + | 
					
						
							|  |  |  | 				        node->uniq_verts - 1; | 
					
						
							| 
									
										
										
										
											2011-07-31 02:34:53 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-11-06 16:46:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-05 04:58:27 +02:00
										 |  |  | 	BKE_pbvh_node_mark_rebuild_draw(node); | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	BLI_ghash_free(map, NULL, NULL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-05 22:24:49 +00:00
										 |  |  | static void update_vb(PBVH *bvh, PBVHNode *node, BBC *prim_bbc, | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  |                       int offset, int count) | 
					
						
							| 
									
										
										
										
											2012-03-05 21:55:53 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	int i; | 
					
						
							| 
									
										
										
										
											2012-03-05 22:24:49 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	BB_reset(&node->vb); | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	for (i = offset + count - 1; i >= offset; --i) { | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		BB_expand_with_bb(&node->vb, (BB *)(&prim_bbc[bvh->prim_indices[i]])); | 
					
						
							| 
									
										
										
										
											2012-03-05 22:24:49 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	node->orig_vb = node->vb; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2012-03-05 21:55:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-05 22:24:49 +00:00
										 |  |  | static void build_leaf(PBVH *bvh, int node_index, BBC *prim_bbc, | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  |                        int offset, int count) | 
					
						
							| 
									
										
										
										
											2012-03-05 22:24:49 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-05 21:55:53 +00:00
										 |  |  | 	bvh->nodes[node_index].flag |= PBVH_Leaf; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	bvh->nodes[node_index].prim_indices = bvh->prim_indices + offset; | 
					
						
							|  |  |  | 	bvh->nodes[node_index].totprim = count; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Still need vb for searches */ | 
					
						
							| 
									
										
										
										
											2012-03-05 22:24:49 +00:00
										 |  |  | 	update_vb(bvh, &bvh->nodes[node_index], prim_bbc, offset, count); | 
					
						
							| 
									
										
										
										
											2012-03-05 21:55:53 +00:00
										 |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (bvh->faces) | 
					
						
							| 
									
										
										
										
											2012-03-05 21:55:53 +00:00
										 |  |  | 		build_mesh_leaf_node(bvh, bvh->nodes + node_index); | 
					
						
							|  |  |  | 	else | 
					
						
							| 
									
										
										
										
											2014-01-09 04:28:17 +02:00
										 |  |  | 		BKE_pbvh_node_mark_rebuild_draw(bvh->nodes + node_index); | 
					
						
							| 
									
										
										
										
											2012-03-05 21:55:53 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-06 02:40:08 +00:00
										 |  |  | /* Return zero if all primitives in the node can be drawn with the
 | 
					
						
							| 
									
										
										
										
											2012-03-09 18:28:30 +00:00
										 |  |  |  * same material (including flat/smooth shading), non-zerootherwise */ | 
					
						
							| 
									
										
										
										
											2012-09-15 01:52:28 +00:00
										 |  |  | static int leaf_needs_material_split(PBVH *bvh, int offset, int count) | 
					
						
							| 
									
										
										
										
											2012-03-06 02:40:08 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	int i, prim; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (count <= 1) | 
					
						
							| 
									
										
										
										
											2012-03-06 02:40:08 +00:00
										 |  |  | 		return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (bvh->faces) { | 
					
						
							| 
									
										
										
										
											2012-03-06 02:40:08 +00:00
										 |  |  | 		const MFace *first = &bvh->faces[bvh->prim_indices[offset]]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		for (i = offset + count - 1; i > offset; --i) { | 
					
						
							| 
									
										
										
										
											2012-03-06 02:40:08 +00:00
										 |  |  | 			prim = bvh->prim_indices[i]; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 			if (!face_materials_match(first, &bvh->faces[prim])) | 
					
						
							| 
									
										
										
										
											2012-03-06 02:40:08 +00:00
										 |  |  | 				return 1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		const DMFlagMat *first = &bvh->grid_flag_mats[bvh->prim_indices[offset]]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		for (i = offset + count - 1; i > offset; --i) { | 
					
						
							| 
									
										
										
										
											2012-03-06 02:40:08 +00:00
										 |  |  | 			prim = bvh->prim_indices[i]; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 			if (!grid_materials_match(first, &bvh->grid_flag_mats[prim])) | 
					
						
							| 
									
										
										
										
											2012-03-06 02:40:08 +00:00
										 |  |  | 				return 1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | /* Recursively build a node in the tree
 | 
					
						
							| 
									
										
										
										
											2012-03-03 20:19:11 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * vb is the voxel box around all of the primitives contained in | 
					
						
							|  |  |  |  * this node. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * cb is the bounding box around all the centroids of the primitives | 
					
						
							|  |  |  |  * contained in this node | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * offset and start indicate a range in the array of primitive indices | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-14 17:55:27 +00:00
										 |  |  | static void build_sub(PBVH *bvh, int node_index, BB *cb, BBC *prim_bbc, | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  |                       int offset, int count) | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-06 02:40:08 +00:00
										 |  |  | 	int i, axis, end, below_leaf_limit; | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 	BB cb_backing; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Decide whether this is a leaf or not */ | 
					
						
							| 
									
										
										
										
											2012-03-06 02:40:08 +00:00
										 |  |  | 	below_leaf_limit = count <= bvh->leaf_limit; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (below_leaf_limit) { | 
					
						
							|  |  |  | 		if (!leaf_needs_material_split(bvh, offset, count)) { | 
					
						
							| 
									
										
										
										
											2012-03-06 02:40:08 +00:00
										 |  |  | 			build_leaf(bvh, node_index, prim_bbc, offset, count); | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-03-05 21:55:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-05 22:12:50 +00:00
										 |  |  | 	/* Add two child nodes */ | 
					
						
							| 
									
										
										
										
											2012-03-05 21:55:53 +00:00
										 |  |  | 	bvh->nodes[node_index].children_offset = bvh->totnode; | 
					
						
							| 
									
										
										
										
											2012-12-30 18:28:36 +00:00
										 |  |  | 	pbvh_grow_nodes(bvh, bvh->totnode + 2); | 
					
						
							| 
									
										
										
										
											2012-03-05 21:55:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-05 22:12:50 +00:00
										 |  |  | 	/* Update parent node bounding box */ | 
					
						
							| 
									
										
										
										
											2012-03-05 22:24:49 +00:00
										 |  |  | 	update_vb(bvh, &bvh->nodes[node_index], prim_bbc, offset, count); | 
					
						
							| 
									
										
										
										
											2012-03-05 22:12:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (!below_leaf_limit) { | 
					
						
							| 
									
										
										
										
											2012-03-06 02:40:08 +00:00
										 |  |  | 		/* Find axis with widest range of primitive centroids */ | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (!cb) { | 
					
						
							| 
									
										
										
										
											2012-03-06 02:40:08 +00:00
										 |  |  | 			cb = &cb_backing; | 
					
						
							|  |  |  | 			BB_reset(cb); | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 			for (i = offset + count - 1; i >= offset; --i) | 
					
						
							| 
									
										
										
										
											2012-03-06 02:40:08 +00:00
										 |  |  | 				BB_expand(cb, prim_bbc[bvh->prim_indices[i]].bcentroid); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		axis = BB_widest_axis(cb); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* Partition primitives along that axis */ | 
					
						
							|  |  |  | 		end = partition_indices(bvh->prim_indices, | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		                        offset, offset + count - 1, | 
					
						
							|  |  |  | 		                        axis, | 
					
						
							|  |  |  | 		                        (cb->bmax[axis] + cb->bmin[axis]) * 0.5f, | 
					
						
							|  |  |  | 		                        prim_bbc); | 
					
						
							| 
									
										
										
										
											2012-03-06 02:40:08 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		/* Partition primitives by material */ | 
					
						
							|  |  |  | 		end = partition_indices_material(bvh, offset, offset + count - 1); | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-05 22:12:50 +00:00
										 |  |  | 	/* Build children */ | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 	build_sub(bvh, bvh->nodes[node_index].children_offset, NULL, | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	          prim_bbc, offset, end - offset); | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 	build_sub(bvh, bvh->nodes[node_index].children_offset + 1, NULL, | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	          prim_bbc, end, offset + count - end); | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | static void pbvh_build(PBVH *bvh, BB *cb, BBC *prim_bbc, int totprim) | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 	int i; | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (totprim != bvh->totprim) { | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 		bvh->totprim = totprim; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (bvh->nodes) MEM_freeN(bvh->nodes); | 
					
						
							|  |  |  | 		if (bvh->prim_indices) MEM_freeN(bvh->prim_indices); | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 		bvh->prim_indices = MEM_callocN(sizeof(int) * totprim, | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		                                "bvh prim indices"); | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		for (i = 0; i < totprim; ++i) | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 			bvh->prim_indices[i] = i; | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 		bvh->totnode = 0; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (bvh->node_mem_count < 100) { | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 			bvh->node_mem_count = 100; | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 			bvh->nodes = MEM_callocN(sizeof(PBVHNode) * | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 			                         bvh->node_mem_count, | 
					
						
							|  |  |  | 			                         "bvh initial nodes"); | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 	bvh->totnode = 1; | 
					
						
							|  |  |  | 	build_sub(bvh, 0, cb, prim_bbc, 0, totprim); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Do a full rebuild with on Mesh data structure */ | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | void BKE_pbvh_build_mesh(PBVH *bvh, MFace *faces, MVert *verts, int totface, int totvert, struct CustomData *vdata) | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	BBC *prim_bbc = NULL; | 
					
						
							|  |  |  | 	BB cb; | 
					
						
							|  |  |  | 	int i, j; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-12 23:03:43 +00:00
										 |  |  | 	bvh->type = PBVH_FACES; | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 	bvh->faces = faces; | 
					
						
							|  |  |  | 	bvh->verts = verts; | 
					
						
							| 
									
										
										
										
											2012-02-22 23:57:31 +00:00
										 |  |  | 	bvh->vert_bitmap = BLI_BITMAP_NEW(totvert, "bvh->vert_bitmap"); | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 	bvh->totvert = totvert; | 
					
						
							|  |  |  | 	bvh->leaf_limit = LEAF_LIMIT; | 
					
						
							| 
									
										
										
										
											2012-05-10 20:33:36 +00:00
										 |  |  | 	bvh->vdata = vdata; | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	BB_reset(&cb); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* For each face, store the AABB and the AABB centroid */ | 
					
						
							|  |  |  | 	prim_bbc = MEM_mallocN(sizeof(BBC) * totface, "prim_bbc"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	for (i = 0; i < totface; ++i) { | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 		MFace *f = faces + i; | 
					
						
							|  |  |  | 		const int sides = f->v4 ? 4 : 3; | 
					
						
							|  |  |  | 		BBC *bbc = prim_bbc + i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		BB_reset((BB *)bbc); | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		for (j = 0; j < sides; ++j) | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 			BB_expand((BB *)bbc, verts[(&f->v1)[j]].co); | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		BBC_update_centroid(bbc); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		BB_expand(&cb, bbc->bcentroid); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (totface) | 
					
						
							| 
									
										
										
										
											2009-12-18 17:15:58 +00:00
										 |  |  | 		pbvh_build(bvh, &cb, prim_bbc, totface); | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	MEM_freeN(prim_bbc); | 
					
						
							|  |  |  | 	MEM_freeN(bvh->vert_bitmap); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | /* Do a full rebuild with on Grids data structure */ | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | void BKE_pbvh_build_grids(PBVH *bvh, CCGElem **grids, DMGridAdjacency *gridadj, | 
					
						
							| 
									
										
										
										
											2013-07-22 23:20:48 +00:00
										 |  |  |                           int totgrid, CCGKey *key, void **gridfaces, DMFlagMat *flagmats, BLI_bitmap **grid_hidden) | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	BBC *prim_bbc = NULL; | 
					
						
							|  |  |  | 	BB cb; | 
					
						
							| 
									
										
										
										
											2012-05-10 20:33:09 +00:00
										 |  |  | 	int gridsize = key->grid_size; | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 	int i, j; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-12 23:03:43 +00:00
										 |  |  | 	bvh->type = PBVH_GRIDS; | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	bvh->grids = grids; | 
					
						
							|  |  |  | 	bvh->gridadj = gridadj; | 
					
						
							|  |  |  | 	bvh->gridfaces = gridfaces; | 
					
						
							|  |  |  | 	bvh->grid_flag_mats = flagmats; | 
					
						
							|  |  |  | 	bvh->totgrid = totgrid; | 
					
						
							| 
									
										
										
										
											2012-05-10 20:33:09 +00:00
										 |  |  | 	bvh->gridkey = *key; | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	bvh->grid_hidden = grid_hidden; | 
					
						
							| 
									
										
										
										
											2012-10-23 13:28:22 +00:00
										 |  |  | 	bvh->leaf_limit = max_ii(LEAF_LIMIT / ((gridsize - 1) * (gridsize - 1)), 1); | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	BB_reset(&cb); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* For each grid, store the AABB and the AABB centroid */ | 
					
						
							|  |  |  | 	prim_bbc = MEM_mallocN(sizeof(BBC) * totgrid, "prim_bbc"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	for (i = 0; i < totgrid; ++i) { | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		CCGElem *grid = grids[i]; | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 		BBC *bbc = prim_bbc + i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		BB_reset((BB *)bbc); | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		for (j = 0; j < gridsize * gridsize; ++j) | 
					
						
							|  |  |  | 			BB_expand((BB *)bbc, CCG_elem_offset_co(key, grid, j)); | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		BBC_update_centroid(bbc); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		BB_expand(&cb, bbc->bcentroid); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (totgrid) | 
					
						
							| 
									
										
										
										
											2009-12-18 17:15:58 +00:00
										 |  |  | 		pbvh_build(bvh, &cb, prim_bbc, totgrid); | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	MEM_freeN(prim_bbc); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | PBVH *BKE_pbvh_new(void) | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	PBVH *bvh = MEM_callocN(sizeof(PBVH), "pbvh"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return bvh; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | void BKE_pbvh_free(PBVH *bvh) | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 	PBVHNode *node; | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 	int i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	for (i = 0; i < bvh->totnode; ++i) { | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		node = &bvh->nodes[i]; | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (node->flag & PBVH_Leaf) { | 
					
						
							|  |  |  | 			if (node->draw_buffers) | 
					
						
							| 
									
										
										
										
											2014-01-02 22:22:36 +02:00
										 |  |  | 				GPU_free_pbvh_buffers(node->draw_buffers); | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 			if (node->vert_indices) | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 				MEM_freeN(node->vert_indices); | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 			if (node->face_vert_indices) | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 				MEM_freeN(node->face_vert_indices); | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | 			BKE_pbvh_node_layer_disp_free(node); | 
					
						
							| 
									
										
										
										
											2012-12-30 18:28:36 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			if (node->bm_faces) | 
					
						
							|  |  |  | 				BLI_ghash_free(node->bm_faces, NULL, NULL); | 
					
						
							|  |  |  | 			if (node->bm_unique_verts) | 
					
						
							| 
									
										
										
										
											2013-08-25 20:03:45 +00:00
										 |  |  | 				BLI_gset_free(node->bm_unique_verts, NULL); | 
					
						
							| 
									
										
										
										
											2012-12-30 18:28:36 +00:00
										 |  |  | 			if (node->bm_other_verts) | 
					
						
							| 
									
										
										
										
											2013-08-25 20:03:45 +00:00
										 |  |  | 				BLI_gset_free(node->bm_other_verts, NULL); | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 20:10:59 +00:00
										 |  |  | 	if (bvh->deformed) { | 
					
						
							|  |  |  | 		if (bvh->verts) { | 
					
						
							|  |  |  | 			/* if pbvh was deformed, new memory was allocated for verts/faces -- free it */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			MEM_freeN(bvh->verts); | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 			if (bvh->faces) | 
					
						
							| 
									
										
										
										
											2011-11-24 13:39:43 +00:00
										 |  |  | 				MEM_freeN(bvh->faces); | 
					
						
							| 
									
										
										
										
											2010-06-21 20:10:59 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (bvh->nodes) | 
					
						
							| 
									
										
										
										
											2011-11-24 13:39:43 +00:00
										 |  |  | 		MEM_freeN(bvh->nodes); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (bvh->prim_indices) | 
					
						
							| 
									
										
										
										
											2011-11-24 13:39:43 +00:00
										 |  |  | 		MEM_freeN(bvh->prim_indices); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:28:36 +00:00
										 |  |  | 	if (bvh->bm_vert_to_node) | 
					
						
							|  |  |  | 		BLI_ghash_free(bvh->bm_vert_to_node, NULL, NULL); | 
					
						
							|  |  |  | 	if (bvh->bm_face_to_node) | 
					
						
							|  |  |  | 		BLI_ghash_free(bvh->bm_face_to_node, NULL, NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 	MEM_freeN(bvh); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-07 16:58:56 +02:00
										 |  |  | void BKE_pbvh_free_layer_disp(PBVH *bvh) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 	for (i = 0; i < bvh->totnode; ++i) | 
					
						
							|  |  |  | 		BKE_pbvh_node_layer_disp_free(&bvh->nodes[i]); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | static void pbvh_iter_begin(PBVHIter *iter, PBVH *bvh, BKE_pbvh_SearchCallback scb, void *search_data) | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	iter->bvh = bvh; | 
					
						
							|  |  |  | 	iter->scb = scb; | 
					
						
							|  |  |  | 	iter->search_data = search_data; | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	iter->stack = iter->stackfixed; | 
					
						
							|  |  |  | 	iter->stackspace = STACK_FIXED_DEPTH; | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	iter->stack[0].node = bvh->nodes; | 
					
						
							|  |  |  | 	iter->stack[0].revisiting = 0; | 
					
						
							|  |  |  | 	iter->stacksize = 1; | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void pbvh_iter_end(PBVHIter *iter) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (iter->stackspace > STACK_FIXED_DEPTH) | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 		MEM_freeN(iter->stack); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void pbvh_stack_push(PBVHIter *iter, PBVHNode *node, int revisiting) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (iter->stacksize == iter->stackspace) { | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 		PBVHStack *newstack; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		iter->stackspace *= 2; | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		newstack = MEM_callocN(sizeof(PBVHStack) * iter->stackspace, "PBVHStack"); | 
					
						
							|  |  |  | 		memcpy(newstack, iter->stack, sizeof(PBVHStack) * iter->stacksize); | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (iter->stackspace > STACK_FIXED_DEPTH) | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 			MEM_freeN(iter->stack); | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		iter->stack = newstack; | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	iter->stack[iter->stacksize].node = node; | 
					
						
							|  |  |  | 	iter->stack[iter->stacksize].revisiting = revisiting; | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 	iter->stacksize++; | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | static PBVHNode *pbvh_iter_next(PBVHIter *iter) | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 	PBVHNode *node; | 
					
						
							|  |  |  | 	int revisiting; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* purpose here is to traverse tree, visiting child nodes before their
 | 
					
						
							| 
									
										
										
										
											2012-03-03 20:19:11 +00:00
										 |  |  | 	 * parents, this order is necessary for e.g. computing bounding boxes */ | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	while (iter->stacksize) { | 
					
						
							| 
									
										
										
										
											2011-04-21 13:11:51 +00:00
										 |  |  | 		/* pop node */ | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 		iter->stacksize--; | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		node = iter->stack[iter->stacksize].node; | 
					
						
							| 
									
										
										
										
											2010-01-10 10:20:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-10 10:50:11 +00:00
										 |  |  | 		/* on a mesh with no faces this can happen
 | 
					
						
							|  |  |  | 		 * can remove this check if we know meshes have at least 1 face */ | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		if (node == NULL) | 
					
						
							| 
									
										
										
										
											2010-01-10 10:20:44 +00:00
										 |  |  | 			return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		revisiting = iter->stack[iter->stacksize].revisiting; | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 		/* revisiting node already checked */ | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (revisiting) | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 			return node; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (iter->scb && !iter->scb(node, iter->search_data)) | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 			continue;  /* don't traverse, outside of search zone */ | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (node->flag & PBVH_Leaf) { | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 			/* immediately hit leaf node */ | 
					
						
							|  |  |  | 			return node; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							|  |  |  | 			/* come back later when children are done */ | 
					
						
							|  |  |  | 			pbvh_stack_push(iter, node, 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			/* push two child nodes on the stack */ | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 			pbvh_stack_push(iter, iter->bvh->nodes + node->children_offset + 1, 0); | 
					
						
							|  |  |  | 			pbvh_stack_push(iter, iter->bvh->nodes + node->children_offset, 0); | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | static PBVHNode *pbvh_iter_next_occluded(PBVHIter *iter) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-04-21 13:11:51 +00:00
										 |  |  | 	PBVHNode *node; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	while (iter->stacksize) { | 
					
						
							| 
									
										
										
										
											2011-04-21 13:11:51 +00:00
										 |  |  | 		/* pop node */ | 
					
						
							|  |  |  | 		iter->stacksize--; | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		node = iter->stack[iter->stacksize].node; | 
					
						
							| 
									
										
										
										
											2011-04-21 13:11:51 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		/* on a mesh with no faces this can happen
 | 
					
						
							|  |  |  | 		 * can remove this check if we know meshes have at least 1 face */ | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		if (node == NULL) return NULL; | 
					
						
							| 
									
										
										
										
											2011-04-21 13:11:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		if (iter->scb && !iter->scb(node, iter->search_data)) continue;  /* don't traverse, outside of search zone */ | 
					
						
							| 
									
										
										
										
											2011-04-21 13:11:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (node->flag & PBVH_Leaf) { | 
					
						
							| 
									
										
										
										
											2011-04-21 13:11:51 +00:00
										 |  |  | 			/* immediately hit leaf node */ | 
					
						
							|  |  |  | 			return node; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 			pbvh_stack_push(iter, iter->bvh->nodes + node->children_offset + 1, 0); | 
					
						
							|  |  |  | 			pbvh_stack_push(iter, iter->bvh->nodes + node->children_offset, 0); | 
					
						
							| 
									
										
										
										
											2011-04-21 13:11:51 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | void BKE_pbvh_search_gather(PBVH *bvh, | 
					
						
							|  |  |  |                             BKE_pbvh_SearchCallback scb, void *search_data, | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  |                             PBVHNode ***r_array, int *r_tot) | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	PBVHIter iter; | 
					
						
							| 
									
										
										
										
											2013-08-03 18:05:30 +00:00
										 |  |  | 	PBVHNode **array = NULL, *node; | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	int tot = 0, space = 0; | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	pbvh_iter_begin(&iter, bvh, scb, search_data); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	while ((node = pbvh_iter_next(&iter))) { | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (node->flag & PBVH_Leaf) { | 
					
						
							|  |  |  | 			if (tot == space) { | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 				/* resize array if needed */ | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 				space = (tot == 0) ? 32 : space * 2; | 
					
						
							| 
									
										
										
										
											2013-08-03 18:05:30 +00:00
										 |  |  | 				array = MEM_recallocN_id(array, sizeof(PBVHNode *) * space, __func__); | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 			array[tot] = node; | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 			tot++; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	pbvh_iter_end(&iter); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (tot == 0 && array) { | 
					
						
							| 
									
										
										
										
											2010-01-05 14:26:38 +00:00
										 |  |  | 		MEM_freeN(array); | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		array = NULL; | 
					
						
							| 
									
										
										
										
											2010-01-05 14:26:38 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	*r_array = array; | 
					
						
							|  |  |  | 	*r_tot = tot; | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | void BKE_pbvh_search_callback(PBVH *bvh, | 
					
						
							|  |  |  |                               BKE_pbvh_SearchCallback scb, void *search_data, | 
					
						
							|  |  |  |                               BKE_pbvh_HitCallback hcb, void *hit_data) | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	PBVHIter iter; | 
					
						
							|  |  |  | 	PBVHNode *node; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	pbvh_iter_begin(&iter, bvh, scb, search_data); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	while ((node = pbvh_iter_next(&iter))) | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | 		if (node->flag & PBVH_Leaf) | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 			hcb(node, hit_data); | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	pbvh_iter_end(&iter); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | typedef struct node_tree { | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	PBVHNode *data; | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	struct node_tree *left; | 
					
						
							|  |  |  | 	struct node_tree *right; | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | } node_tree; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | static void node_tree_insert(node_tree *tree, node_tree *new_node) | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-04-21 13:11:51 +00:00
										 |  |  | 	if (new_node->data->tmin < tree->data->tmin) { | 
					
						
							|  |  |  | 		if (tree->left) { | 
					
						
							|  |  |  | 			node_tree_insert(tree->left, new_node); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							|  |  |  | 			tree->left = new_node; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		if (tree->right) { | 
					
						
							|  |  |  | 			node_tree_insert(tree->right, new_node); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							|  |  |  | 			tree->right = new_node; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | static void traverse_tree(node_tree *tree, BKE_pbvh_HitOccludedCallback hcb, void *hit_data, float *tmin) | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-04-21 13:11:51 +00:00
										 |  |  | 	if (tree->left) traverse_tree(tree->left, hcb, hit_data, tmin); | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-21 13:11:51 +00:00
										 |  |  | 	hcb(tree->data, hit_data, tmin); | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-21 13:11:51 +00:00
										 |  |  | 	if (tree->right) traverse_tree(tree->right, hcb, hit_data, tmin); | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | static void free_tree(node_tree *tree) | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-04-21 13:11:51 +00:00
										 |  |  | 	if (tree->left) { | 
					
						
							|  |  |  | 		free_tree(tree->left); | 
					
						
							| 
									
										
										
										
											2013-08-07 03:55:21 +00:00
										 |  |  | 		tree->left = NULL; | 
					
						
							| 
									
										
										
										
											2011-04-21 13:11:51 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-21 13:11:51 +00:00
										 |  |  | 	if (tree->right) { | 
					
						
							|  |  |  | 		free_tree(tree->right); | 
					
						
							| 
									
										
										
										
											2013-08-07 03:55:21 +00:00
										 |  |  | 		tree->right = NULL; | 
					
						
							| 
									
										
										
										
											2011-04-21 13:11:51 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-21 13:11:51 +00:00
										 |  |  | 	free(tree); | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | float BKE_pbvh_node_get_tmin(PBVHNode *node) | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-04-21 13:11:51 +00:00
										 |  |  | 	return node->tmin; | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | static void BKE_pbvh_search_callback_occluded(PBVH *bvh, | 
					
						
							|  |  |  |                                               BKE_pbvh_SearchCallback scb, void *search_data, | 
					
						
							|  |  |  |                                               BKE_pbvh_HitOccludedCallback hcb, void *hit_data) | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	PBVHIter iter; | 
					
						
							|  |  |  | 	PBVHNode *node; | 
					
						
							| 
									
										
										
										
											2013-08-07 03:55:21 +00:00
										 |  |  | 	node_tree *tree = NULL; | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	pbvh_iter_begin(&iter, bvh, scb, search_data); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	while ((node = pbvh_iter_next_occluded(&iter))) { | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (node->flag & PBVH_Leaf) { | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 			node_tree *new_node = malloc(sizeof(node_tree)); | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			new_node->data = node; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			new_node->left  = NULL; | 
					
						
							|  |  |  | 			new_node->right = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (tree) { | 
					
						
							|  |  |  | 				node_tree_insert(tree, new_node); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else { | 
					
						
							|  |  |  | 				tree = new_node; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	pbvh_iter_end(&iter); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (tree) { | 
					
						
							|  |  |  | 		float tmin = FLT_MAX; | 
					
						
							|  |  |  | 		traverse_tree(tree, hcb, hit_data, &tmin); | 
					
						
							|  |  |  | 		free_tree(tree); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-03 18:55:59 +11:00
										 |  |  | static bool update_search_cb(PBVHNode *node, void *data_v) | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	int flag = GET_INT_FROM_POINTER(data_v); | 
					
						
							| 
									
										
										
										
											2009-11-06 16:46:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (node->flag & PBVH_Leaf) | 
					
						
							| 
									
										
										
										
											2014-02-03 18:55:59 +11:00
										 |  |  | 		return (node->flag & flag) != 0; | 
					
						
							| 
									
										
										
										
											2012-10-22 17:33:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 	return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-04 20:36:38 +00:00
										 |  |  | static void pbvh_update_normals(PBVH *bvh, PBVHNode **nodes, | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  |                                 int totnode, float (*face_nors)[3]) | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-11-04 20:36:38 +00:00
										 |  |  | 	float (*vnor)[3]; | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 	int n; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:28:36 +00:00
										 |  |  | 	if (bvh->type == PBVH_BMESH) { | 
					
						
							|  |  |  | 		pbvh_bmesh_normals_update(nodes, totnode); | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (bvh->type != PBVH_FACES) | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-04 20:36:38 +00:00
										 |  |  | 	/* could be per node to save some memory, but also means
 | 
					
						
							| 
									
										
										
										
											2012-03-03 20:19:11 +00:00
										 |  |  | 	 * we have to store for each vertex which node it is in */ | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	vnor = MEM_callocN(sizeof(float) * 3 * bvh->totvert, "bvh temp vnors"); | 
					
						
							| 
									
										
										
										
											2009-11-04 20:36:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-01 17:51:03 +00:00
										 |  |  | 	/* subtle assumptions:
 | 
					
						
							|  |  |  | 	 * - We know that for all edited vertices, the nodes with faces | 
					
						
							|  |  |  | 	 *   adjacent to these vertices have been marked with PBVH_UpdateNormals. | 
					
						
							|  |  |  | 	 *   This is true because if the vertex is inside the brush radius, the | 
					
						
							|  |  |  | 	 *   bounding box of it's adjacent faces will be as well. | 
					
						
							|  |  |  | 	 * - However this is only true for the vertices that have actually been | 
					
						
							|  |  |  | 	 *   edited, not for all vertices in the nodes marked for update, so we | 
					
						
							|  |  |  | 	 *   can only update vertices marked with ME_VERT_PBVH_UPDATE. | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2009-11-04 20:36:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-22 14:11:10 +11:00
										 |  |  | #pragma omp parallel for private(n) schedule(static)
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	for (n = 0; n < totnode; n++) { | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		PBVHNode *node = nodes[n]; | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if ((node->flag & PBVH_UpdateNormals)) { | 
					
						
							| 
									
										
										
										
											2009-11-04 20:36:38 +00:00
										 |  |  | 			int i, j, totface, *faces; | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 			faces = node->prim_indices; | 
					
						
							|  |  |  | 			totface = node->totprim; | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 			for (i = 0; i < totface; ++i) { | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 				MFace *f = bvh->faces + faces[i]; | 
					
						
							| 
									
										
										
										
											2009-11-04 20:36:38 +00:00
										 |  |  | 				float fn[3]; | 
					
						
							|  |  |  | 				unsigned int *fv = &f->v1; | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 				int sides = (f->v4) ? 4 : 3; | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 				if (f->v4) | 
					
						
							| 
									
										
										
										
											2009-11-11 10:44:46 +00:00
										 |  |  | 					normal_quad_v3(fn, bvh->verts[f->v1].co, bvh->verts[f->v2].co, | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 					               bvh->verts[f->v3].co, bvh->verts[f->v4].co); | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 				else | 
					
						
							| 
									
										
										
										
											2009-11-11 10:44:46 +00:00
										 |  |  | 					normal_tri_v3(fn, bvh->verts[f->v1].co, bvh->verts[f->v2].co, | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 					              bvh->verts[f->v3].co); | 
					
						
							| 
									
										
										
										
											2009-11-04 20:36:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 				for (j = 0; j < sides; ++j) { | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 					int v = fv[j]; | 
					
						
							| 
									
										
										
										
											2009-11-04 20:36:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 					if (bvh->verts[v].flag & ME_VERT_PBVH_UPDATE) { | 
					
						
							| 
									
										
										
										
											2009-11-04 20:36:38 +00:00
										 |  |  | 						/* this seems like it could be very slow but profile
 | 
					
						
							| 
									
										
										
										
											2012-03-03 20:19:11 +00:00
										 |  |  | 						 * does not show this, so just leave it for now? */ | 
					
						
							| 
									
										
										
										
											2013-12-22 14:11:10 +11:00
										 |  |  | #pragma omp atomic
 | 
					
						
							| 
									
										
										
										
											2009-11-04 20:36:38 +00:00
										 |  |  | 						vnor[v][0] += fn[0]; | 
					
						
							| 
									
										
										
										
											2013-12-22 14:11:10 +11:00
										 |  |  | #pragma omp atomic
 | 
					
						
							| 
									
										
										
										
											2009-11-04 20:36:38 +00:00
										 |  |  | 						vnor[v][1] += fn[1]; | 
					
						
							| 
									
										
										
										
											2013-12-22 14:11:10 +11:00
										 |  |  | #pragma omp atomic
 | 
					
						
							| 
									
										
										
										
											2009-11-04 20:36:38 +00:00
										 |  |  | 						vnor[v][2] += fn[2]; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 				if (face_nors) | 
					
						
							| 
									
										
										
										
											2009-11-11 10:44:46 +00:00
										 |  |  | 					copy_v3_v3(face_nors[faces[i]], fn); | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-22 14:11:10 +11:00
										 |  |  | #pragma omp parallel for private(n) schedule(static)
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	for (n = 0; n < totnode; n++) { | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		PBVHNode *node = nodes[n]; | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (node->flag & PBVH_UpdateNormals) { | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 			int i, *verts, totvert; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 			verts = node->vert_indices; | 
					
						
							|  |  |  | 			totvert = node->uniq_verts; | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 			for (i = 0; i < totvert; ++i) { | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 				const int v = verts[i]; | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 				MVert *mvert = &bvh->verts[v]; | 
					
						
							| 
									
										
										
										
											2009-11-04 20:36:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 				if (mvert->flag & ME_VERT_PBVH_UPDATE) { | 
					
						
							| 
									
										
										
										
											2009-11-04 20:36:38 +00:00
										 |  |  | 					float no[3]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-11 10:44:46 +00:00
										 |  |  | 					copy_v3_v3(no, vnor[v]); | 
					
						
							|  |  |  | 					normalize_v3(no); | 
					
						
							| 
									
										
										
										
											2012-05-13 11:05:52 +00:00
										 |  |  | 					normal_float_to_short_v3(mvert->no, no); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-04 20:36:38 +00:00
										 |  |  | 					mvert->flag &= ~ME_VERT_PBVH_UPDATE; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			node->flag &= ~PBVH_UpdateNormals; | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-11-04 20:36:38 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	MEM_freeN(vnor); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:28:36 +00:00
										 |  |  | void pbvh_update_BB_redraw(PBVH *bvh, PBVHNode **nodes, int totnode, int flag) | 
					
						
							| 
									
										
										
										
											2009-11-04 20:36:38 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	int n; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* update BB, redraw flag */ | 
					
						
							| 
									
										
										
										
											2013-12-22 14:11:10 +11:00
										 |  |  | #pragma omp parallel for private(n) schedule(static)
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	for (n = 0; n < totnode; n++) { | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		PBVHNode *node = nodes[n]; | 
					
						
							| 
									
										
										
										
											2009-11-04 20:36:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if ((flag & PBVH_UpdateBB) && (node->flag & PBVH_UpdateBB)) | 
					
						
							| 
									
										
										
										
											2009-11-04 20:36:38 +00:00
										 |  |  | 			/* don't clear flag yet, leave it for flushing later */ | 
					
						
							|  |  |  | 			update_node_vb(bvh, node); | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if ((flag & PBVH_UpdateOriginalBB) && (node->flag & PBVH_UpdateOriginalBB)) | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 			node->orig_vb = node->vb; | 
					
						
							| 
									
										
										
										
											2009-11-06 16:46:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if ((flag & PBVH_UpdateRedraw) && (node->flag & PBVH_UpdateRedraw)) | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 			node->flag &= ~PBVH_UpdateRedraw; | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-06 02:40:08 +00:00
										 |  |  | static void pbvh_update_draw_buffers(PBVH *bvh, PBVHNode **nodes, int totnode) | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	PBVHNode *node; | 
					
						
							|  |  |  | 	int n; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* can't be done in parallel with OpenGL */ | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	for (n = 0; n < totnode; n++) { | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		node = nodes[n]; | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (node->flag & PBVH_RebuildDrawBuffers) { | 
					
						
							| 
									
										
										
										
											2014-01-02 22:22:36 +02:00
										 |  |  | 			GPU_free_pbvh_buffers(node->draw_buffers); | 
					
						
							| 
									
										
										
										
											2012-06-10 16:37:22 +00:00
										 |  |  | 			switch (bvh->type) { | 
					
						
							|  |  |  | 				case PBVH_GRIDS: | 
					
						
							|  |  |  | 					node->draw_buffers = | 
					
						
							| 
									
										
										
										
											2014-01-02 22:22:36 +02:00
										 |  |  | 						GPU_build_grid_pbvh_buffers(node->prim_indices, | 
					
						
							| 
									
										
										
										
											2012-06-10 19:59:02 +00:00
										 |  |  | 					                           node->totprim, | 
					
						
							|  |  |  | 					                           bvh->grid_hidden, | 
					
						
							|  |  |  | 					                           bvh->gridkey.grid_size); | 
					
						
							| 
									
										
										
										
											2012-06-10 16:37:22 +00:00
										 |  |  | 					break; | 
					
						
							|  |  |  | 				case PBVH_FACES: | 
					
						
							|  |  |  | 					node->draw_buffers = | 
					
						
							| 
									
										
										
										
											2014-01-02 22:22:36 +02:00
										 |  |  | 						GPU_build_pbvh_mesh_buffers(node->face_vert_indices, | 
					
						
							| 
									
										
										
										
											2012-06-10 19:59:02 +00:00
										 |  |  | 					                           bvh->faces, bvh->verts, | 
					
						
							|  |  |  | 					                           node->prim_indices, | 
					
						
							|  |  |  | 					                           node->totprim); | 
					
						
							| 
									
										
										
										
											2012-06-10 16:37:22 +00:00
										 |  |  | 					break; | 
					
						
							| 
									
										
										
										
											2013-03-02 12:05:25 +00:00
										 |  |  | 				case PBVH_BMESH: | 
					
						
							| 
									
										
										
										
											2012-12-30 18:28:36 +00:00
										 |  |  | 					node->draw_buffers = | 
					
						
							| 
									
										
										
										
											2014-01-02 22:22:36 +02:00
										 |  |  | 						GPU_build_bmesh_pbvh_buffers(bvh->flags & | 
					
						
							| 
									
										
										
										
											2013-03-02 12:05:25 +00:00
										 |  |  | 					                            PBVH_DYNTOPO_SMOOTH_SHADING); | 
					
						
							| 
									
										
										
										
											2012-12-30 18:28:36 +00:00
										 |  |  | 					break; | 
					
						
							| 
									
										
										
										
											2012-03-14 06:32:25 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  |   | 
					
						
							|  |  |  | 			node->flag &= ~PBVH_RebuildDrawBuffers; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (node->flag & PBVH_UpdateDrawBuffers) { | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 			switch (bvh->type) { | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 				case PBVH_GRIDS: | 
					
						
							| 
									
										
										
										
											2014-01-02 22:22:36 +02:00
										 |  |  | 					GPU_update_grid_pbvh_buffers(node->draw_buffers, | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 					                        bvh->grids, | 
					
						
							|  |  |  | 					                        bvh->grid_flag_mats, | 
					
						
							|  |  |  | 					                        node->prim_indices, | 
					
						
							|  |  |  | 					                        node->totprim, | 
					
						
							| 
									
										
										
										
											2012-10-22 17:33:53 +00:00
										 |  |  | 					                        &bvh->gridkey, | 
					
						
							|  |  |  | 					                        bvh->show_diffuse_color); | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 					break; | 
					
						
							|  |  |  | 				case PBVH_FACES: | 
					
						
							| 
									
										
										
										
											2014-01-02 22:22:36 +02:00
										 |  |  | 					GPU_update_mesh_pbvh_buffers(node->draw_buffers, | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 					                        bvh->verts, | 
					
						
							|  |  |  | 					                        node->vert_indices, | 
					
						
							|  |  |  | 					                        node->uniq_verts + | 
					
						
							|  |  |  | 					                        node->face_verts, | 
					
						
							|  |  |  | 					                        CustomData_get_layer(bvh->vdata, | 
					
						
							| 
									
										
										
										
											2012-10-22 17:33:53 +00:00
										 |  |  | 					                                             CD_PAINT_MASK), | 
					
						
							|  |  |  | 					                        node->face_vert_indices, | 
					
						
							|  |  |  | 					                        bvh->show_diffuse_color); | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 					break; | 
					
						
							| 
									
										
										
										
											2013-03-02 12:05:25 +00:00
										 |  |  | 				case PBVH_BMESH: | 
					
						
							| 
									
										
										
										
											2014-01-02 22:22:36 +02:00
										 |  |  | 					GPU_update_bmesh_pbvh_buffers(node->draw_buffers, | 
					
						
							| 
									
										
										
										
											2013-03-02 12:05:25 +00:00
										 |  |  | 					                         bvh->bm, | 
					
						
							|  |  |  | 					                         node->bm_faces, | 
					
						
							|  |  |  | 					                         node->bm_unique_verts, | 
					
						
							|  |  |  | 					                         node->bm_other_verts); | 
					
						
							| 
									
										
										
										
											2012-12-30 18:28:36 +00:00
										 |  |  | 					break; | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			node->flag &= ~PBVH_UpdateDrawBuffers; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-06 16:46:35 +00:00
										 |  |  | static int pbvh_flush_bb(PBVH *bvh, PBVHNode *node, int flag) | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	int update = 0; | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* difficult to multithread well, we just do single threaded recursive */ | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (node->flag & PBVH_Leaf) { | 
					
						
							|  |  |  | 		if (flag & PBVH_UpdateBB) { | 
					
						
							| 
									
										
										
										
											2009-11-06 16:46:35 +00:00
										 |  |  | 			update |= (node->flag & PBVH_UpdateBB); | 
					
						
							|  |  |  | 			node->flag &= ~PBVH_UpdateBB; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (flag & PBVH_UpdateOriginalBB) { | 
					
						
							| 
									
										
										
										
											2009-11-06 16:46:35 +00:00
										 |  |  | 			update |= (node->flag & PBVH_UpdateOriginalBB); | 
					
						
							|  |  |  | 			node->flag &= ~PBVH_UpdateOriginalBB; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 		return update; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2009-11-06 16:46:35 +00:00
										 |  |  | 		update |= pbvh_flush_bb(bvh, bvh->nodes + node->children_offset, flag); | 
					
						
							|  |  |  | 		update |= pbvh_flush_bb(bvh, bvh->nodes + node->children_offset + 1, flag); | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (update & PBVH_UpdateBB) | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 			update_node_vb(bvh, node); | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (update & PBVH_UpdateOriginalBB) | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 			node->orig_vb = node->vb; | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 	return update; | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | void BKE_pbvh_update(PBVH *bvh, int flag, float (*face_nors)[3]) | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 	PBVHNode **nodes; | 
					
						
							|  |  |  | 	int totnode; | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (!bvh->nodes) | 
					
						
							| 
									
										
										
										
											2011-11-24 13:39:43 +00:00
										 |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | 	BKE_pbvh_search_gather(bvh, update_search_cb, SET_INT_IN_POINTER(flag), | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	                       &nodes, &totnode); | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (flag & PBVH_UpdateNormals) | 
					
						
							| 
									
										
										
										
											2009-11-04 20:36:38 +00:00
										 |  |  | 		pbvh_update_normals(bvh, nodes, totnode, face_nors); | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	if (flag & (PBVH_UpdateBB | PBVH_UpdateOriginalBB | PBVH_UpdateRedraw)) | 
					
						
							| 
									
										
										
										
											2009-11-04 20:36:38 +00:00
										 |  |  | 		pbvh_update_BB_redraw(bvh, nodes, totnode, flag); | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	if (flag & (PBVH_UpdateBB | PBVH_UpdateOriginalBB)) | 
					
						
							| 
									
										
										
										
											2009-11-06 16:46:35 +00:00
										 |  |  | 		pbvh_flush_bb(bvh, bvh->nodes, flag); | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (nodes) MEM_freeN(nodes); | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | void BKE_pbvh_redraw_BB(PBVH *bvh, float bb_min[3], float bb_max[3]) | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	PBVHIter iter; | 
					
						
							|  |  |  | 	PBVHNode *node; | 
					
						
							|  |  |  | 	BB bb; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	BB_reset(&bb); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	pbvh_iter_begin(&iter, bvh, NULL, NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	while ((node = pbvh_iter_next(&iter))) | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (node->flag & PBVH_UpdateRedraw) | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | 			BB_expand_with_bb(&bb, &node->vb); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	pbvh_iter_end(&iter); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-11 10:44:46 +00:00
										 |  |  | 	copy_v3_v3(bb_min, bb.bmin); | 
					
						
							|  |  |  | 	copy_v3_v3(bb_max, bb.bmax); | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | void BKE_pbvh_get_grid_updates(PBVH *bvh, int clear, void ***gridfaces, int *totface) | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	PBVHIter iter; | 
					
						
							|  |  |  | 	PBVHNode *node; | 
					
						
							|  |  |  | 	GHashIterator *hiter; | 
					
						
							|  |  |  | 	GHash *map; | 
					
						
							|  |  |  | 	void *face, **faces; | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | 	unsigned i; | 
					
						
							| 
									
										
										
										
											2011-04-21 13:11:51 +00:00
										 |  |  | 	int tot; | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-16 00:51:36 +00:00
										 |  |  | 	map = BLI_ghash_ptr_new("pbvh_get_grid_updates gh"); | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	pbvh_iter_begin(&iter, bvh, NULL, NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	while ((node = pbvh_iter_next(&iter))) { | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (node->flag & PBVH_UpdateNormals) { | 
					
						
							|  |  |  | 			for (i = 0; i < node->totprim; ++i) { | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 				face = bvh->gridfaces[node->prim_indices[i]]; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 				if (!BLI_ghash_lookup(map, face)) | 
					
						
							| 
									
										
										
										
											2009-12-09 13:37:19 +00:00
										 |  |  | 					BLI_ghash_insert(map, face, face); | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 			if (clear) | 
					
						
							| 
									
										
										
										
											2009-12-09 13:37:19 +00:00
										 |  |  | 				node->flag &= ~PBVH_UpdateNormals; | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	pbvh_iter_end(&iter); | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	tot = BLI_ghash_size(map); | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (tot == 0) { | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		*totface = 0; | 
					
						
							|  |  |  | 		*gridfaces = NULL; | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 		BLI_ghash_free(map, NULL, NULL); | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	faces = MEM_callocN(sizeof(void *) * tot, "PBVH Grid Faces"); | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	for (hiter = BLI_ghashIterator_new(map), i = 0; | 
					
						
							| 
									
										
										
										
											2013-05-08 12:58:11 +00:00
										 |  |  | 	     BLI_ghashIterator_done(hiter) == false; | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	     BLI_ghashIterator_step(hiter), ++i) | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		faces[i] = BLI_ghashIterator_getKey(hiter); | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-01 11:36:22 +00:00
										 |  |  | 	BLI_ghashIterator_free(hiter); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 	BLI_ghash_free(map, NULL, NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	*totface = tot; | 
					
						
							|  |  |  | 	*gridfaces = faces; | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-12 23:03:43 +00:00
										 |  |  | /***************************** PBVH Access ***********************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | PBVHType BKE_pbvh_type(const PBVH *bvh) | 
					
						
							| 
									
										
										
										
											2012-03-12 23:03:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	return bvh->type; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-20 00:19:57 +00:00
										 |  |  | void BKE_pbvh_bounding_box(const PBVH *bvh, float min[3], float max[3]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (bvh->totnode) { | 
					
						
							|  |  |  | 		const BB *bb = &bvh->nodes[0].vb; | 
					
						
							|  |  |  | 		copy_v3_v3(min, bb->bmin); | 
					
						
							|  |  |  | 		copy_v3_v3(max, bb->bmax); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		zero_v3(min); | 
					
						
							|  |  |  | 		zero_v3(max); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-22 23:20:48 +00:00
										 |  |  | BLI_bitmap **BKE_pbvh_grid_hidden(const PBVH *bvh) | 
					
						
							| 
									
										
										
										
											2012-03-14 06:32:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	BLI_assert(bvh->type == PBVH_GRIDS); | 
					
						
							|  |  |  | 	return bvh->grid_hidden; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | void BKE_pbvh_get_grid_key(const PBVH *bvh, CCGKey *key) | 
					
						
							| 
									
										
										
										
											2012-05-10 20:33:09 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	BLI_assert(bvh->type == PBVH_GRIDS); | 
					
						
							|  |  |  | 	*key = bvh->gridkey; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | BMesh *BKE_pbvh_get_bmesh(PBVH *bvh) | 
					
						
							| 
									
										
										
										
											2012-12-30 18:28:36 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	BLI_assert(bvh->type == PBVH_BMESH); | 
					
						
							|  |  |  | 	return bvh->bm; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | /***************************** Node Access ***********************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | void BKE_pbvh_node_mark_update(PBVHNode *node) | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	node->flag |= PBVH_UpdateNormals | PBVH_UpdateBB | PBVH_UpdateOriginalBB | PBVH_UpdateDrawBuffers | PBVH_UpdateRedraw; | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | void BKE_pbvh_node_mark_rebuild_draw(PBVHNode *node) | 
					
						
							| 
									
										
										
										
											2012-03-14 06:32:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	node->flag |= PBVH_RebuildDrawBuffers | PBVH_UpdateDrawBuffers | PBVH_UpdateRedraw; | 
					
						
							| 
									
										
										
										
											2012-03-14 06:32:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-15 23:00:15 +02:00
										 |  |  | void BKE_pbvh_node_mark_redraw(PBVHNode *node) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	node->flag |= PBVH_UpdateDrawBuffers | PBVH_UpdateRedraw; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | void BKE_pbvh_node_fully_hidden_set(PBVHNode *node, int fully_hidden) | 
					
						
							| 
									
										
										
										
											2012-03-14 06:32:25 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	BLI_assert(node->flag & PBVH_Leaf); | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (fully_hidden) | 
					
						
							| 
									
										
										
										
											2012-03-14 06:32:25 +00:00
										 |  |  | 		node->flag |= PBVH_FullyHidden; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		node->flag &= ~PBVH_FullyHidden; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | void BKE_pbvh_node_get_verts(PBVH *bvh, PBVHNode *node, int **vert_indices, MVert **verts) | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	if (vert_indices) *vert_indices = node->vert_indices; | 
					
						
							|  |  |  | 	if (verts) *verts = bvh->verts; | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | void BKE_pbvh_node_num_verts(PBVH *bvh, PBVHNode *node, int *uniquevert, int *totvert) | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-12 23:03:43 +00:00
										 |  |  | 	int tot; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	switch (bvh->type) { | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		case PBVH_GRIDS: | 
					
						
							|  |  |  | 			tot = node->totprim * bvh->gridkey.grid_area; | 
					
						
							|  |  |  | 			if (totvert) *totvert = tot; | 
					
						
							|  |  |  | 			if (uniquevert) *uniquevert = tot; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case PBVH_FACES: | 
					
						
							|  |  |  | 			if (totvert) *totvert = node->uniq_verts + node->face_verts; | 
					
						
							|  |  |  | 			if (uniquevert) *uniquevert = node->uniq_verts; | 
					
						
							|  |  |  | 			break; | 
					
						
							| 
									
										
										
										
											2012-12-30 18:28:36 +00:00
										 |  |  | 		case PBVH_BMESH: | 
					
						
							| 
									
										
										
										
											2013-08-25 20:03:45 +00:00
										 |  |  | 			tot = BLI_gset_size(node->bm_unique_verts); | 
					
						
							|  |  |  | 			if (totvert) *totvert = tot + BLI_gset_size(node->bm_other_verts); | 
					
						
							| 
									
										
										
										
											2012-12-30 18:28:36 +00:00
										 |  |  | 			if (uniquevert) *uniquevert = tot; | 
					
						
							|  |  |  | 			break; | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | void BKE_pbvh_node_get_grids(PBVH *bvh, PBVHNode *node, int **grid_indices, int *totgrid, int *maxgrid, int *gridsize, CCGElem ***griddata, DMGridAdjacency **gridadj) | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	switch (bvh->type) { | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		case PBVH_GRIDS: | 
					
						
							|  |  |  | 			if (grid_indices) *grid_indices = node->prim_indices; | 
					
						
							|  |  |  | 			if (totgrid) *totgrid = node->totprim; | 
					
						
							|  |  |  | 			if (maxgrid) *maxgrid = bvh->totgrid; | 
					
						
							|  |  |  | 			if (gridsize) *gridsize = bvh->gridkey.grid_size; | 
					
						
							|  |  |  | 			if (griddata) *griddata = bvh->grids; | 
					
						
							|  |  |  | 			if (gridadj) *gridadj = bvh->gridadj; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case PBVH_FACES: | 
					
						
							| 
									
										
										
										
											2012-12-30 18:28:36 +00:00
										 |  |  | 		case PBVH_BMESH: | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 			if (grid_indices) *grid_indices = NULL; | 
					
						
							|  |  |  | 			if (totgrid) *totgrid = 0; | 
					
						
							|  |  |  | 			if (maxgrid) *maxgrid = 0; | 
					
						
							|  |  |  | 			if (gridsize) *gridsize = 0; | 
					
						
							|  |  |  | 			if (griddata) *griddata = NULL; | 
					
						
							|  |  |  | 			if (gridadj) *gridadj = NULL; | 
					
						
							|  |  |  | 			break; | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | void BKE_pbvh_node_get_BB(PBVHNode *node, float bb_min[3], float bb_max[3]) | 
					
						
							| 
									
										
										
										
											2009-11-06 16:46:35 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-11-11 10:44:46 +00:00
										 |  |  | 	copy_v3_v3(bb_min, node->vb.bmin); | 
					
						
							|  |  |  | 	copy_v3_v3(bb_max, node->vb.bmax); | 
					
						
							| 
									
										
										
										
											2009-11-06 16:46:35 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | void BKE_pbvh_node_get_original_BB(PBVHNode *node, float bb_min[3], float bb_max[3]) | 
					
						
							| 
									
										
										
										
											2009-11-06 16:46:35 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-11-11 10:44:46 +00:00
										 |  |  | 	copy_v3_v3(bb_min, node->orig_vb.bmin); | 
					
						
							|  |  |  | 	copy_v3_v3(bb_max, node->orig_vb.bmax); | 
					
						
							| 
									
										
										
										
											2009-11-06 16:46:35 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | void BKE_pbvh_node_get_proxies(PBVHNode *node, PBVHProxyNode **proxies, int *proxy_count) | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	if (node->proxy_count > 0) { | 
					
						
							|  |  |  | 		if (proxies) *proxies = node->proxies; | 
					
						
							|  |  |  | 		if (proxy_count) *proxy_count = node->proxy_count; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2013-08-07 03:55:21 +00:00
										 |  |  | 		if (proxies) *proxies = NULL; | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | 		if (proxy_count) *proxy_count = 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Sculpt: Multithreading & PBVH Changes
* Sculpting, normal update and bounding box code is now multithreaded
  using OpenMP.
* Fix a number of update issues: normals on node boundaries, outdated
  bounding boxes, partial redraw, .. . There's probably still a few
  left, but should be better now.
* Clicking once now does a single paint instead of two (was also
  painting on mouse up event).
* Smooth shading now is enabled for the full mesh when the first face
  uses it (so it can be tested at least).
Implementation Notes:
* PBVH search can now be done either using a callback or bt gathering the
  nodes in an array. The latter makes multithreading with OpenMP easier.
* Normals update code is now inside PBVH, was doing it per node before but
  should do all faces first and only then vertices.
* Instead of using search modes + 1 modified flag, now nodes get 4 flags
  to indicate what needs to be updated for them, found that this makes it
  easier for me to understand the code and fix update bugs.
* PBVHNode is now exposed as an abstract type, I think this makes it more
  clear what is happening than having it's data passed as part of callback
  functions.
* Active_verts list was replaced by looping over nodes and the vertices
  inside them. However the grab brush still uses the active_verts system,
  will fix that later.
* Some micro-optimizations, like avoiding a few multiplications/divisions,
  using local variables instead of pointers, or looping over fewer vertices
  to update the bounding boxes.
											
										 
											2009-11-02 18:47:03 +00:00
										 |  |  | /********************************* Raycast ***********************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | typedef struct { | 
					
						
							| 
									
										
										
										
											2012-07-05 03:55:55 +00:00
										 |  |  | 	IsectRayAABBData ray; | 
					
						
							| 
									
										
										
										
											2009-11-06 16:46:35 +00:00
										 |  |  | 	int original; | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | } RaycastData; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-03 18:55:59 +11:00
										 |  |  | static bool ray_aabb_intersect(PBVHNode *node, void *data_v) | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-07-05 03:55:55 +00:00
										 |  |  | 	RaycastData *rcd = data_v; | 
					
						
							|  |  |  | 	float bb_min[3], bb_max[3]; | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-07-05 03:55:55 +00:00
										 |  |  | 	if (rcd->original) | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | 		BKE_pbvh_node_get_original_BB(node, bb_min, bb_max); | 
					
						
							| 
									
										
										
										
											2009-11-06 16:46:35 +00:00
										 |  |  | 	else | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | 		BKE_pbvh_node_get_BB(node, bb_min, bb_max); | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-07-05 03:55:55 +00:00
										 |  |  | 	return isect_ray_aabb(&rcd->ray, bb_min, bb_max, &node->tmin); | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | void BKE_pbvh_raycast(PBVH *bvh, BKE_pbvh_HitOccludedCallback cb, void *data, | 
					
						
							| 
									
										
										
										
											2012-06-10 16:37:22 +00:00
										 |  |  |                       const float ray_start[3], const float ray_normal[3], | 
					
						
							| 
									
										
										
										
											2012-06-10 19:59:02 +00:00
										 |  |  |                       int original) | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	RaycastData rcd; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-07-05 03:55:55 +00:00
										 |  |  | 	isect_ray_aabb_initialize(&rcd.ray, ray_start, ray_normal); | 
					
						
							| 
									
										
										
										
											2009-11-06 16:46:35 +00:00
										 |  |  | 	rcd.original = original; | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | 	BKE_pbvh_search_callback_occluded(bvh, ray_aabb_intersect, &rcd, cb, data); | 
					
						
							| 
									
										
										
										
											2009-10-27 19:53:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-03 18:55:59 +11:00
										 |  |  | bool ray_face_intersection(const float ray_start[3], | 
					
						
							|  |  |  |                            const float ray_normal[3], | 
					
						
							|  |  |  |                            const float t0[3], const float t1[3], | 
					
						
							|  |  |  |                            const float t2[3], const float t3[3], | 
					
						
							|  |  |  |                            float *fdist) | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-04-21 13:11:51 +00:00
										 |  |  | 	float dist; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if ((isect_ray_tri_epsilon_v3(ray_start, ray_normal, t0, t1, t2, &dist, NULL, 0.1f) && dist < *fdist) || | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	    (t3 && isect_ray_tri_epsilon_v3(ray_start, ray_normal, t0, t2, t3, &dist, NULL, 0.1f) && dist < *fdist)) | 
					
						
							| 
									
										
										
										
											2011-04-21 13:11:51 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 		*fdist = dist; | 
					
						
							|  |  |  | 		return 1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-03 18:55:59 +11:00
										 |  |  | static bool pbvh_faces_node_raycast(PBVH *bvh, const PBVHNode *node, | 
					
						
							|  |  |  |                                     float (*origco)[3], | 
					
						
							|  |  |  |                                     const float ray_start[3], | 
					
						
							|  |  |  |                                     const float ray_normal[3], float *dist) | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-06-10 16:37:22 +00:00
										 |  |  | 	const MVert *vert = bvh->verts; | 
					
						
							|  |  |  | 	const int *faces = node->prim_indices; | 
					
						
							| 
									
										
										
										
											2014-02-03 18:55:59 +11:00
										 |  |  | 	int i, totface = node->totprim; | 
					
						
							|  |  |  | 	bool hit = false; | 
					
						
							| 
									
										
										
										
											2012-06-10 16:37:22 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for (i = 0; i < totface; ++i) { | 
					
						
							|  |  |  | 		const MFace *f = bvh->faces + faces[i]; | 
					
						
							|  |  |  | 		const int *face_verts = node->face_vert_indices[i]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (paint_is_face_hidden(f, vert)) | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (origco) { | 
					
						
							|  |  |  | 			/* intersect with backuped original coordinates */ | 
					
						
							|  |  |  | 			hit |= ray_face_intersection(ray_start, ray_normal, | 
					
						
							| 
									
										
										
										
											2012-06-10 19:59:02 +00:00
										 |  |  | 			                             origco[face_verts[0]], | 
					
						
							|  |  |  | 			                             origco[face_verts[1]], | 
					
						
							|  |  |  | 			                             origco[face_verts[2]], | 
					
						
							|  |  |  | 			                             f->v4 ? origco[face_verts[3]] : NULL, | 
					
						
							|  |  |  | 			                             dist); | 
					
						
							| 
									
										
										
										
											2012-06-10 16:37:22 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							|  |  |  | 			/* intersect with current coordinates */ | 
					
						
							|  |  |  | 			hit |= ray_face_intersection(ray_start, ray_normal, | 
					
						
							| 
									
										
										
										
											2012-06-10 19:59:02 +00:00
										 |  |  | 			                             vert[f->v1].co, | 
					
						
							|  |  |  | 			                             vert[f->v2].co, | 
					
						
							|  |  |  | 			                             vert[f->v3].co, | 
					
						
							|  |  |  | 			                             f->v4 ? vert[f->v4].co : NULL, | 
					
						
							|  |  |  | 			                             dist); | 
					
						
							| 
									
										
										
										
											2012-06-10 16:37:22 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return hit; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-24 13:21:58 +11:00
										 |  |  | static bool pbvh_grids_node_raycast( | 
					
						
							|  |  |  |         PBVH *bvh, PBVHNode *node, | 
					
						
							|  |  |  |         float (*origco)[3], | 
					
						
							|  |  |  |         const float ray_start[3], const float ray_normal[3], | 
					
						
							|  |  |  |         float *dist) | 
					
						
							| 
									
										
										
										
											2012-06-10 16:37:22 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	int totgrid = node->totprim; | 
					
						
							|  |  |  | 	int gridsize = bvh->gridkey.grid_size; | 
					
						
							| 
									
										
										
										
											2014-03-24 13:21:58 +11:00
										 |  |  | 	int i, x, y; | 
					
						
							|  |  |  | 	bool hit = false; | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-10 16:37:22 +00:00
										 |  |  | 	for (i = 0; i < totgrid; ++i) { | 
					
						
							|  |  |  | 		CCGElem *grid = bvh->grids[node->prim_indices[i]]; | 
					
						
							| 
									
										
										
										
											2013-07-22 23:20:48 +00:00
										 |  |  | 		BLI_bitmap *gh; | 
					
						
							| 
									
										
										
										
											2012-03-14 06:32:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-10 16:37:22 +00:00
										 |  |  | 		if (!grid) | 
					
						
							|  |  |  | 			continue; | 
					
						
							| 
									
										
										
										
											2012-03-14 06:32:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-10 16:37:22 +00:00
										 |  |  | 		gh = bvh->grid_hidden[node->prim_indices[i]]; | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-10 16:37:22 +00:00
										 |  |  | 		for (y = 0; y < gridsize - 1; ++y) { | 
					
						
							|  |  |  | 			for (x = 0; x < gridsize - 1; ++x) { | 
					
						
							|  |  |  | 				/* check if grid face is hidden */ | 
					
						
							|  |  |  | 				if (gh) { | 
					
						
							|  |  |  | 					if (paint_is_grid_face_hidden(gh, gridsize, x, y)) | 
					
						
							|  |  |  | 						continue; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				if (origco) { | 
					
						
							|  |  |  | 					hit |= ray_face_intersection(ray_start, ray_normal, | 
					
						
							| 
									
										
										
										
											2012-06-10 19:59:02 +00:00
										 |  |  | 					                             origco[y * gridsize + x], | 
					
						
							|  |  |  | 					                             origco[y * gridsize + x + 1], | 
					
						
							|  |  |  | 					                             origco[(y + 1) * gridsize + x + 1], | 
					
						
							|  |  |  | 					                             origco[(y + 1) * gridsize + x], | 
					
						
							|  |  |  | 					                             dist); | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 				else { | 
					
						
							|  |  |  | 					hit |= ray_face_intersection(ray_start, ray_normal, | 
					
						
							| 
									
										
										
										
											2012-06-10 19:59:02 +00:00
										 |  |  | 					                             CCG_grid_elem_co(&bvh->gridkey, grid, x, y), | 
					
						
							|  |  |  | 					                             CCG_grid_elem_co(&bvh->gridkey, grid, x + 1, y), | 
					
						
							|  |  |  | 					                             CCG_grid_elem_co(&bvh->gridkey, grid, x + 1, y + 1), | 
					
						
							|  |  |  | 					                             CCG_grid_elem_co(&bvh->gridkey, grid, x, y + 1), | 
					
						
							|  |  |  | 					                             dist); | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2012-06-10 16:37:22 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (origco) | 
					
						
							|  |  |  | 			origco += gridsize * gridsize; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return hit; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-24 13:21:58 +11:00
										 |  |  | bool BKE_pbvh_node_raycast( | 
					
						
							|  |  |  |         PBVH *bvh, PBVHNode *node, float (*origco)[3], int use_origco, | 
					
						
							|  |  |  |         const float ray_start[3], const float ray_normal[3], | 
					
						
							|  |  |  |         float *dist) | 
					
						
							| 
									
										
										
										
											2012-06-10 16:37:22 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2014-02-03 18:55:59 +11:00
										 |  |  | 	bool hit = false; | 
					
						
							| 
									
										
										
										
											2012-06-10 16:37:22 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (node->flag & PBVH_FullyHidden) | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	switch (bvh->type) { | 
					
						
							|  |  |  | 		case PBVH_FACES: | 
					
						
							|  |  |  | 			hit |= pbvh_faces_node_raycast(bvh, node, origco, | 
					
						
							| 
									
										
										
										
											2012-06-10 19:59:02 +00:00
										 |  |  | 			                               ray_start, ray_normal, dist); | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		case PBVH_GRIDS: | 
					
						
							| 
									
										
										
										
											2012-06-10 16:37:22 +00:00
										 |  |  | 			hit |= pbvh_grids_node_raycast(bvh, node, origco, | 
					
						
							| 
									
										
										
										
											2012-06-10 19:59:02 +00:00
										 |  |  | 			                               ray_start, ray_normal, dist); | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 			break; | 
					
						
							| 
									
										
										
										
											2012-12-30 18:28:36 +00:00
										 |  |  | 		case PBVH_BMESH: | 
					
						
							|  |  |  | 			hit = pbvh_bmesh_node_raycast(node, ray_start, ray_normal, dist, use_origco); | 
					
						
							|  |  |  | 			break; | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return hit; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-18 18:34:02 +02:00
										 |  |  | void BKE_pbvh_raycast_project_ray_root (PBVH *bvh, bool original, float ray_start[3], float ray_end[3], float ray_normal[3]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (bvh->nodes) { | 
					
						
							|  |  |  | 		float rootmin_start, rootmin_end; | 
					
						
							|  |  |  | 		float bb_min_root[3], bb_max_root[3], bb_center[3], bb_diff[3]; | 
					
						
							|  |  |  | 		IsectRayAABBData ray; | 
					
						
							|  |  |  | 		float ray_normal_inv[3]; | 
					
						
							|  |  |  | 		float offset = 1.0f + 1e-3f; | 
					
						
							|  |  |  | 		float offset_vec[3] = {1e-3f, 1e-3f, 1e-3f}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (original) | 
					
						
							|  |  |  | 			BKE_pbvh_node_get_original_BB(bvh->nodes, bb_min_root, bb_max_root); | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			BKE_pbvh_node_get_BB(bvh->nodes, bb_min_root, bb_max_root); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* slightly offset min and max in case we have a zero width node (due to a plane mesh for instance),
 | 
					
						
							|  |  |  | 		 * or faces very close to the bounding box boundary. */ | 
					
						
							|  |  |  | 		mid_v3_v3v3(bb_center, bb_max_root, bb_min_root); | 
					
						
							|  |  |  | 		/* diff should be same for both min/max since it's calculated from center */ | 
					
						
							|  |  |  | 		sub_v3_v3v3(bb_diff, bb_max_root, bb_center); | 
					
						
							|  |  |  | 		/* handles case of zero width bb */ | 
					
						
							|  |  |  | 		add_v3_v3(bb_diff, offset_vec); | 
					
						
							|  |  |  | 		madd_v3_v3v3fl(bb_max_root, bb_center, bb_diff, offset); | 
					
						
							|  |  |  | 		madd_v3_v3v3fl(bb_min_root, bb_center, bb_diff, -offset); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* first project start ray */ | 
					
						
							|  |  |  | 		isect_ray_aabb_initialize(&ray, ray_start, ray_normal); | 
					
						
							|  |  |  | 		if (!isect_ray_aabb(&ray, bb_min_root, bb_max_root, &rootmin_start)) | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* then the end ray */ | 
					
						
							|  |  |  | 		mul_v3_v3fl(ray_normal_inv, ray_normal, -1.0); | 
					
						
							|  |  |  | 		isect_ray_aabb_initialize(&ray, ray_end, ray_normal_inv); | 
					
						
							|  |  |  | 		/* unlikely to fail exiting if entering succeeded, still keep this here */ | 
					
						
							|  |  |  | 		if (!isect_ray_aabb(&ray, bb_min_root, bb_max_root, &rootmin_end)) | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		madd_v3_v3v3fl(ray_start, ray_start, ray_normal, rootmin_start); | 
					
						
							|  |  |  | 		madd_v3_v3v3fl(ray_end, ray_end, ray_normal_inv, rootmin_end); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-09 06:53:39 +00:00
										 |  |  | //#include <GL/glew.h>
 | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:30:13 +00:00
										 |  |  | typedef struct { | 
					
						
							|  |  |  | 	DMSetMaterial setMaterial; | 
					
						
							|  |  |  | 	int wireframe; | 
					
						
							|  |  |  | } PBVHNodeDrawData; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | void BKE_pbvh_node_draw(PBVHNode *node, void *data_v) | 
					
						
							| 
									
										
										
										
											2009-12-11 14:16:17 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-12-30 18:30:13 +00:00
										 |  |  | 	PBVHNodeDrawData *data = data_v; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-11 14:16:17 +00:00
										 |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 	/* XXX: Just some quick code to show leaf nodes in different colors */ | 
					
						
							| 
									
										
										
										
											2009-12-11 14:16:17 +00:00
										 |  |  | 	float col[3]; int i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (0) { //is_partial) {
 | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 		col[0] = (rand() / (float)RAND_MAX); col[1] = col[2] = 0.6; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2009-12-11 14:16:17 +00:00
										 |  |  | 		srand((long long)node); | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		for (i = 0; i < 3; ++i) | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 			col[i] = (rand() / (float)RAND_MAX) * 0.3 + 0.7; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	glMaterialfv(GL_FRONT_AND_BACK, GL_DIFFUSE, col); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-11 14:16:17 +00:00
										 |  |  | 	glColor3f(1, 0, 0); | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2012-03-14 06:32:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-31 22:35:25 +00:00
										 |  |  | 	if (!(node->flag & PBVH_FullyHidden)) { | 
					
						
							| 
									
										
										
										
											2014-01-02 22:22:36 +02:00
										 |  |  | 		GPU_draw_pbvh_buffers(node->draw_buffers, | 
					
						
							| 
									
										
										
										
											2013-05-31 22:35:25 +00:00
										 |  |  | 		                 data->setMaterial, | 
					
						
							|  |  |  | 		                 data->wireframe); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-14 06:32:25 +00:00
										 |  |  | typedef enum { | 
					
						
							|  |  |  | 	ISECT_INSIDE, | 
					
						
							|  |  |  | 	ISECT_OUTSIDE, | 
					
						
							|  |  |  | 	ISECT_INTERSECT | 
					
						
							|  |  |  | } PlaneAABBIsect; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | /* Adapted from:
 | 
					
						
							| 
									
										
										
										
											2012-03-03 20:19:11 +00:00
										 |  |  |  * http://www.gamedev.net/community/forums/topic.asp?topic_id=512123
 | 
					
						
							|  |  |  |  * Returns true if the AABB is at least partially within the frustum | 
					
						
							|  |  |  |  * (ok, not a real frustum), false otherwise. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2012-03-14 06:32:25 +00:00
										 |  |  | static PlaneAABBIsect test_planes_aabb(const float bb_min[3], | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  |                                        const float bb_max[3], | 
					
						
							|  |  |  |                                        const float (*planes)[4]) | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-14 06:32:25 +00:00
										 |  |  | 	float vmin[3], vmax[3]; | 
					
						
							|  |  |  | 	PlaneAABBIsect ret = ISECT_INSIDE; | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 	int i, axis; | 
					
						
							| 
									
										
										
										
											2012-03-14 06:32:25 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-10-21 05:46:41 +00:00
										 |  |  | 	for (i = 0; i < 4; ++i) { | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		for (axis = 0; axis < 3; ++axis) { | 
					
						
							| 
									
										
										
										
											2012-10-21 05:46:41 +00:00
										 |  |  | 			if (planes[i][axis] > 0) { | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 				vmin[axis] = bb_min[axis]; | 
					
						
							| 
									
										
										
										
											2012-03-14 06:32:25 +00:00
										 |  |  | 				vmax[axis] = bb_max[axis]; | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			else { | 
					
						
							|  |  |  | 				vmin[axis] = bb_max[axis]; | 
					
						
							| 
									
										
										
										
											2012-03-14 06:32:25 +00:00
										 |  |  | 				vmax[axis] = bb_min[axis]; | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (dot_v3v3(planes[i], vmin) + planes[i][3] > 0) | 
					
						
							| 
									
										
										
										
											2012-03-14 06:32:25 +00:00
										 |  |  | 			return ISECT_OUTSIDE; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		else if (dot_v3v3(planes[i], vmax) + planes[i][3] >= 0) | 
					
						
							| 
									
										
										
										
											2012-03-14 06:32:25 +00:00
										 |  |  | 			ret = ISECT_INTERSECT; | 
					
						
							| 
									
										
										
										
											2012-10-21 05:46:41 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-14 06:32:25 +00:00
										 |  |  | 	return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-03 18:55:59 +11:00
										 |  |  | bool BKE_pbvh_node_planes_contain_AABB(PBVHNode *node, void *data) | 
					
						
							| 
									
										
										
										
											2012-03-14 06:32:25 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	float bb_min[3], bb_max[3]; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | 	BKE_pbvh_node_get_BB(node, bb_min, bb_max); | 
					
						
							| 
									
										
										
										
											2012-03-14 06:32:25 +00:00
										 |  |  | 	return test_planes_aabb(bb_min, bb_max, data) != ISECT_OUTSIDE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-03 18:55:59 +11:00
										 |  |  | bool BKE_pbvh_node_planes_exclude_AABB(PBVHNode *node, void *data) | 
					
						
							| 
									
										
										
										
											2012-03-14 06:32:25 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	float bb_min[3], bb_max[3]; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | 	BKE_pbvh_node_get_BB(node, bb_min, bb_max); | 
					
						
							| 
									
										
										
										
											2012-03-14 06:32:25 +00:00
										 |  |  | 	return test_planes_aabb(bb_min, bb_max, data) != ISECT_INSIDE; | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-22 17:33:53 +00:00
										 |  |  | static void pbvh_node_check_diffuse_changed(PBVH *bvh, PBVHNode *node) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (!node->draw_buffers) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-02 22:22:36 +02:00
										 |  |  | 	if (GPU_pbvh_buffers_diffuse_changed(node->draw_buffers, bvh->show_diffuse_color)) | 
					
						
							| 
									
										
										
										
											2012-10-22 17:33:53 +00:00
										 |  |  | 		node->flag |= PBVH_UpdateDrawBuffers; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | void BKE_pbvh_draw(PBVH *bvh, float (*planes)[4], float (*face_nors)[3], | 
					
						
							| 
									
										
										
										
											2012-12-30 18:30:13 +00:00
										 |  |  |                    DMSetMaterial setMaterial, int wireframe) | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-12-30 18:30:13 +00:00
										 |  |  | 	PBVHNodeDrawData draw_data = {setMaterial, wireframe}; | 
					
						
							| 
									
										
										
										
											2010-03-22 17:17:36 +00:00
										 |  |  | 	PBVHNode **nodes; | 
					
						
							| 
									
										
										
										
											2012-10-22 17:33:53 +00:00
										 |  |  | 	int a, totnode; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (a = 0; a < bvh->totnode; a++) | 
					
						
							|  |  |  | 		pbvh_node_check_diffuse_changed(bvh, &bvh->nodes[a]); | 
					
						
							| 
									
										
										
										
											2010-03-22 17:17:36 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | 	BKE_pbvh_search_gather(bvh, update_search_cb, SET_INT_IN_POINTER(PBVH_UpdateNormals | PBVH_UpdateDrawBuffers), | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	                       &nodes, &totnode); | 
					
						
							| 
									
										
										
										
											2010-03-22 17:17:36 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	pbvh_update_normals(bvh, nodes, totnode, face_nors); | 
					
						
							| 
									
										
										
										
											2012-03-06 02:40:08 +00:00
										 |  |  | 	pbvh_update_draw_buffers(bvh, nodes, totnode); | 
					
						
							| 
									
										
										
										
											2010-03-22 17:17:36 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (nodes) MEM_freeN(nodes); | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (planes) { | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | 		BKE_pbvh_search_callback(bvh, BKE_pbvh_node_planes_contain_AABB, | 
					
						
							|  |  |  | 		                         planes, BKE_pbvh_node_draw, &draw_data); | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | 		BKE_pbvh_search_callback(bvh, NULL, NULL, BKE_pbvh_node_draw, &draw_data); | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | void BKE_pbvh_grids_update(PBVH *bvh, CCGElem **grids, DMGridAdjacency *gridadj, void **gridfaces, | 
					
						
							| 
									
										
										
										
											2013-07-22 23:20:48 +00:00
										 |  |  |                            DMFlagMat *flagmats, BLI_bitmap **grid_hidden) | 
					
						
							| 
									
										
										
										
											2010-06-02 18:04:31 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-11-08 14:22:05 +00:00
										 |  |  | 	int a; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	bvh->grids = grids; | 
					
						
							|  |  |  | 	bvh->gridadj = gridadj; | 
					
						
							|  |  |  | 	bvh->gridfaces = gridfaces; | 
					
						
							| 
									
										
										
										
											2012-11-08 14:22:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-14 08:05:40 +00:00
										 |  |  | 	if (flagmats != bvh->grid_flag_mats || bvh->grid_hidden != grid_hidden) { | 
					
						
							|  |  |  | 		bvh->grid_flag_mats = flagmats; | 
					
						
							|  |  |  | 		bvh->grid_hidden = grid_hidden; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		for (a = 0; a < bvh->totnode; ++a) | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | 			BKE_pbvh_node_mark_rebuild_draw(&bvh->nodes[a]); | 
					
						
							| 
									
										
										
										
											2012-11-14 08:05:40 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2010-06-02 18:04:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:26:11 +00:00
										 |  |  | /* Get the node's displacement layer, creating it if necessary */ | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | float *BKE_pbvh_node_layer_disp_get(PBVH *bvh, PBVHNode *node) | 
					
						
							| 
									
										
										
										
											2012-12-30 18:26:11 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	if (!node->layer_disp) { | 
					
						
							|  |  |  | 		int totvert = 0; | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | 		BKE_pbvh_node_num_verts(bvh, node, &totvert, NULL); | 
					
						
							| 
									
										
										
										
											2012-12-30 18:26:11 +00:00
										 |  |  | 		node->layer_disp = MEM_callocN(sizeof(float) * totvert, "layer disp"); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return node->layer_disp; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* If the node has a displacement layer, free it and set to null */ | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | void BKE_pbvh_node_layer_disp_free(PBVHNode *node) | 
					
						
							| 
									
										
										
										
											2012-12-30 18:26:11 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	if (node->layer_disp) { | 
					
						
							|  |  |  | 		MEM_freeN(node->layer_disp); | 
					
						
							|  |  |  | 		node->layer_disp = NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-26 07:29:01 +00:00
										 |  |  | float (*BKE_pbvh_get_vertCos(PBVH *pbvh))[3] | 
					
						
							| 
									
										
										
										
											2010-06-21 20:10:59 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	int a; | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	float (*vertCos)[3] = NULL; | 
					
						
							| 
									
										
										
										
											2010-06-21 20:10:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (pbvh->verts) { | 
					
						
							|  |  |  | 		float *co; | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		MVert *mvert = pbvh->verts; | 
					
						
							| 
									
										
										
										
											2010-06-21 20:10:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | 		vertCos = MEM_callocN(3 * pbvh->totvert * sizeof(float), "BKE_pbvh_get_vertCoords"); | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		co = (float *)vertCos; | 
					
						
							| 
									
										
										
										
											2010-06-21 20:10:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		for (a = 0; a < pbvh->totvert; a++, mvert++, co += 3) { | 
					
						
							| 
									
										
										
										
											2010-06-21 20:10:59 +00:00
										 |  |  | 			copy_v3_v3(co, mvert->co); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return vertCos; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | void BKE_pbvh_apply_vertCos(PBVH *pbvh, float (*vertCos)[3]) | 
					
						
							| 
									
										
										
										
											2010-06-21 20:10:59 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	int a; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!pbvh->deformed) { | 
					
						
							|  |  |  | 		if (pbvh->verts) { | 
					
						
							|  |  |  | 			/* if pbvh is not already deformed, verts/faces points to the */ | 
					
						
							|  |  |  | 			/* original data and applying new coords to this arrays would lead to */ | 
					
						
							|  |  |  | 			/* unneeded deformation -- duplicate verts/faces to avoid this */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 			pbvh->verts = MEM_dupallocN(pbvh->verts); | 
					
						
							|  |  |  | 			pbvh->faces = MEM_dupallocN(pbvh->faces); | 
					
						
							| 
									
										
										
										
											2010-06-21 20:10:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 			pbvh->deformed = 1; | 
					
						
							| 
									
										
										
										
											2010-06-21 20:10:59 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (pbvh->verts) { | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		MVert *mvert = pbvh->verts; | 
					
						
							| 
									
										
										
										
											2010-06-21 20:10:59 +00:00
										 |  |  | 		/* copy new verts coords */ | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		for (a = 0; a < pbvh->totvert; ++a, ++mvert) { | 
					
						
							| 
									
										
										
										
											2011-01-31 20:02:51 +00:00
										 |  |  | 			copy_v3_v3(mvert->co, vertCos[a]); | 
					
						
							|  |  |  | 			mvert->flag |= ME_VERT_PBVH_UPDATE; | 
					
						
							| 
									
										
										
										
											2010-06-21 20:10:59 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* coordinates are new -- normals should also be updated */ | 
					
						
							| 
									
										
										
										
											2012-05-05 21:28:12 +00:00
										 |  |  | 		BKE_mesh_calc_normals_tessface(pbvh->verts, pbvh->totvert, pbvh->faces, pbvh->totprim, NULL); | 
					
						
							| 
									
										
										
										
											2011-01-31 20:02:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		for (a = 0; a < pbvh->totnode; ++a) | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | 			BKE_pbvh_node_mark_update(&pbvh->nodes[a]); | 
					
						
							| 
									
										
										
										
											2011-01-31 20:02:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | 		BKE_pbvh_update(pbvh, PBVH_UpdateBB, NULL); | 
					
						
							|  |  |  | 		BKE_pbvh_update(pbvh, PBVH_UpdateOriginalBB, NULL); | 
					
						
							| 
									
										
										
										
											2011-01-31 20:02:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-21 20:10:59 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-03 18:55:59 +11:00
										 |  |  | bool BKE_pbvh_isDeformed(PBVH *pbvh) | 
					
						
							| 
									
										
										
										
											2010-06-21 20:10:59 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	return pbvh->deformed; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | /* Proxies */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | PBVHProxyNode *BKE_pbvh_node_add_proxy(PBVH *bvh, PBVHNode *node) | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	int index, totverts; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-22 14:11:10 +11:00
										 |  |  | #pragma omp critical
 | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		index = node->proxy_count; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		node->proxy_count++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (node->proxies) | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 			node->proxies = MEM_reallocN(node->proxies, node->proxy_count * sizeof(PBVHProxyNode)); | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | 		else | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 			node->proxies = MEM_mallocN(sizeof(PBVHProxyNode), "PBVHNodeProxy"); | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | 		BKE_pbvh_node_num_verts(bvh, node, &totverts, NULL); | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		node->proxies[index].co = MEM_callocN(sizeof(float[3]) * totverts, "PBVHNodeProxy.co"); | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return node->proxies + index; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | void BKE_pbvh_node_free_proxies(PBVHNode *node) | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-12-22 14:11:10 +11:00
										 |  |  | #pragma omp critical
 | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 		int p; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		for (p = 0; p < node->proxy_count; p++) { | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | 			MEM_freeN(node->proxies[p].co); | 
					
						
							| 
									
										
										
										
											2013-08-07 03:55:21 +00:00
										 |  |  | 			node->proxies[p].co = NULL; | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		MEM_freeN(node->proxies); | 
					
						
							| 
									
										
										
										
											2013-08-07 03:55:21 +00:00
										 |  |  | 		node->proxies = NULL; | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		node->proxy_count = 0; | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | void BKE_pbvh_gather_proxies(PBVH *pbvh, PBVHNode ***r_array,  int *r_tot) | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-08-03 18:05:30 +00:00
										 |  |  | 	PBVHNode **array = NULL, *node; | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	int tot = 0, space = 0; | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | 	int n; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	for (n = 0; n < pbvh->totnode; n++) { | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | 		node = pbvh->nodes + n; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		if (node->proxy_count > 0) { | 
					
						
							|  |  |  | 			if (tot == space) { | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | 				/* resize array if needed */ | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 				space = (tot == 0) ? 32 : space * 2; | 
					
						
							| 
									
										
										
										
											2013-08-03 18:05:30 +00:00
										 |  |  | 				array = MEM_recallocN_id(array, sizeof(PBVHNode *) * space, __func__); | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 			array[tot] = node; | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | 			tot++; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (tot == 0 && array) { | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | 		MEM_freeN(array); | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		array = NULL; | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	*r_array = array; | 
					
						
							|  |  |  | 	*r_tot = tot; | 
					
						
							| 
									
										
										
										
											2010-07-14 14:11:03 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2012-02-22 22:37:01 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | void pbvh_vertex_iter_init(PBVH *bvh, PBVHNode *node, | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  |                            PBVHVertexIter *vi, int mode) | 
					
						
							| 
									
										
										
										
											2012-02-22 22:37:01 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-10 20:33:09 +00:00
										 |  |  | 	struct CCGElem **grids; | 
					
						
							| 
									
										
										
										
											2012-02-22 22:37:01 +00:00
										 |  |  | 	struct MVert *verts; | 
					
						
							|  |  |  | 	int *grid_indices, *vert_indices; | 
					
						
							|  |  |  | 	int totgrid, gridsize, uniq_verts, totvert; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2013-08-07 03:55:21 +00:00
										 |  |  | 	vi->grid = NULL; | 
					
						
							|  |  |  | 	vi->no = NULL; | 
					
						
							|  |  |  | 	vi->fno = NULL; | 
					
						
							|  |  |  | 	vi->mvert = NULL; | 
					
						
							| 
									
										
										
										
											2012-02-22 22:37:01 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:31:31 +00:00
										 |  |  | 	BKE_pbvh_node_get_grids(bvh, node, &grid_indices, &totgrid, NULL, &gridsize, &grids, NULL); | 
					
						
							|  |  |  | 	BKE_pbvh_node_num_verts(bvh, node, &uniq_verts, &totvert); | 
					
						
							|  |  |  | 	BKE_pbvh_node_get_verts(bvh, node, &vert_indices, &verts); | 
					
						
							| 
									
										
										
										
											2012-05-10 20:33:09 +00:00
										 |  |  | 	vi->key = &bvh->gridkey; | 
					
						
							| 
									
										
										
										
											2012-02-22 22:37:01 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	vi->grids = grids; | 
					
						
							|  |  |  | 	vi->grid_indices = grid_indices; | 
					
						
							|  |  |  | 	vi->totgrid = (grids) ? totgrid : 1; | 
					
						
							|  |  |  | 	vi->gridsize = gridsize; | 
					
						
							| 
									
										
										
										
											2012-02-22 22:37:01 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (mode == PBVH_ITER_ALL) | 
					
						
							| 
									
										
										
										
											2012-02-22 22:37:01 +00:00
										 |  |  | 		vi->totvert = totvert; | 
					
						
							|  |  |  | 	else | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		vi->totvert = uniq_verts; | 
					
						
							|  |  |  | 	vi->vert_indices = vert_indices; | 
					
						
							|  |  |  | 	vi->mverts = verts; | 
					
						
							| 
									
										
										
										
											2012-03-14 06:32:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:28:36 +00:00
										 |  |  | 	if (bvh->type == PBVH_BMESH) { | 
					
						
							| 
									
										
										
										
											2013-08-25 20:03:45 +00:00
										 |  |  | 		BLI_gsetIterator_init(&vi->bm_unique_verts, node->bm_unique_verts); | 
					
						
							|  |  |  | 		BLI_gsetIterator_init(&vi->bm_other_verts, node->bm_other_verts); | 
					
						
							| 
									
										
										
										
											2012-12-30 18:28:36 +00:00
										 |  |  | 		vi->bm_vdata = &bvh->bm->vdata; | 
					
						
							| 
									
										
										
										
											2013-08-27 20:39:08 +00:00
										 |  |  | 		vi->cd_vert_mask_offset = CustomData_get_offset(vi->bm_vdata, CD_PAINT_MASK); | 
					
						
							| 
									
										
										
										
											2012-12-30 18:28:36 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	vi->gh = NULL; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	if (vi->grids && mode == PBVH_ITER_UNIQUE) | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		vi->grid_hidden = bvh->grid_hidden; | 
					
						
							| 
									
										
										
										
											2012-05-10 20:33:45 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 	vi->mask = NULL; | 
					
						
							| 
									
										
										
										
											2012-12-30 18:28:36 +00:00
										 |  |  | 	if (bvh->type == PBVH_FACES) | 
					
						
							| 
									
										
										
										
											2012-05-11 08:05:47 +00:00
										 |  |  | 		vi->vmask = CustomData_get_layer(bvh->vdata, CD_PAINT_MASK); | 
					
						
							| 
									
										
										
										
											2012-02-22 22:37:01 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2012-10-22 17:33:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-03 18:55:59 +11:00
										 |  |  | void pbvh_show_diffuse_color_set(PBVH *bvh, bool show_diffuse_color) | 
					
						
							| 
									
										
										
										
											2012-10-22 17:33:53 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	bvh->show_diffuse_color = show_diffuse_color; | 
					
						
							|  |  |  | } |