| 
									
										
										
										
											2011-02-23 10:52:22 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2009-10-03 15:35:01 +00:00
										 |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU General Public License | 
					
						
							|  |  |  |  * as published by the Free Software Foundation; either version 2 | 
					
						
							| 
									
										
										
										
											2011-11-29 10:54:47 +00:00
										 |  |  |  * of the License, or (at your option) any later version. | 
					
						
							| 
									
										
										
										
											2009-10-03 15:35:01 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * 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. | 
					
						
							| 
									
										
										
										
											2009-10-03 15:35:01 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * The Original Code is Copyright (C) 2005 Blender Foundation. | 
					
						
							|  |  |  |  * All rights reserved. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-18 08:08:12 +11:00
										 |  |  | /** \file
 | 
					
						
							|  |  |  |  * \ingroup gpu | 
					
						
							| 
									
										
										
										
											2014-01-19 23:14:24 +11:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2015-11-25 19:49:31 -05:00
										 |  |  |  * Mesh drawing using OpenGL VBO (Vertex Buffer Objects) | 
					
						
							| 
									
										
										
										
											2011-02-27 20:25:53 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-25 13:40:43 +00:00
										 |  |  | #include <limits.h>
 | 
					
						
							| 
									
										
										
										
											2009-11-25 18:20:46 +00:00
										 |  |  | #include <stddef.h>
 | 
					
						
							| 
									
										
										
										
											2009-10-03 15:35:01 +00:00
										 |  |  | #include <string.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							| 
									
										
										
										
											2009-10-05 16:48:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-14 06:32:25 +00:00
										 |  |  | #include "BLI_bitmap.h"
 | 
					
						
							| 
									
										
										
										
											2010-07-13 13:31:43 +00:00
										 |  |  | #include "BLI_math.h"
 | 
					
						
							| 
									
										
										
										
											2011-01-07 18:36:47 +00:00
										 |  |  | #include "BLI_utildefines.h"
 | 
					
						
							|  |  |  | #include "BLI_ghash.h"
 | 
					
						
							| 
									
										
										
										
											2009-10-05 16:48:52 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "DNA_meshdata_types.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-10 20:33:09 +00:00
										 |  |  | #include "BKE_ccg.h"
 | 
					
						
							| 
									
										
										
										
											2009-10-03 15:35:01 +00:00
										 |  |  | #include "BKE_DerivedMesh.h"
 | 
					
						
							| 
									
										
										
										
											2012-03-14 06:32:25 +00:00
										 |  |  | #include "BKE_paint.h"
 | 
					
						
							| 
									
										
										
										
											2015-07-31 21:04:23 +10:00
										 |  |  | #include "BKE_mesh.h"
 | 
					
						
							| 
									
										
										
										
											2014-05-05 22:21:30 +03:00
										 |  |  | #include "BKE_pbvh.h"
 | 
					
						
							| 
									
										
										
										
											2009-10-05 16:48:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-14 10:46:12 +00:00
										 |  |  | #include "GPU_buffers.h"
 | 
					
						
							| 
									
										
										
										
											2017-05-10 20:14:52 +10:00
										 |  |  | #include "GPU_batch.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-27 22:42:57 +02:00
										 |  |  | #include "gpu_private.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:24:54 +00:00
										 |  |  | #include "bmesh.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												== GPU Buffers ==
This patch attempts to clean up and document the GPU buffers
code. There are a few bug fixes as well.
Patch reviewed here: http://codereview.appspot.com/4631052/
Summary:
* Bugfix: make GPU_buffer_copy_normal convert from shorts to floats
  correctly, also fixed the use of cached face normal CustomData.
* Bugfix: changed the `mat_nr' field of GPUBufferMaterial from char to
  short.
* Changed color buffer setup to not alloc a temporary copy of color
  data, just passes the MCol data in directly.
* Changed the GPU buffer pool code to make clearer what operates
  specifically on the global pool.
* Lots of refactoring for GPU_drawobject_new; should operate mostly
  the same (except got rid of one unecessary allocation), just split
  into more functions and without macros now.
* Converted some #defines into enumerations.
* Made some stuff private, pulled out of header file.
* Deleted unused function GPU_buffer_pool_free_unused().
* Removed GPU_interleaved_setup and related #defines. (I think this
  was used for editmode VBOs, but those were disabled.)
* Added lots of comments.
* Added a few comments in the code signed `--nicholas' to note places
  where I am unsure about design or usage, would be good to address
  these better.
* Code formatting changed to be more consistent with the rest of
  Blender.
* Renamed some fields and variables to be more consistent with
  Blender's naming conventions.
* Renamed some fields and variables to use more descriptive names,
  e.g. renamed `redir' to `mat_orig_to_new'.
* Removed print outs with DEBUG_VBO -- don't feel too strongly about
  this one, just not used elsewhere in Blender, could be easily added
  back if others disagree though.
* Moved the PBVH drawing code down to the bottom of the file, before
  was sitting in the middle of the other VBO code
											
										 
											2011-07-08 19:58:02 +00:00
										 |  |  | /* XXX: the rest of the code in this file is used for optimized PBVH
 | 
					
						
							| 
									
										
										
										
											2012-04-22 11:54:53 +00:00
										 |  |  |  * drawing and doesn't interact at all with the buffer code above */ | 
					
						
							| 
									
										
											  
											
												== GPU Buffers ==
This patch attempts to clean up and document the GPU buffers
code. There are a few bug fixes as well.
Patch reviewed here: http://codereview.appspot.com/4631052/
Summary:
* Bugfix: make GPU_buffer_copy_normal convert from shorts to floats
  correctly, also fixed the use of cached face normal CustomData.
* Bugfix: changed the `mat_nr' field of GPUBufferMaterial from char to
  short.
* Changed color buffer setup to not alloc a temporary copy of color
  data, just passes the MCol data in directly.
* Changed the GPU buffer pool code to make clearer what operates
  specifically on the global pool.
* Lots of refactoring for GPU_drawobject_new; should operate mostly
  the same (except got rid of one unecessary allocation), just split
  into more functions and without macros now.
* Converted some #defines into enumerations.
* Made some stuff private, pulled out of header file.
* Deleted unused function GPU_buffer_pool_free_unused().
* Removed GPU_interleaved_setup and related #defines. (I think this
  was used for editmode VBOs, but those were disabled.)
* Added lots of comments.
* Added a few comments in the code signed `--nicholas' to note places
  where I am unsure about design or usage, would be good to address
  these better.
* Code formatting changed to be more consistent with the rest of
  Blender.
* Renamed some fields and variables to be more consistent with
  Blender's naming conventions.
* Renamed some fields and variables to use more descriptive names,
  e.g. renamed `redir' to `mat_orig_to_new'.
* Removed print outs with DEBUG_VBO -- don't feel too strongly about
  this one, just not used elsewhere in Blender, could be easily added
  back if others disagree though.
* Moved the PBVH drawing code down to the bottom of the file, before
  was sitting in the middle of the other VBO code
											
										 
											2011-07-08 19:58:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-02 22:22:36 +02:00
										 |  |  | struct GPU_PBVH_Buffers { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   GPUIndexBuf *index_buf, *index_buf_fast; | 
					
						
							|  |  |  |   GPUIndexBuf *index_lines_buf, *index_lines_buf_fast; | 
					
						
							|  |  |  |   GPUVertBuf *vert_buf; | 
					
						
							| 
									
										
											  
											
												== GPU Buffers ==
This patch attempts to clean up and document the GPU buffers
code. There are a few bug fixes as well.
Patch reviewed here: http://codereview.appspot.com/4631052/
Summary:
* Bugfix: make GPU_buffer_copy_normal convert from shorts to floats
  correctly, also fixed the use of cached face normal CustomData.
* Bugfix: changed the `mat_nr' field of GPUBufferMaterial from char to
  short.
* Changed color buffer setup to not alloc a temporary copy of color
  data, just passes the MCol data in directly.
* Changed the GPU buffer pool code to make clearer what operates
  specifically on the global pool.
* Lots of refactoring for GPU_drawobject_new; should operate mostly
  the same (except got rid of one unecessary allocation), just split
  into more functions and without macros now.
* Converted some #defines into enumerations.
* Made some stuff private, pulled out of header file.
* Deleted unused function GPU_buffer_pool_free_unused().
* Removed GPU_interleaved_setup and related #defines. (I think this
  was used for editmode VBOs, but those were disabled.)
* Added lots of comments.
* Added a few comments in the code signed `--nicholas' to note places
  where I am unsure about design or usage, would be good to address
  these better.
* Code formatting changed to be more consistent with the rest of
  Blender.
* Renamed some fields and variables to be more consistent with
  Blender's naming conventions.
* Renamed some fields and variables to use more descriptive names,
  e.g. renamed `redir' to `mat_orig_to_new'.
* Removed print outs with DEBUG_VBO -- don't feel too strongly about
  this one, just not used elsewhere in Blender, could be easily added
  back if others disagree though.
* Moved the PBVH drawing code down to the bottom of the file, before
  was sitting in the middle of the other VBO code
											
										 
											2011-07-08 19:58:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   GPUBatch *lines; | 
					
						
							|  |  |  |   GPUBatch *lines_fast; | 
					
						
							|  |  |  |   GPUBatch *triangles; | 
					
						
							|  |  |  |   GPUBatch *triangles_fast; | 
					
						
							| 
									
										
										
										
											2015-08-03 19:08:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* mesh pointers in case buffer allocation fails */ | 
					
						
							|  |  |  |   const MPoly *mpoly; | 
					
						
							|  |  |  |   const MLoop *mloop; | 
					
						
							|  |  |  |   const MLoopTri *looptri; | 
					
						
							|  |  |  |   const MVert *mvert; | 
					
						
							| 
									
										
										
										
											2015-07-17 03:36:03 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const int *face_indices; | 
					
						
							|  |  |  |   int face_indices_len; | 
					
						
							| 
									
										
											  
											
												== GPU Buffers ==
This patch attempts to clean up and document the GPU buffers
code. There are a few bug fixes as well.
Patch reviewed here: http://codereview.appspot.com/4631052/
Summary:
* Bugfix: make GPU_buffer_copy_normal convert from shorts to floats
  correctly, also fixed the use of cached face normal CustomData.
* Bugfix: changed the `mat_nr' field of GPUBufferMaterial from char to
  short.
* Changed color buffer setup to not alloc a temporary copy of color
  data, just passes the MCol data in directly.
* Changed the GPU buffer pool code to make clearer what operates
  specifically on the global pool.
* Lots of refactoring for GPU_drawobject_new; should operate mostly
  the same (except got rid of one unecessary allocation), just split
  into more functions and without macros now.
* Converted some #defines into enumerations.
* Made some stuff private, pulled out of header file.
* Deleted unused function GPU_buffer_pool_free_unused().
* Removed GPU_interleaved_setup and related #defines. (I think this
  was used for editmode VBOs, but those were disabled.)
* Added lots of comments.
* Added a few comments in the code signed `--nicholas' to note places
  where I am unsure about design or usage, would be good to address
  these better.
* Code formatting changed to be more consistent with the rest of
  Blender.
* Renamed some fields and variables to be more consistent with
  Blender's naming conventions.
* Renamed some fields and variables to use more descriptive names,
  e.g. renamed `redir' to `mat_orig_to_new'.
* Removed print outs with DEBUG_VBO -- don't feel too strongly about
  this one, just not used elsewhere in Blender, could be easily added
  back if others disagree though.
* Moved the PBVH drawing code down to the bottom of the file, before
  was sitting in the middle of the other VBO code
											
										 
											2011-07-08 19:58:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* grid pointers */ | 
					
						
							|  |  |  |   CCGKey gridkey; | 
					
						
							|  |  |  |   CCGElem **grids; | 
					
						
							|  |  |  |   const DMFlagMat *grid_flag_mats; | 
					
						
							|  |  |  |   BLI_bitmap *const *grid_hidden; | 
					
						
							|  |  |  |   const int *grid_indices; | 
					
						
							|  |  |  |   int totgrid; | 
					
						
							| 
									
										
											  
											
												== GPU Buffers ==
This patch attempts to clean up and document the GPU buffers
code. There are a few bug fixes as well.
Patch reviewed here: http://codereview.appspot.com/4631052/
Summary:
* Bugfix: make GPU_buffer_copy_normal convert from shorts to floats
  correctly, also fixed the use of cached face normal CustomData.
* Bugfix: changed the `mat_nr' field of GPUBufferMaterial from char to
  short.
* Changed color buffer setup to not alloc a temporary copy of color
  data, just passes the MCol data in directly.
* Changed the GPU buffer pool code to make clearer what operates
  specifically on the global pool.
* Lots of refactoring for GPU_drawobject_new; should operate mostly
  the same (except got rid of one unecessary allocation), just split
  into more functions and without macros now.
* Converted some #defines into enumerations.
* Made some stuff private, pulled out of header file.
* Deleted unused function GPU_buffer_pool_free_unused().
* Removed GPU_interleaved_setup and related #defines. (I think this
  was used for editmode VBOs, but those were disabled.)
* Added lots of comments.
* Added a few comments in the code signed `--nicholas' to note places
  where I am unsure about design or usage, would be good to address
  these better.
* Code formatting changed to be more consistent with the rest of
  Blender.
* Renamed some fields and variables to be more consistent with
  Blender's naming conventions.
* Renamed some fields and variables to use more descriptive names,
  e.g. renamed `redir' to `mat_orig_to_new'.
* Removed print outs with DEBUG_VBO -- don't feel too strongly about
  this one, just not used elsewhere in Blender, could be easily added
  back if others disagree though.
* Moved the PBVH drawing code down to the bottom of the file, before
  was sitting in the middle of the other VBO code
											
										 
											2011-07-08 19:58:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   bool use_bmesh; | 
					
						
							| 
									
										
										
										
											2019-09-27 22:42:57 +02:00
										 |  |  |   bool clear_bmesh_on_flush; | 
					
						
							| 
									
										
										
										
											2012-12-30 18:24:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   uint tot_tri, tot_quad; | 
					
						
							| 
									
										
										
										
											2012-10-06 16:42:11 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-04 00:46:26 +02:00
										 |  |  |   short material_index; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* The PBVH ensures that either all faces in the node are
 | 
					
						
							|  |  |  |    * smooth-shaded or all faces are flat-shaded */ | 
					
						
							|  |  |  |   bool smooth; | 
					
						
							| 
									
										
										
										
											2012-10-22 17:33:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   bool show_mask; | 
					
						
							| 
									
										
										
										
											2011-11-05 03:29:37 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2015-03-23 15:29:42 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-19 18:09:00 -03:00
										 |  |  | static struct { | 
					
						
							| 
									
										
										
										
											2019-09-27 22:42:57 +02:00
										 |  |  |   GPUVertFormat format; | 
					
						
							| 
									
										
										
										
											2019-04-16 14:00:16 +02:00
										 |  |  |   uint pos, nor, msk, col; | 
					
						
							| 
									
										
										
										
											2019-09-27 22:42:57 +02:00
										 |  |  | } g_vbo_id = {{0}}; | 
					
						
							| 
									
										
										
										
											2017-05-10 20:14:52 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-14 20:24:13 +01:00
										 |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name PBVH Utils
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-27 22:42:57 +02:00
										 |  |  | void gpu_pbvh_init() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   /* Initialize vertex buffer (match 'VertexBufferFormat'). */ | 
					
						
							|  |  |  |   if (g_vbo_id.format.attr_len == 0) { | 
					
						
							|  |  |  |     g_vbo_id.pos = GPU_vertformat_attr_add( | 
					
						
							|  |  |  |         &g_vbo_id.format, "pos", GPU_COMP_F32, 3, GPU_FETCH_FLOAT); | 
					
						
							|  |  |  |     g_vbo_id.nor = GPU_vertformat_attr_add( | 
					
						
							|  |  |  |         &g_vbo_id.format, "nor", GPU_COMP_I16, 3, GPU_FETCH_INT_TO_FLOAT_UNIT); | 
					
						
							|  |  |  |     /* TODO: Do not allocate these `.msk` and `.col` when they are not used. */ | 
					
						
							|  |  |  |     g_vbo_id.msk = GPU_vertformat_attr_add( | 
					
						
							|  |  |  |         &g_vbo_id.format, "msk", GPU_COMP_F32, 1, GPU_FETCH_FLOAT); | 
					
						
							|  |  |  |     g_vbo_id.col = GPU_vertformat_attr_add( | 
					
						
							|  |  |  |         &g_vbo_id.format, "ac", GPU_COMP_U8, 4, GPU_FETCH_INT_TO_FLOAT_UNIT); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void gpu_pbvh_exit() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   /* Nothing to do. */ | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-19 18:09:00 -03:00
										 |  |  | /* Allocates a non-initialized buffer to be sent to GPU.
 | 
					
						
							|  |  |  |  * Return is false it indicates that the memory map failed. */ | 
					
						
							| 
									
										
										
										
											2018-09-12 12:18:35 +10:00
										 |  |  | static bool gpu_pbvh_vert_buf_data_set(GPU_PBVH_Buffers *buffers, uint vert_len) | 
					
						
							| 
									
										
										
										
											2017-05-10 20:14:52 +10:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   /* Keep so we can test #GPU_USAGE_DYNAMIC buffer use.
 | 
					
						
							|  |  |  |    * Not that format initialization match in both blocks. | 
					
						
							|  |  |  |    * Do this to keep braces balanced - otherwise indentation breaks. */ | 
					
						
							|  |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (buffers->vert_buf == NULL) { | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |     /* Initialize vertex buffer (match 'VertexBufferFormat'). */ | 
					
						
							| 
									
										
										
										
											2019-09-27 22:42:57 +02:00
										 |  |  |     buffers->vert_buf = GPU_vertbuf_create_with_format_ex(&g_vbo_id.format, GPU_USAGE_DYNAMIC); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     GPU_vertbuf_data_alloc(buffers->vert_buf, vert_len); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else if (vert_len != buffers->vert_buf->vertex_len) { | 
					
						
							|  |  |  |     GPU_vertbuf_data_resize(buffers->vert_buf, vert_len); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-03-19 18:09:00 -03:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   if (buffers->vert_buf == NULL) { | 
					
						
							|  |  |  |     /* Initialize vertex buffer (match 'VertexBufferFormat'). */ | 
					
						
							| 
									
										
										
										
											2019-09-27 22:42:57 +02:00
										 |  |  |     buffers->vert_buf = GPU_vertbuf_create_with_format_ex(&g_vbo_id.format, GPU_USAGE_STATIC); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							|  |  |  |   GPU_vertbuf_data_alloc(buffers->vert_buf, vert_len); | 
					
						
							| 
									
										
										
										
											2018-03-19 18:09:00 -03:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2019-09-27 22:42:57 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   return buffers->vert_buf->data != NULL; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | static void gpu_pbvh_batch_init(GPU_PBVH_Buffers *buffers, GPUPrimType prim) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (buffers->triangles == NULL) { | 
					
						
							|  |  |  |     buffers->triangles = GPU_batch_create(prim, | 
					
						
							|  |  |  |                                           buffers->vert_buf, | 
					
						
							| 
									
										
										
										
											2019-04-17 01:38:15 +02:00
										 |  |  |                                           /* can be NULL if buffer is empty */ | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |                                           buffers->index_buf); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   if ((buffers->triangles_fast == NULL) && buffers->index_buf_fast) { | 
					
						
							|  |  |  |     buffers->triangles_fast = GPU_batch_create(prim, buffers->vert_buf, buffers->index_buf_fast); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   if (buffers->lines == NULL) { | 
					
						
							| 
									
										
										
										
											2019-04-17 01:38:15 +02:00
										 |  |  |     buffers->lines = GPU_batch_create(GPU_PRIM_LINES, | 
					
						
							|  |  |  |                                       buffers->vert_buf, | 
					
						
							|  |  |  |                                       /* can be NULL if buffer is empty */ | 
					
						
							|  |  |  |                                       buffers->index_lines_buf); | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   if ((buffers->lines_fast == NULL) && buffers->index_lines_buf_fast) { | 
					
						
							|  |  |  |     buffers->lines_fast = GPU_batch_create( | 
					
						
							|  |  |  |         GPU_PRIM_LINES, buffers->vert_buf, buffers->index_lines_buf_fast); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | /** \} */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Mesh PBVH
 | 
					
						
							| 
									
										
										
										
											2019-02-14 20:24:13 +01:00
										 |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-27 22:42:57 +02:00
										 |  |  | /* Threaded - do not call any functions that use OpenGL calls! */ | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | void GPU_pbvh_mesh_buffers_update(GPU_PBVH_Buffers *buffers, | 
					
						
							|  |  |  |                                   const MVert *mvert, | 
					
						
							|  |  |  |                                   const int *vert_indices, | 
					
						
							|  |  |  |                                   int totvert, | 
					
						
							|  |  |  |                                   const float *vmask, | 
					
						
							| 
									
										
										
										
											2019-04-16 14:00:16 +02:00
										 |  |  |                                   const MLoopCol *vcol, | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |                                   const int (*face_vert_indices)[3], | 
					
						
							|  |  |  |                                   const int update_flags) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-09-28 19:15:06 +02:00
										 |  |  |   const bool show_mask = vmask && (update_flags & GPU_PBVH_BUFFERS_SHOW_MASK) != 0; | 
					
						
							|  |  |  |   const bool show_vcol = vcol && (update_flags & GPU_PBVH_BUFFERS_SHOW_VCOL) != 0; | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   bool empty_mask = true; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |     int totelem = (buffers->smooth ? totvert : (buffers->tot_tri * 3)); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |     /* Build VBO */ | 
					
						
							|  |  |  |     if (gpu_pbvh_vert_buf_data_set(buffers, totelem)) { | 
					
						
							| 
									
										
										
										
											2019-09-28 19:15:06 +02:00
										 |  |  |       GPUVertBufRaw pos_step = {0}; | 
					
						
							|  |  |  |       GPUVertBufRaw nor_step = {0}; | 
					
						
							|  |  |  |       GPUVertBufRaw msk_step = {0}; | 
					
						
							|  |  |  |       GPUVertBufRaw col_step = {0}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       GPU_vertbuf_attr_get_raw_data(buffers->vert_buf, g_vbo_id.pos, &pos_step); | 
					
						
							|  |  |  |       GPU_vertbuf_attr_get_raw_data(buffers->vert_buf, g_vbo_id.nor, &nor_step); | 
					
						
							|  |  |  |       if (show_mask) { | 
					
						
							|  |  |  |         GPU_vertbuf_attr_get_raw_data(buffers->vert_buf, g_vbo_id.msk, &msk_step); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (show_vcol) { | 
					
						
							|  |  |  |         GPU_vertbuf_attr_get_raw_data(buffers->vert_buf, g_vbo_id.col, &col_step); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       /* Vertex data is shared if smooth-shaded, but separate
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |        * copies are made for flat shading because normals | 
					
						
							|  |  |  |        * shouldn't be shared. */ | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       if (buffers->smooth) { | 
					
						
							| 
									
										
										
										
											2019-09-08 00:12:26 +10:00
										 |  |  |         for (uint i = 0; i < totvert; i++) { | 
					
						
							| 
									
										
										
										
											2019-09-28 19:15:06 +02:00
										 |  |  |           const int vidx = vert_indices[i]; | 
					
						
							|  |  |  |           const MVert *v = &mvert[vidx]; | 
					
						
							|  |  |  |           copy_v3_v3(GPU_vertbuf_raw_step(&pos_step), v->co); | 
					
						
							|  |  |  |           copy_v3_v3_short(GPU_vertbuf_raw_step(&nor_step), v->no); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           if (show_mask) { | 
					
						
							|  |  |  |             float mask = vmask[vidx]; | 
					
						
							|  |  |  |             *(float *)GPU_vertbuf_raw_step(&msk_step) = mask; | 
					
						
							|  |  |  |             empty_mask = empty_mask && (mask == 0.0f); | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-16 14:00:16 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-28 19:15:06 +02:00
										 |  |  |         if (show_vcol) { | 
					
						
							| 
									
										
										
										
											2019-04-16 14:00:16 +02:00
										 |  |  |           for (uint i = 0; i < buffers->face_indices_len; i++) { | 
					
						
							|  |  |  |             const MLoopTri *lt = &buffers->looptri[buffers->face_indices[i]]; | 
					
						
							|  |  |  |             for (int j = 0; j < 3; j++) { | 
					
						
							|  |  |  |               const int loop_index = lt->tri[j]; | 
					
						
							|  |  |  |               const int vidx = face_vert_indices[i][j]; | 
					
						
							|  |  |  |               const uchar *elem = &vcol[loop_index].r; | 
					
						
							|  |  |  |               GPU_vertbuf_attr_set(buffers->vert_buf, g_vbo_id.col, vidx, elem); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         /* calculate normal for each polygon only once */ | 
					
						
							|  |  |  |         uint mpoly_prev = UINT_MAX; | 
					
						
							|  |  |  |         short no[3]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (uint i = 0; i < buffers->face_indices_len; i++) { | 
					
						
							|  |  |  |           const MLoopTri *lt = &buffers->looptri[buffers->face_indices[i]]; | 
					
						
							|  |  |  |           const uint vtri[3] = { | 
					
						
							|  |  |  |               buffers->mloop[lt->tri[0]].v, | 
					
						
							|  |  |  |               buffers->mloop[lt->tri[1]].v, | 
					
						
							|  |  |  |               buffers->mloop[lt->tri[2]].v, | 
					
						
							|  |  |  |           }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-22 09:32:37 +10:00
										 |  |  |           if (paint_is_face_hidden(lt, mvert, buffers->mloop)) { | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |             continue; | 
					
						
							| 
									
										
										
										
											2019-04-22 09:32:37 +10:00
										 |  |  |           } | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |           /* Face normal and mask */ | 
					
						
							|  |  |  |           if (lt->poly != mpoly_prev) { | 
					
						
							|  |  |  |             const MPoly *mp = &buffers->mpoly[lt->poly]; | 
					
						
							|  |  |  |             float fno[3]; | 
					
						
							|  |  |  |             BKE_mesh_calc_poly_normal(mp, &buffers->mloop[mp->loopstart], mvert, fno); | 
					
						
							|  |  |  |             normal_float_to_short_v3(no, fno); | 
					
						
							|  |  |  |             mpoly_prev = lt->poly; | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |           float fmask = 0.0f; | 
					
						
							| 
									
										
										
										
											2019-09-28 19:15:06 +02:00
										 |  |  |           if (show_mask) { | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |             fmask = (vmask[vtri[0]] + vmask[vtri[1]] + vmask[vtri[2]]) / 3.0f; | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |           for (uint j = 0; j < 3; j++) { | 
					
						
							|  |  |  |             const MVert *v = &mvert[vtri[j]]; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-28 19:15:06 +02:00
										 |  |  |             copy_v3_v3(GPU_vertbuf_raw_step(&pos_step), v->co); | 
					
						
							| 
									
										
										
										
											2019-09-29 23:42:24 +02:00
										 |  |  |             copy_v3_v3_short(GPU_vertbuf_raw_step(&nor_step), no); | 
					
						
							| 
									
										
										
										
											2019-09-28 19:15:06 +02:00
										 |  |  |             if (show_mask) { | 
					
						
							|  |  |  |               *(float *)GPU_vertbuf_raw_step(&msk_step) = fmask; | 
					
						
							|  |  |  |               empty_mask = empty_mask && (fmask == 0.0f); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-28 19:15:06 +02:00
										 |  |  |             if (show_vcol) { | 
					
						
							| 
									
										
										
										
											2019-04-16 14:00:16 +02:00
										 |  |  |               const uint loop_index = lt->tri[j]; | 
					
						
							|  |  |  |               const uchar *elem = &vcol[loop_index].r; | 
					
						
							| 
									
										
										
										
											2019-09-28 19:15:06 +02:00
										 |  |  |               memcpy(GPU_vertbuf_raw_step(&col_step), elem, sizeof(uchar) * 4); | 
					
						
							| 
									
										
										
										
											2019-04-16 14:00:16 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |           } | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       gpu_pbvh_batch_init(buffers, GPU_PRIM_TRIS); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-04 00:46:26 +02:00
										 |  |  |   /* Get material index from the first face of this buffer. */ | 
					
						
							|  |  |  |   const MLoopTri *lt = &buffers->looptri[buffers->face_indices[0]]; | 
					
						
							|  |  |  |   const MPoly *mp = &buffers->mpoly[lt->poly]; | 
					
						
							|  |  |  |   buffers->material_index = mp->mat_nr; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   buffers->show_mask = !empty_mask; | 
					
						
							|  |  |  |   buffers->mvert = mvert; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-27 22:42:57 +02:00
										 |  |  | /* Threaded - do not call any functions that use OpenGL calls! */ | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | GPU_PBVH_Buffers *GPU_pbvh_mesh_buffers_build(const int (*face_vert_indices)[3], | 
					
						
							|  |  |  |                                               const MPoly *mpoly, | 
					
						
							|  |  |  |                                               const MLoop *mloop, | 
					
						
							|  |  |  |                                               const MLoopTri *looptri, | 
					
						
							|  |  |  |                                               const MVert *mvert, | 
					
						
							|  |  |  |                                               const int *face_indices, | 
					
						
							|  |  |  |                                               const int face_indices_len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   GPU_PBVH_Buffers *buffers; | 
					
						
							|  |  |  |   int i, tottri; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   buffers = MEM_callocN(sizeof(GPU_PBVH_Buffers), "GPU_Buffers"); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   /* smooth or flat for all */ | 
					
						
							|  |  |  |   buffers->smooth = mpoly[looptri[face_indices[0]].poly].flag & ME_SMOOTH; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   buffers->show_mask = false; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   /* Count the number of visible triangles */ | 
					
						
							| 
									
										
										
										
											2019-09-08 00:12:26 +10:00
										 |  |  |   for (i = 0, tottri = 0; i < face_indices_len; i++) { | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |     const MLoopTri *lt = &looptri[face_indices[i]]; | 
					
						
							| 
									
										
										
										
											2019-04-22 09:32:37 +10:00
										 |  |  |     if (!paint_is_face_hidden(lt, mvert, mloop)) { | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       tottri++; | 
					
						
							| 
									
										
										
										
											2019-04-22 09:32:37 +10:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   if (tottri == 0) { | 
					
						
							|  |  |  |     buffers->tot_tri = 0; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |     buffers->mpoly = mpoly; | 
					
						
							|  |  |  |     buffers->mloop = mloop; | 
					
						
							|  |  |  |     buffers->looptri = looptri; | 
					
						
							|  |  |  |     buffers->face_indices = face_indices; | 
					
						
							|  |  |  |     buffers->face_indices_len = 0; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |     return buffers; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   /* An element index buffer is used for smooth shading, but flat
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |    * shading requires separate vertex normals so an index buffer | 
					
						
							|  |  |  |    * can't be used there. */ | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   if (buffers->smooth) { | 
					
						
							|  |  |  |     /* Fill the triangle and line buffers. */ | 
					
						
							|  |  |  |     GPUIndexBufBuilder elb, elb_lines; | 
					
						
							|  |  |  |     GPU_indexbuf_init(&elb, GPU_PRIM_TRIS, tottri, INT_MAX); | 
					
						
							|  |  |  |     GPU_indexbuf_init(&elb_lines, GPU_PRIM_LINES, tottri * 3, INT_MAX); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-08 00:12:26 +10:00
										 |  |  |     for (i = 0; i < face_indices_len; i++) { | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       const MLoopTri *lt = &looptri[face_indices[i]]; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       /* Skip hidden faces */ | 
					
						
							| 
									
										
										
										
											2019-04-22 09:32:37 +10:00
										 |  |  |       if (paint_is_face_hidden(lt, mvert, mloop)) { | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |         continue; | 
					
						
							| 
									
										
										
										
											2019-04-22 09:32:37 +10:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       GPU_indexbuf_add_tri_verts(&elb, UNPACK3(face_vert_indices[i])); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       /* TODO skip "non-real" edges. */ | 
					
						
							|  |  |  |       GPU_indexbuf_add_line_verts(&elb_lines, face_vert_indices[i][0], face_vert_indices[i][1]); | 
					
						
							|  |  |  |       GPU_indexbuf_add_line_verts(&elb_lines, face_vert_indices[i][1], face_vert_indices[i][2]); | 
					
						
							|  |  |  |       GPU_indexbuf_add_line_verts(&elb_lines, face_vert_indices[i][2], face_vert_indices[i][0]); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |     buffers->index_buf = GPU_indexbuf_build(&elb); | 
					
						
							|  |  |  |     buffers->index_lines_buf = GPU_indexbuf_build(&elb_lines); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     /* Fill the only the line buffer. */ | 
					
						
							|  |  |  |     GPUIndexBufBuilder elb_lines; | 
					
						
							|  |  |  |     GPU_indexbuf_init(&elb_lines, GPU_PRIM_LINES, tottri * 3, INT_MAX); | 
					
						
							| 
									
										
										
										
											2019-07-08 11:56:57 +02:00
										 |  |  |     int vert_idx = 0; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-08 00:12:26 +10:00
										 |  |  |     for (i = 0; i < face_indices_len; i++) { | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       const MLoopTri *lt = &looptri[face_indices[i]]; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       /* Skip hidden faces */ | 
					
						
							| 
									
										
										
										
											2019-04-22 09:32:37 +10:00
										 |  |  |       if (paint_is_face_hidden(lt, mvert, mloop)) { | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |         continue; | 
					
						
							| 
									
										
										
										
											2019-04-22 09:32:37 +10:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       /* TODO skip "non-real" edges. */ | 
					
						
							| 
									
										
										
										
											2019-07-08 11:56:57 +02:00
										 |  |  |       GPU_indexbuf_add_line_verts(&elb_lines, vert_idx * 3 + 0, vert_idx * 3 + 1); | 
					
						
							|  |  |  |       GPU_indexbuf_add_line_verts(&elb_lines, vert_idx * 3 + 1, vert_idx * 3 + 2); | 
					
						
							|  |  |  |       GPU_indexbuf_add_line_verts(&elb_lines, vert_idx * 3 + 2, vert_idx * 3 + 0); | 
					
						
							|  |  |  |       vert_idx++; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |     buffers->index_lines_buf = GPU_indexbuf_build(&elb_lines); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   buffers->tot_tri = tottri; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   buffers->mpoly = mpoly; | 
					
						
							|  |  |  |   buffers->mloop = mloop; | 
					
						
							|  |  |  |   buffers->looptri = looptri; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   buffers->face_indices = face_indices; | 
					
						
							|  |  |  |   buffers->face_indices_len = face_indices_len; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   return buffers; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | /** \} */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Grid PBVH
 | 
					
						
							| 
									
										
										
										
											2019-02-14 20:24:13 +01:00
										 |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | static void gpu_pbvh_grid_fill_index_buffers( | 
					
						
							|  |  |  |     GPU_PBVH_Buffers *buffers, int *grid_indices, uint visible_quad_len, int totgrid, int gridsize) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   GPUIndexBufBuilder elb, elb_lines; | 
					
						
							|  |  |  |   GPUIndexBufBuilder elb_fast, elb_lines_fast; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   GPU_indexbuf_init(&elb, GPU_PRIM_TRIS, 2 * visible_quad_len, INT_MAX); | 
					
						
							|  |  |  |   GPU_indexbuf_init(&elb_fast, GPU_PRIM_TRIS, 2 * totgrid, INT_MAX); | 
					
						
							|  |  |  |   GPU_indexbuf_init(&elb_lines, GPU_PRIM_LINES, 2 * totgrid * gridsize * (gridsize - 1), INT_MAX); | 
					
						
							|  |  |  |   GPU_indexbuf_init(&elb_lines_fast, GPU_PRIM_LINES, 4 * totgrid, INT_MAX); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (buffers->smooth) { | 
					
						
							|  |  |  |     uint offset = 0; | 
					
						
							|  |  |  |     const uint grid_vert_len = gridsize * gridsize; | 
					
						
							|  |  |  |     for (int i = 0; i < totgrid; i++, offset += grid_vert_len) { | 
					
						
							|  |  |  |       uint v0, v1, v2, v3; | 
					
						
							|  |  |  |       bool grid_visible = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       BLI_bitmap *gh = buffers->grid_hidden[grid_indices[i]]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-08 00:12:26 +10:00
										 |  |  |       for (int j = 0; j < gridsize - 1; j++) { | 
					
						
							|  |  |  |         for (int k = 0; k < gridsize - 1; k++) { | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |           /* Skip hidden grid face */ | 
					
						
							|  |  |  |           if (gh && paint_is_grid_face_hidden(gh, gridsize, k, j)) { | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           /* Indices in a Clockwise QUAD disposition. */ | 
					
						
							|  |  |  |           v0 = offset + j * gridsize + k; | 
					
						
							|  |  |  |           v1 = v0 + 1; | 
					
						
							|  |  |  |           v2 = v1 + gridsize; | 
					
						
							|  |  |  |           v3 = v2 - 1; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |           GPU_indexbuf_add_tri_verts(&elb, v0, v2, v1); | 
					
						
							|  |  |  |           GPU_indexbuf_add_tri_verts(&elb, v0, v3, v2); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |           GPU_indexbuf_add_line_verts(&elb_lines, v0, v1); | 
					
						
							|  |  |  |           GPU_indexbuf_add_line_verts(&elb_lines, v0, v3); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |           if (j + 2 == gridsize) { | 
					
						
							|  |  |  |             GPU_indexbuf_add_line_verts(&elb_lines, v2, v3); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |           } | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |           grid_visible = true; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-09-13 16:18:18 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (grid_visible) { | 
					
						
							|  |  |  |           GPU_indexbuf_add_line_verts(&elb_lines, v1, v2); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       if (grid_visible) { | 
					
						
							|  |  |  |         /* Grid corners */ | 
					
						
							|  |  |  |         v0 = offset; | 
					
						
							|  |  |  |         v1 = offset + gridsize - 1; | 
					
						
							|  |  |  |         v2 = offset + grid_vert_len - 1; | 
					
						
							|  |  |  |         v3 = offset + grid_vert_len - gridsize; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |         GPU_indexbuf_add_tri_verts(&elb_fast, v0, v2, v1); | 
					
						
							|  |  |  |         GPU_indexbuf_add_tri_verts(&elb_fast, v0, v3, v2); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |         GPU_indexbuf_add_line_verts(&elb_lines_fast, v0, v1); | 
					
						
							|  |  |  |         GPU_indexbuf_add_line_verts(&elb_lines_fast, v1, v2); | 
					
						
							|  |  |  |         GPU_indexbuf_add_line_verts(&elb_lines_fast, v2, v3); | 
					
						
							|  |  |  |         GPU_indexbuf_add_line_verts(&elb_lines_fast, v3, v0); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     uint offset = 0; | 
					
						
							|  |  |  |     const uint grid_vert_len = SQUARE(gridsize - 1) * 4; | 
					
						
							|  |  |  |     for (int i = 0; i < totgrid; i++, offset += grid_vert_len) { | 
					
						
							|  |  |  |       bool grid_visible = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       BLI_bitmap *gh = buffers->grid_hidden[grid_indices[i]]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       uint v0, v1, v2, v3; | 
					
						
							|  |  |  |       for (int j = 0; j < gridsize - 1; j++) { | 
					
						
							|  |  |  |         for (int k = 0; k < gridsize - 1; k++) { | 
					
						
							|  |  |  |           /* Skip hidden grid face */ | 
					
						
							|  |  |  |           if (gh && paint_is_grid_face_hidden(gh, gridsize, k, j)) { | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           /* VBO data are in a Clockwise QUAD disposition. */ | 
					
						
							|  |  |  |           v0 = offset + (j * (gridsize - 1) + k) * 4; | 
					
						
							|  |  |  |           v1 = v0 + 1; | 
					
						
							|  |  |  |           v2 = v0 + 2; | 
					
						
							|  |  |  |           v3 = v0 + 3; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |           GPU_indexbuf_add_tri_verts(&elb, v0, v2, v1); | 
					
						
							|  |  |  |           GPU_indexbuf_add_tri_verts(&elb, v0, v3, v2); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |           GPU_indexbuf_add_line_verts(&elb_lines, v0, v1); | 
					
						
							|  |  |  |           GPU_indexbuf_add_line_verts(&elb_lines, v0, v3); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |           if (j + 2 == gridsize) { | 
					
						
							|  |  |  |             GPU_indexbuf_add_line_verts(&elb_lines, v2, v3); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |           } | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |           grid_visible = true; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-09-13 16:18:18 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (grid_visible) { | 
					
						
							|  |  |  |           GPU_indexbuf_add_line_verts(&elb_lines, v1, v2); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       if (grid_visible) { | 
					
						
							|  |  |  |         /* Grid corners */ | 
					
						
							|  |  |  |         v0 = offset; | 
					
						
							|  |  |  |         v1 = offset + (gridsize - 1) * 4 - 3; | 
					
						
							|  |  |  |         v2 = offset + grid_vert_len - 2; | 
					
						
							|  |  |  |         v3 = offset + grid_vert_len - (gridsize - 1) * 4 + 3; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |         GPU_indexbuf_add_tri_verts(&elb_fast, v0, v2, v1); | 
					
						
							|  |  |  |         GPU_indexbuf_add_tri_verts(&elb_fast, v0, v3, v2); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |         GPU_indexbuf_add_line_verts(&elb_lines_fast, v0, v1); | 
					
						
							|  |  |  |         GPU_indexbuf_add_line_verts(&elb_lines_fast, v1, v2); | 
					
						
							|  |  |  |         GPU_indexbuf_add_line_verts(&elb_lines_fast, v2, v3); | 
					
						
							|  |  |  |         GPU_indexbuf_add_line_verts(&elb_lines_fast, v3, v0); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   buffers->index_buf = GPU_indexbuf_build(&elb); | 
					
						
							|  |  |  |   buffers->index_buf_fast = GPU_indexbuf_build(&elb_fast); | 
					
						
							|  |  |  |   buffers->index_lines_buf = GPU_indexbuf_build(&elb_lines); | 
					
						
							|  |  |  |   buffers->index_lines_buf_fast = GPU_indexbuf_build(&elb_lines_fast); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-27 22:42:57 +02:00
										 |  |  | void GPU_pbvh_grid_buffers_update_free(GPU_PBVH_Buffers *buffers, | 
					
						
							|  |  |  |                                        const struct DMFlagMat *grid_flag_mats, | 
					
						
							|  |  |  |                                        int *grid_indices) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   const bool smooth = grid_flag_mats[grid_indices[0]].flag & ME_SMOOTH; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (buffers->smooth != smooth) { | 
					
						
							|  |  |  |     buffers->smooth = smooth; | 
					
						
							|  |  |  |     GPU_BATCH_DISCARD_SAFE(buffers->triangles); | 
					
						
							|  |  |  |     GPU_BATCH_DISCARD_SAFE(buffers->triangles_fast); | 
					
						
							|  |  |  |     GPU_BATCH_DISCARD_SAFE(buffers->lines); | 
					
						
							|  |  |  |     GPU_BATCH_DISCARD_SAFE(buffers->lines_fast); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     GPU_INDEXBUF_DISCARD_SAFE(buffers->index_buf); | 
					
						
							|  |  |  |     GPU_INDEXBUF_DISCARD_SAFE(buffers->index_buf_fast); | 
					
						
							|  |  |  |     GPU_INDEXBUF_DISCARD_SAFE(buffers->index_lines_buf); | 
					
						
							|  |  |  |     GPU_INDEXBUF_DISCARD_SAFE(buffers->index_lines_buf_fast); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Threaded - do not call any functions that use OpenGL calls! */ | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | void GPU_pbvh_grid_buffers_update(GPU_PBVH_Buffers *buffers, | 
					
						
							|  |  |  |                                   CCGElem **grids, | 
					
						
							|  |  |  |                                   const DMFlagMat *grid_flag_mats, | 
					
						
							|  |  |  |                                   int *grid_indices, | 
					
						
							|  |  |  |                                   int totgrid, | 
					
						
							|  |  |  |                                   const CCGKey *key, | 
					
						
							|  |  |  |                                   const int update_flags) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   const bool show_mask = (update_flags & GPU_PBVH_BUFFERS_SHOW_MASK) != 0; | 
					
						
							| 
									
										
										
										
											2019-04-16 14:00:16 +02:00
										 |  |  |   const bool show_vcol = (update_flags & GPU_PBVH_BUFFERS_SHOW_VCOL) != 0; | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   bool empty_mask = true; | 
					
						
							|  |  |  |   int i, j, k, x, y; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   /* Build VBO */ | 
					
						
							|  |  |  |   const int has_mask = key->has_mask; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-27 22:42:57 +02:00
										 |  |  |   buffers->smooth = grid_flag_mats[grid_indices[0]].flag & ME_SMOOTH; | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-27 22:42:57 +02:00
										 |  |  |   uint vert_per_grid = (buffers->smooth) ? key->grid_area : (SQUARE(key->grid_size - 1) * 4); | 
					
						
							|  |  |  |   uint vert_count = totgrid * vert_per_grid; | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (buffers->index_buf == NULL) { | 
					
						
							|  |  |  |     uint visible_quad_len = BKE_pbvh_count_grid_quads( | 
					
						
							|  |  |  |         (BLI_bitmap **)buffers->grid_hidden, grid_indices, totgrid, key->grid_size); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |     /* totally hidden node, return here to avoid BufferData with zero below. */ | 
					
						
							|  |  |  |     if (visible_quad_len == 0) { | 
					
						
							|  |  |  |       return; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |     gpu_pbvh_grid_fill_index_buffers( | 
					
						
							|  |  |  |         buffers, grid_indices, visible_quad_len, totgrid, key->grid_size); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   uint vbo_index_offset = 0; | 
					
						
							|  |  |  |   /* Build VBO */ | 
					
						
							|  |  |  |   if (gpu_pbvh_vert_buf_data_set(buffers, vert_count)) { | 
					
						
							|  |  |  |     GPUIndexBufBuilder elb_lines; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |     if (buffers->index_lines_buf == NULL) { | 
					
						
							|  |  |  |       GPU_indexbuf_init(&elb_lines, GPU_PRIM_LINES, totgrid * key->grid_area * 2, vert_count); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-08 00:12:26 +10:00
										 |  |  |     for (i = 0; i < totgrid; i++) { | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       CCGElem *grid = grids[grid_indices[i]]; | 
					
						
							|  |  |  |       int vbo_index = vbo_index_offset; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (buffers->smooth) { | 
					
						
							|  |  |  |         for (y = 0; y < key->grid_size; y++) { | 
					
						
							|  |  |  |           for (x = 0; x < key->grid_size; x++) { | 
					
						
							|  |  |  |             CCGElem *elem = CCG_grid_elem(key, grid, x, y); | 
					
						
							|  |  |  |             GPU_vertbuf_attr_set( | 
					
						
							|  |  |  |                 buffers->vert_buf, g_vbo_id.pos, vbo_index, CCG_elem_co(key, elem)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             short no_short[3]; | 
					
						
							|  |  |  |             normal_float_to_short_v3(no_short, CCG_elem_no(key, elem)); | 
					
						
							|  |  |  |             GPU_vertbuf_attr_set(buffers->vert_buf, g_vbo_id.nor, vbo_index, no_short); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (has_mask && show_mask) { | 
					
						
							|  |  |  |               float fmask = *CCG_elem_mask(key, elem); | 
					
						
							|  |  |  |               GPU_vertbuf_attr_set(buffers->vert_buf, g_vbo_id.msk, vbo_index, &fmask); | 
					
						
							|  |  |  |               empty_mask = empty_mask && (fmask == 0.0f); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-04-16 14:00:16 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             if (show_vcol) { | 
					
						
							| 
									
										
										
										
											2019-05-04 00:46:26 +02:00
										 |  |  |               char vcol[4] = {255, 255, 255, 255}; | 
					
						
							| 
									
										
										
										
											2019-04-16 14:00:16 +02:00
										 |  |  |               GPU_vertbuf_attr_set(buffers->vert_buf, g_vbo_id.col, vbo_index, &vcol); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |             vbo_index += 1; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |         vbo_index_offset += key->grid_area; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         for (j = 0; j < key->grid_size - 1; j++) { | 
					
						
							|  |  |  |           for (k = 0; k < key->grid_size - 1; k++) { | 
					
						
							|  |  |  |             CCGElem *elems[4] = { | 
					
						
							|  |  |  |                 CCG_grid_elem(key, grid, k, j), | 
					
						
							|  |  |  |                 CCG_grid_elem(key, grid, k + 1, j), | 
					
						
							|  |  |  |                 CCG_grid_elem(key, grid, k + 1, j + 1), | 
					
						
							|  |  |  |                 CCG_grid_elem(key, grid, k, j + 1), | 
					
						
							|  |  |  |             }; | 
					
						
							|  |  |  |             float *co[4] = { | 
					
						
							|  |  |  |                 CCG_elem_co(key, elems[0]), | 
					
						
							|  |  |  |                 CCG_elem_co(key, elems[1]), | 
					
						
							|  |  |  |                 CCG_elem_co(key, elems[2]), | 
					
						
							|  |  |  |                 CCG_elem_co(key, elems[3]), | 
					
						
							|  |  |  |             }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             float fno[3]; | 
					
						
							|  |  |  |             short no_short[3]; | 
					
						
							|  |  |  |             /* Note: Clockwise indices ordering, that's why we invert order here. */ | 
					
						
							|  |  |  |             normal_quad_v3(fno, co[3], co[2], co[1], co[0]); | 
					
						
							|  |  |  |             normal_float_to_short_v3(no_short, fno); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             GPU_vertbuf_attr_set(buffers->vert_buf, g_vbo_id.pos, vbo_index + 0, co[0]); | 
					
						
							|  |  |  |             GPU_vertbuf_attr_set(buffers->vert_buf, g_vbo_id.nor, vbo_index + 0, no_short); | 
					
						
							|  |  |  |             GPU_vertbuf_attr_set(buffers->vert_buf, g_vbo_id.pos, vbo_index + 1, co[1]); | 
					
						
							|  |  |  |             GPU_vertbuf_attr_set(buffers->vert_buf, g_vbo_id.nor, vbo_index + 1, no_short); | 
					
						
							|  |  |  |             GPU_vertbuf_attr_set(buffers->vert_buf, g_vbo_id.pos, vbo_index + 2, co[2]); | 
					
						
							|  |  |  |             GPU_vertbuf_attr_set(buffers->vert_buf, g_vbo_id.nor, vbo_index + 2, no_short); | 
					
						
							|  |  |  |             GPU_vertbuf_attr_set(buffers->vert_buf, g_vbo_id.pos, vbo_index + 3, co[3]); | 
					
						
							|  |  |  |             GPU_vertbuf_attr_set(buffers->vert_buf, g_vbo_id.nor, vbo_index + 3, no_short); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (has_mask && show_mask) { | 
					
						
							|  |  |  |               float fmask = (*CCG_elem_mask(key, elems[0]) + *CCG_elem_mask(key, elems[1]) + | 
					
						
							|  |  |  |                              *CCG_elem_mask(key, elems[2]) + *CCG_elem_mask(key, elems[3])) * | 
					
						
							|  |  |  |                             0.25f; | 
					
						
							|  |  |  |               GPU_vertbuf_attr_set(buffers->vert_buf, g_vbo_id.msk, vbo_index + 0, &fmask); | 
					
						
							|  |  |  |               GPU_vertbuf_attr_set(buffers->vert_buf, g_vbo_id.msk, vbo_index + 1, &fmask); | 
					
						
							|  |  |  |               GPU_vertbuf_attr_set(buffers->vert_buf, g_vbo_id.msk, vbo_index + 2, &fmask); | 
					
						
							|  |  |  |               GPU_vertbuf_attr_set(buffers->vert_buf, g_vbo_id.msk, vbo_index + 3, &fmask); | 
					
						
							|  |  |  |               empty_mask = empty_mask && (fmask == 0.0f); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2019-05-04 00:46:26 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             char vcol[4] = {255, 255, 255, 255}; | 
					
						
							| 
									
										
										
										
											2019-04-16 14:00:16 +02:00
										 |  |  |             GPU_vertbuf_attr_set(buffers->vert_buf, g_vbo_id.col, vbo_index + 0, &vcol); | 
					
						
							|  |  |  |             GPU_vertbuf_attr_set(buffers->vert_buf, g_vbo_id.col, vbo_index + 1, &vcol); | 
					
						
							|  |  |  |             GPU_vertbuf_attr_set(buffers->vert_buf, g_vbo_id.col, vbo_index + 2, &vcol); | 
					
						
							|  |  |  |             GPU_vertbuf_attr_set(buffers->vert_buf, g_vbo_id.col, vbo_index + 3, &vcol); | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |             vbo_index += 4; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |         vbo_index_offset += SQUARE(key->grid_size - 1) * 4; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |     gpu_pbvh_batch_init(buffers, GPU_PRIM_TRIS); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-04 00:46:26 +02:00
										 |  |  |   /* Get material index from the first face of this buffer. */ | 
					
						
							|  |  |  |   buffers->material_index = grid_flag_mats[grid_indices[0]].mat_nr; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   buffers->grids = grids; | 
					
						
							|  |  |  |   buffers->grid_indices = grid_indices; | 
					
						
							|  |  |  |   buffers->totgrid = totgrid; | 
					
						
							|  |  |  |   buffers->grid_flag_mats = grid_flag_mats; | 
					
						
							|  |  |  |   buffers->gridkey = *key; | 
					
						
							|  |  |  |   buffers->show_mask = !empty_mask; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-27 22:42:57 +02:00
										 |  |  | /* Threaded - do not call any functions that use OpenGL calls! */ | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | GPU_PBVH_Buffers *GPU_pbvh_grid_buffers_build(int totgrid, BLI_bitmap **grid_hidden) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   GPU_PBVH_Buffers *buffers; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   buffers = MEM_callocN(sizeof(GPU_PBVH_Buffers), "GPU_Buffers"); | 
					
						
							|  |  |  |   buffers->grid_hidden = grid_hidden; | 
					
						
							|  |  |  |   buffers->totgrid = totgrid; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   buffers->show_mask = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return buffers; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
											  
											
												== GPU Buffers ==
This patch attempts to clean up and document the GPU buffers
code. There are a few bug fixes as well.
Patch reviewed here: http://codereview.appspot.com/4631052/
Summary:
* Bugfix: make GPU_buffer_copy_normal convert from shorts to floats
  correctly, also fixed the use of cached face normal CustomData.
* Bugfix: changed the `mat_nr' field of GPUBufferMaterial from char to
  short.
* Changed color buffer setup to not alloc a temporary copy of color
  data, just passes the MCol data in directly.
* Changed the GPU buffer pool code to make clearer what operates
  specifically on the global pool.
* Lots of refactoring for GPU_drawobject_new; should operate mostly
  the same (except got rid of one unecessary allocation), just split
  into more functions and without macros now.
* Converted some #defines into enumerations.
* Made some stuff private, pulled out of header file.
* Deleted unused function GPU_buffer_pool_free_unused().
* Removed GPU_interleaved_setup and related #defines. (I think this
  was used for editmode VBOs, but those were disabled.)
* Added lots of comments.
* Added a few comments in the code signed `--nicholas' to note places
  where I am unsure about design or usage, would be good to address
  these better.
* Code formatting changed to be more consistent with the rest of
  Blender.
* Renamed some fields and variables to be more consistent with
  Blender's naming conventions.
* Renamed some fields and variables to use more descriptive names,
  e.g. renamed `redir' to `mat_orig_to_new'.
* Removed print outs with DEBUG_VBO -- don't feel too strongly about
  this one, just not used elsewhere in Blender, could be easily added
  back if others disagree though.
* Moved the PBVH drawing code down to the bottom of the file, before
  was sitting in the middle of the other VBO code
											
										 
											2011-07-08 19:58:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-22 23:30:56 +00:00
										 |  |  | #undef FILL_QUAD_BUFFER
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | /** \} */ | 
					
						
							| 
									
										
										
										
											2019-02-14 20:24:13 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name BMesh PBVH
 | 
					
						
							| 
									
										
										
										
											2019-02-14 20:24:13 +01:00
										 |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | /* Output a BMVert into a VertexBufferFormat array
 | 
					
						
							| 
									
										
										
										
											2012-12-30 18:24:54 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * The vertex is skipped if hidden, otherwise the output goes into | 
					
						
							|  |  |  |  * index '*v_index' in the 'vert_data' array and '*v_index' is | 
					
						
							|  |  |  |  * incremented. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | static void gpu_bmesh_vert_to_buffer_copy__gwn(BMVert *v, | 
					
						
							|  |  |  |                                                GPUVertBuf *vert_buf, | 
					
						
							|  |  |  |                                                int *v_index, | 
					
						
							|  |  |  |                                                const float fno[3], | 
					
						
							|  |  |  |                                                const float *fmask, | 
					
						
							|  |  |  |                                                const int cd_vert_mask_offset, | 
					
						
							|  |  |  |                                                const bool show_mask, | 
					
						
							| 
									
										
										
										
											2019-04-16 14:00:16 +02:00
										 |  |  |                                                const bool show_vcol, | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |                                                bool *empty_mask) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (!BM_elem_flag_test(v, BM_ELEM_HIDDEN)) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |     /* Set coord, normal, and mask */ | 
					
						
							|  |  |  |     GPU_vertbuf_attr_set(vert_buf, g_vbo_id.pos, *v_index, v->co); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |     short no_short[3]; | 
					
						
							|  |  |  |     normal_float_to_short_v3(no_short, fno ? fno : v->no); | 
					
						
							|  |  |  |     GPU_vertbuf_attr_set(vert_buf, g_vbo_id.nor, *v_index, no_short); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |     if (show_mask) { | 
					
						
							|  |  |  |       float effective_mask = fmask ? *fmask : BM_ELEM_CD_GET_FLOAT(v, cd_vert_mask_offset); | 
					
						
							|  |  |  |       GPU_vertbuf_attr_set(vert_buf, g_vbo_id.msk, *v_index, &effective_mask); | 
					
						
							|  |  |  |       *empty_mask = *empty_mask && (effective_mask == 0.0f); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-16 14:00:16 +02:00
										 |  |  |     if (show_vcol) { | 
					
						
							|  |  |  |       static char vcol[4] = {255, 255, 255, 255}; | 
					
						
							|  |  |  |       GPU_vertbuf_attr_set(vert_buf, g_vbo_id.col, *v_index, &vcol); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |     /* Assign index for use in the triangle index buffer */ | 
					
						
							|  |  |  |     /* note: caller must set:  bm->elem_index_dirty |= BM_VERT; */ | 
					
						
							|  |  |  |     BM_elem_index_set(v, (*v_index)); /* set_dirty! */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |     (*v_index)++; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | /* Return the total number of vertices that don't have BM_ELEM_HIDDEN set */ | 
					
						
							|  |  |  | static int gpu_bmesh_vert_visible_count(GSet *bm_unique_verts, GSet *bm_other_verts) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   GSetIterator gs_iter; | 
					
						
							|  |  |  |   int totvert = 0; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   GSET_ITER (gs_iter, bm_unique_verts) { | 
					
						
							|  |  |  |     BMVert *v = BLI_gsetIterator_getKey(&gs_iter); | 
					
						
							| 
									
										
										
										
											2019-04-22 09:32:37 +10:00
										 |  |  |     if (!BM_elem_flag_test(v, BM_ELEM_HIDDEN)) { | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       totvert++; | 
					
						
							| 
									
										
										
										
											2019-04-22 09:32:37 +10:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   } | 
					
						
							|  |  |  |   GSET_ITER (gs_iter, bm_other_verts) { | 
					
						
							|  |  |  |     BMVert *v = BLI_gsetIterator_getKey(&gs_iter); | 
					
						
							| 
									
										
										
										
											2019-04-22 09:32:37 +10:00
										 |  |  |     if (!BM_elem_flag_test(v, BM_ELEM_HIDDEN)) { | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       totvert++; | 
					
						
							| 
									
										
										
										
											2019-04-22 09:32:37 +10:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   return totvert; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | /* Return the total number of visible faces */ | 
					
						
							|  |  |  | static int gpu_bmesh_face_visible_count(GSet *bm_faces) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   GSetIterator gh_iter; | 
					
						
							|  |  |  |   int totface = 0; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   GSET_ITER (gh_iter, bm_faces) { | 
					
						
							|  |  |  |     BMFace *f = BLI_gsetIterator_getKey(&gh_iter); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-22 09:32:37 +10:00
										 |  |  |     if (!BM_elem_flag_test(f, BM_ELEM_HIDDEN)) { | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       totface++; | 
					
						
							| 
									
										
										
										
											2019-04-22 09:32:37 +10:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   return totface; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-27 22:42:57 +02:00
										 |  |  | void GPU_pbvh_bmesh_buffers_update_free(GPU_PBVH_Buffers *buffers) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (buffers->smooth) { | 
					
						
							|  |  |  |     /* Smooth needs to recreate index buffer, so we have to invalidate the batch. */ | 
					
						
							|  |  |  |     GPU_BATCH_DISCARD_SAFE(buffers->triangles); | 
					
						
							|  |  |  |     GPU_BATCH_DISCARD_SAFE(buffers->lines); | 
					
						
							|  |  |  |     GPU_INDEXBUF_DISCARD_SAFE(buffers->index_lines_buf); | 
					
						
							|  |  |  |     GPU_INDEXBUF_DISCARD_SAFE(buffers->index_buf); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     GPU_BATCH_DISCARD_SAFE(buffers->lines); | 
					
						
							|  |  |  |     GPU_INDEXBUF_DISCARD_SAFE(buffers->index_lines_buf); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | /* Creates a vertex buffer (coordinate, normal, color) and, if smooth
 | 
					
						
							| 
									
										
										
										
											2019-09-27 22:42:57 +02:00
										 |  |  |  * shading, an element index buffer. | 
					
						
							|  |  |  |  * Threaded - do not call any functions that use OpenGL calls! */ | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | void GPU_pbvh_bmesh_buffers_update(GPU_PBVH_Buffers *buffers, | 
					
						
							|  |  |  |                                    BMesh *bm, | 
					
						
							|  |  |  |                                    GSet *bm_faces, | 
					
						
							|  |  |  |                                    GSet *bm_unique_verts, | 
					
						
							|  |  |  |                                    GSet *bm_other_verts, | 
					
						
							|  |  |  |                                    const int update_flags) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   const bool show_mask = (update_flags & GPU_PBVH_BUFFERS_SHOW_MASK) != 0; | 
					
						
							| 
									
										
										
										
											2019-04-16 14:00:16 +02:00
										 |  |  |   const bool show_vcol = (update_flags & GPU_PBVH_BUFFERS_SHOW_VCOL) != 0; | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   int tottri, totvert, maxvert = 0; | 
					
						
							|  |  |  |   bool empty_mask = true; | 
					
						
							| 
									
										
										
										
											2019-09-13 16:18:18 +02:00
										 |  |  |   BMFace *f = NULL; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   /* Count visible triangles */ | 
					
						
							|  |  |  |   tottri = gpu_bmesh_face_visible_count(bm_faces); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   if (buffers->smooth) { | 
					
						
							|  |  |  |     /* Count visible vertices */ | 
					
						
							|  |  |  |     totvert = gpu_bmesh_vert_visible_count(bm_unique_verts, bm_other_verts); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     totvert = tottri * 3; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   if (!tottri) { | 
					
						
							| 
									
										
										
										
											2019-06-06 10:46:47 -03:00
										 |  |  |     if (BLI_gset_len(bm_faces) != 0) { | 
					
						
							|  |  |  |       /* Node is just hidden. */ | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2019-09-27 22:42:57 +02:00
										 |  |  |       buffers->clear_bmesh_on_flush = true; | 
					
						
							| 
									
										
										
										
											2019-06-06 10:46:47 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |     buffers->tot_tri = 0; | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-06 10:46:47 -03:00
										 |  |  |   /* TODO, make mask layer optional for bmesh buffer */ | 
					
						
							|  |  |  |   const int cd_vert_mask_offset = CustomData_get_offset(&bm->vdata, CD_PAINT_MASK); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   /* Fill vertex buffer */ | 
					
						
							|  |  |  |   if (gpu_pbvh_vert_buf_data_set(buffers, totvert)) { | 
					
						
							|  |  |  |     int v_index = 0; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |     if (buffers->smooth) { | 
					
						
							|  |  |  |       GSetIterator gs_iter; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       /* Vertices get an index assigned for use in the triangle
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |        * index buffer */ | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       bm->elem_index_dirty |= BM_VERT; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       GSET_ITER (gs_iter, bm_unique_verts) { | 
					
						
							|  |  |  |         gpu_bmesh_vert_to_buffer_copy__gwn(BLI_gsetIterator_getKey(&gs_iter), | 
					
						
							|  |  |  |                                            buffers->vert_buf, | 
					
						
							|  |  |  |                                            &v_index, | 
					
						
							|  |  |  |                                            NULL, | 
					
						
							|  |  |  |                                            NULL, | 
					
						
							|  |  |  |                                            cd_vert_mask_offset, | 
					
						
							|  |  |  |                                            show_mask, | 
					
						
							| 
									
										
										
										
											2019-04-16 14:00:16 +02:00
										 |  |  |                                            show_vcol, | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |                                            &empty_mask); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       GSET_ITER (gs_iter, bm_other_verts) { | 
					
						
							|  |  |  |         gpu_bmesh_vert_to_buffer_copy__gwn(BLI_gsetIterator_getKey(&gs_iter), | 
					
						
							|  |  |  |                                            buffers->vert_buf, | 
					
						
							|  |  |  |                                            &v_index, | 
					
						
							|  |  |  |                                            NULL, | 
					
						
							|  |  |  |                                            NULL, | 
					
						
							|  |  |  |                                            cd_vert_mask_offset, | 
					
						
							|  |  |  |                                            show_mask, | 
					
						
							| 
									
										
										
										
											2019-04-16 14:00:16 +02:00
										 |  |  |                                            show_vcol, | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |                                            &empty_mask); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       maxvert = v_index; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       GSetIterator gs_iter; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       GPUIndexBufBuilder elb_lines; | 
					
						
							|  |  |  |       GPU_indexbuf_init(&elb_lines, GPU_PRIM_LINES, tottri * 3, totvert); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       GSET_ITER (gs_iter, bm_faces) { | 
					
						
							| 
									
										
										
										
											2019-05-04 00:46:26 +02:00
										 |  |  |         f = BLI_gsetIterator_getKey(&gs_iter); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |         BLI_assert(f->len == 3); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |         if (!BM_elem_flag_test(f, BM_ELEM_HIDDEN)) { | 
					
						
							|  |  |  |           BMVert *v[3]; | 
					
						
							|  |  |  |           float fmask = 0.0f; | 
					
						
							|  |  |  |           int i; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |           BM_face_as_array_vert_tri(f, v); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |           /* Average mask value */ | 
					
						
							|  |  |  |           for (i = 0; i < 3; i++) { | 
					
						
							|  |  |  |             fmask += BM_ELEM_CD_GET_FLOAT(v[i], cd_vert_mask_offset); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           fmask /= 3.0f; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           GPU_indexbuf_add_line_verts(&elb_lines, v_index + 0, v_index + 1); | 
					
						
							|  |  |  |           GPU_indexbuf_add_line_verts(&elb_lines, v_index + 1, v_index + 2); | 
					
						
							|  |  |  |           GPU_indexbuf_add_line_verts(&elb_lines, v_index + 2, v_index + 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           for (i = 0; i < 3; i++) { | 
					
						
							|  |  |  |             gpu_bmesh_vert_to_buffer_copy__gwn(v[i], | 
					
						
							|  |  |  |                                                buffers->vert_buf, | 
					
						
							|  |  |  |                                                &v_index, | 
					
						
							|  |  |  |                                                f->no, | 
					
						
							|  |  |  |                                                &fmask, | 
					
						
							|  |  |  |                                                cd_vert_mask_offset, | 
					
						
							|  |  |  |                                                show_mask, | 
					
						
							| 
									
										
										
										
											2019-04-16 14:00:16 +02:00
										 |  |  |                                                show_vcol, | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |                                                &empty_mask); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       buffers->index_lines_buf = GPU_indexbuf_build(&elb_lines); | 
					
						
							|  |  |  |       buffers->tot_tri = tottri; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |     /* gpu_bmesh_vert_to_buffer_copy sets dirty index values */ | 
					
						
							|  |  |  |     bm->elem_index_dirty |= BM_VERT; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     /* Memory map failed */ | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   if (buffers->smooth) { | 
					
						
							|  |  |  |     /* Fill the triangle buffer */ | 
					
						
							|  |  |  |     GPUIndexBufBuilder elb, elb_lines; | 
					
						
							|  |  |  |     GPU_indexbuf_init(&elb, GPU_PRIM_TRIS, tottri, maxvert); | 
					
						
							|  |  |  |     GPU_indexbuf_init(&elb_lines, GPU_PRIM_LINES, tottri * 3, maxvert); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |     /* Fill triangle index buffer */ | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       GSetIterator gs_iter; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       GSET_ITER (gs_iter, bm_faces) { | 
					
						
							| 
									
										
										
										
											2019-05-04 00:46:26 +02:00
										 |  |  |         f = BLI_gsetIterator_getKey(&gs_iter); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |         if (!BM_elem_flag_test(f, BM_ELEM_HIDDEN)) { | 
					
						
							|  |  |  |           BMVert *v[3]; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |           BM_face_as_array_vert_tri(f, v); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-14 08:10:50 +10:00
										 |  |  |           const uint idx[3] = { | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |               BM_elem_index_get(v[0]), BM_elem_index_get(v[1]), BM_elem_index_get(v[2])}; | 
					
						
							|  |  |  |           GPU_indexbuf_add_tri_verts(&elb, idx[0], idx[1], idx[2]); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |           GPU_indexbuf_add_line_verts(&elb_lines, idx[0], idx[1]); | 
					
						
							|  |  |  |           GPU_indexbuf_add_line_verts(&elb_lines, idx[1], idx[2]); | 
					
						
							|  |  |  |           GPU_indexbuf_add_line_verts(&elb_lines, idx[2], idx[0]); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |       buffers->tot_tri = tottri; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (buffers->index_buf == NULL) { | 
					
						
							|  |  |  |         buffers->index_buf = GPU_indexbuf_build(&elb); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         GPU_indexbuf_build_in_place(&elb, buffers->index_buf); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |       buffers->index_lines_buf = GPU_indexbuf_build(&elb_lines); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-04 00:46:26 +02:00
										 |  |  |   /* Get material index from the last face we iterated on. */ | 
					
						
							| 
									
										
										
										
											2019-09-13 16:18:18 +02:00
										 |  |  |   buffers->material_index = (f) ? f->mat_nr : 0; | 
					
						
							| 
									
										
										
										
											2019-05-04 00:46:26 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   buffers->show_mask = !empty_mask; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   gpu_pbvh_batch_init(buffers, GPU_PRIM_TRIS); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | /** \} */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Generic
 | 
					
						
							| 
									
										
										
										
											2019-02-14 20:24:13 +01:00
										 |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-27 22:42:57 +02:00
										 |  |  | /* Threaded - do not call any functions that use OpenGL calls! */ | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | GPU_PBVH_Buffers *GPU_pbvh_bmesh_buffers_build(bool smooth_shading) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   GPU_PBVH_Buffers *buffers; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   buffers = MEM_callocN(sizeof(GPU_PBVH_Buffers), "GPU_Buffers"); | 
					
						
							|  |  |  |   buffers->use_bmesh = true; | 
					
						
							|  |  |  |   buffers->smooth = smooth_shading; | 
					
						
							|  |  |  |   buffers->show_mask = true; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   return buffers; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | GPUBatch *GPU_pbvh_buffers_batch_get(GPU_PBVH_Buffers *buffers, bool fast, bool wires) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (wires) { | 
					
						
							|  |  |  |     return (fast && buffers->lines_fast) ? buffers->lines_fast : buffers->lines; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  |   else { | 
					
						
							|  |  |  |     return (fast && buffers->triangles_fast) ? buffers->triangles_fast : buffers->triangles; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | bool GPU_pbvh_buffers_has_mask(GPU_PBVH_Buffers *buffers) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return buffers->show_mask; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-04 00:46:26 +02:00
										 |  |  | short GPU_pbvh_buffers_material_index_get(GPU_PBVH_Buffers *buffers) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return buffers->material_index; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-27 22:42:57 +02:00
										 |  |  | void GPU_pbvh_buffers_update_flush(GPU_PBVH_Buffers *buffers) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   /* Free empty bmesh node buffers. */ | 
					
						
							|  |  |  |   if (buffers->clear_bmesh_on_flush) { | 
					
						
							|  |  |  |     GPU_BATCH_DISCARD_SAFE(buffers->triangles); | 
					
						
							|  |  |  |     GPU_INDEXBUF_DISCARD_SAFE(buffers->index_buf); | 
					
						
							|  |  |  |     GPU_VERTBUF_DISCARD_SAFE(buffers->vert_buf); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Force flushing to the GPU. */ | 
					
						
							|  |  |  |   if (buffers->vert_buf && buffers->vert_buf->data) { | 
					
						
							|  |  |  |     GPU_vertbuf_use(buffers->vert_buf); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | void GPU_pbvh_buffers_free(GPU_PBVH_Buffers *buffers) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (buffers) { | 
					
						
							|  |  |  |     GPU_BATCH_DISCARD_SAFE(buffers->lines); | 
					
						
							|  |  |  |     GPU_BATCH_DISCARD_SAFE(buffers->lines_fast); | 
					
						
							|  |  |  |     GPU_BATCH_DISCARD_SAFE(buffers->triangles); | 
					
						
							|  |  |  |     GPU_BATCH_DISCARD_SAFE(buffers->triangles_fast); | 
					
						
							|  |  |  |     GPU_INDEXBUF_DISCARD_SAFE(buffers->index_lines_buf_fast); | 
					
						
							|  |  |  |     GPU_INDEXBUF_DISCARD_SAFE(buffers->index_lines_buf); | 
					
						
							|  |  |  |     GPU_INDEXBUF_DISCARD_SAFE(buffers->index_buf_fast); | 
					
						
							|  |  |  |     GPU_INDEXBUF_DISCARD_SAFE(buffers->index_buf); | 
					
						
							|  |  |  |     GPU_VERTBUF_DISCARD_SAFE(buffers->vert_buf); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     MEM_freeN(buffers); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 08:56:59 +02:00
										 |  |  | /** \} */ |