| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU General Public License | 
					
						
							|  |  |  |  * as published by the Free Software Foundation; either version 2 | 
					
						
							|  |  |  |  * of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  * GNU General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  * along with this program; if not, write to the Free Software Foundation, | 
					
						
							|  |  |  |  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The Original Code is Copyright (C) 2016 by Mike Erwin. | 
					
						
							|  |  |  |  * All rights reserved. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | /** \file blender/gpu/intern/gpu_element.c
 | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  |  *  \ingroup gpu | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  |  * GPU element list (AKA index buffer) | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-31 16:54:58 +02:00
										 |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-17 21:11:23 +02:00
										 |  |  | #include "GPU_element.h"
 | 
					
						
							| 
									
										
										
										
											2018-07-19 15:48:13 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "gpu_context_private.h"
 | 
					
						
							| 
									
										
										
										
											2018-07-17 21:11:23 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define KEEP_SINGLE_COPY 1
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | static GLenum convert_index_type_to_gl(GPUIndexBufType type) | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-05-21 18:25:30 -04:00
										 |  |  | 	static const GLenum table[] = { | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | 		[GPU_INDEX_U8] = GL_UNSIGNED_BYTE, /* GL has this, Vulkan does not */ | 
					
						
							|  |  |  | 		[GPU_INDEX_U16] = GL_UNSIGNED_SHORT, | 
					
						
							| 
									
										
										
										
											2019-02-03 14:01:45 +11:00
										 |  |  | 		[GPU_INDEX_U32] = GL_UNSIGNED_INT, | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | 	}; | 
					
						
							| 
									
										
										
										
											2017-05-21 18:25:30 -04:00
										 |  |  | 	return table[type]; | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2017-05-21 18:25:30 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 23:09:31 +10:00
										 |  |  | uint GPU_indexbuf_size_get(const GPUIndexBuf *elem) | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | #if GPU_TRACK_INDEX_RANGE
 | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | 	static const uint table[] = { | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | 		[GPU_INDEX_U8] = sizeof(GLubyte), /* GL has this, Vulkan does not */ | 
					
						
							|  |  |  | 		[GPU_INDEX_U16] = sizeof(GLushort), | 
					
						
							| 
									
										
										
										
											2019-02-03 14:01:45 +11:00
										 |  |  | 		[GPU_INDEX_U32] = sizeof(GLuint), | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | 	}; | 
					
						
							| 
									
										
										
										
											2018-07-08 13:05:41 +02:00
										 |  |  | 	return elem->index_len * table[elem->index_type]; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:51:10 +02:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2018-07-08 13:05:41 +02:00
										 |  |  | 	return elem->index_len * sizeof(GLuint); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:51:10 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  | int GPU_indexbuf_primitive_len(GPUPrimType prim_type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	switch (prim_type) { | 
					
						
							|  |  |  | 		case GPU_PRIM_POINTS: | 
					
						
							|  |  |  | 			return 1; | 
					
						
							|  |  |  | 		case GPU_PRIM_LINES: | 
					
						
							|  |  |  | 			return 2; | 
					
						
							|  |  |  | 		case GPU_PRIM_TRIS: | 
					
						
							|  |  |  | 			return 3; | 
					
						
							|  |  |  | 		case GPU_PRIM_LINES_ADJ: | 
					
						
							|  |  |  | 			return 4; | 
					
						
							| 
									
										
										
										
											2018-09-27 21:15:42 +10:00
										 |  |  | 		default: | 
					
						
							|  |  |  | 			break; | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | #if TRUST_NO_ONE
 | 
					
						
							|  |  |  | 	assert(false); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	return -1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | void GPU_indexbuf_init_ex( | 
					
						
							| 
									
										
										
										
											2018-07-18 23:09:31 +10:00
										 |  |  |         GPUIndexBufBuilder *builder, GPUPrimType prim_type, | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  |         uint index_len, uint vertex_len, bool use_prim_restart) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-14 22:06:20 +01:00
										 |  |  | 	builder->use_prim_restart = use_prim_restart; | 
					
						
							| 
									
										
										
										
											2018-07-08 13:05:41 +02:00
										 |  |  | 	builder->max_allowed_index = vertex_len - 1; | 
					
						
							|  |  |  | 	builder->max_index_len = index_len; | 
					
						
							|  |  |  | 	builder->index_len = 0; // start empty
 | 
					
						
							| 
									
										
										
										
											2018-03-14 22:06:20 +01:00
										 |  |  | 	builder->prim_type = prim_type; | 
					
						
							| 
									
										
										
										
											2018-07-31 16:54:58 +02:00
										 |  |  | 	builder->data = MEM_callocN(builder->max_index_len * sizeof(uint), "GPUIndexBuf data"); | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-03-14 22:06:20 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 23:09:31 +10:00
										 |  |  | void GPU_indexbuf_init(GPUIndexBufBuilder *builder, GPUPrimType prim_type, uint prim_len, uint vertex_len) | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  | 	int verts_per_prim = GPU_indexbuf_primitive_len(prim_type); | 
					
						
							| 
									
										
										
										
											2016-10-13 12:24:01 -04:00
										 |  |  | #if TRUST_NO_ONE
 | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  | 	assert(verts_per_prim != -1); | 
					
						
							| 
									
										
										
										
											2016-10-13 12:24:01 -04:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2018-09-27 00:53:45 -03:00
										 |  |  | 	GPU_indexbuf_init_ex(builder, prim_type, prim_len * (uint)verts_per_prim, vertex_len, false); | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 23:09:31 +10:00
										 |  |  | void GPU_indexbuf_add_generic_vert(GPUIndexBufBuilder *builder, uint v) | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | #if TRUST_NO_ONE
 | 
					
						
							|  |  |  | 	assert(builder->data != NULL); | 
					
						
							| 
									
										
										
										
											2018-07-08 13:05:41 +02:00
										 |  |  | 	assert(builder->index_len < builder->max_index_len); | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 	assert(v <= builder->max_allowed_index); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2018-07-08 13:05:41 +02:00
										 |  |  | 	builder->data[builder->index_len++] = v; | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 23:09:31 +10:00
										 |  |  | void GPU_indexbuf_add_primitive_restart(GPUIndexBufBuilder *builder) | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-14 22:06:20 +01:00
										 |  |  | #if TRUST_NO_ONE
 | 
					
						
							|  |  |  | 	assert(builder->data != NULL); | 
					
						
							| 
									
										
										
										
											2018-07-08 13:05:41 +02:00
										 |  |  | 	assert(builder->index_len < builder->max_index_len); | 
					
						
							| 
									
										
										
										
											2018-03-14 22:06:20 +01:00
										 |  |  | 	assert(builder->use_prim_restart); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | 	builder->data[builder->index_len++] = GPU_PRIM_RESTART; | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-03-14 22:06:20 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 23:09:31 +10:00
										 |  |  | void GPU_indexbuf_add_point_vert(GPUIndexBufBuilder *builder, uint v) | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | #if TRUST_NO_ONE
 | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | 	assert(builder->prim_type == GPU_PRIM_POINTS); | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | 	GPU_indexbuf_add_generic_vert(builder, v); | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 23:09:31 +10:00
										 |  |  | void GPU_indexbuf_add_line_verts(GPUIndexBufBuilder *builder, uint v1, uint v2) | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | #if TRUST_NO_ONE
 | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | 	assert(builder->prim_type == GPU_PRIM_LINES); | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 	assert(v1 != v2); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | 	GPU_indexbuf_add_generic_vert(builder, v1); | 
					
						
							|  |  |  | 	GPU_indexbuf_add_generic_vert(builder, v2); | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 23:09:31 +10:00
										 |  |  | void GPU_indexbuf_add_tri_verts(GPUIndexBufBuilder *builder, uint v1, uint v2, uint v3) | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | #if TRUST_NO_ONE
 | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | 	assert(builder->prim_type == GPU_PRIM_TRIS); | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 	assert(v1 != v2 && v2 != v3 && v3 != v1); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | 	GPU_indexbuf_add_generic_vert(builder, v1); | 
					
						
							|  |  |  | 	GPU_indexbuf_add_generic_vert(builder, v2); | 
					
						
							|  |  |  | 	GPU_indexbuf_add_generic_vert(builder, v3); | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 23:09:31 +10:00
										 |  |  | void GPU_indexbuf_add_line_adj_verts(GPUIndexBufBuilder *builder, uint v1, uint v2, uint v3, uint v4) | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-05-19 13:09:47 +02:00
										 |  |  | #if TRUST_NO_ONE
 | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | 	assert(builder->prim_type == GPU_PRIM_LINES_ADJ); | 
					
						
							| 
									
										
										
										
											2018-05-19 13:09:47 +02:00
										 |  |  | 	assert(v2 != v3); /* only the line need diff indices */ | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | 	GPU_indexbuf_add_generic_vert(builder, v1); | 
					
						
							|  |  |  | 	GPU_indexbuf_add_generic_vert(builder, v2); | 
					
						
							|  |  |  | 	GPU_indexbuf_add_generic_vert(builder, v3); | 
					
						
							|  |  |  | 	GPU_indexbuf_add_generic_vert(builder, v4); | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-05-19 13:09:47 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | #if GPU_TRACK_INDEX_RANGE
 | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | /* Everything remains 32 bit while building to keep things simple.
 | 
					
						
							|  |  |  |  * Find min/max after, then convert to smallest index type possible. */ | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 23:09:31 +10:00
										 |  |  | static uint index_range(const uint values[], uint value_len, uint *min_out, uint *max_out) | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | { | 
					
						
							|  |  |  | 	if (value_len == 0) { | 
					
						
							| 
									
										
										
										
											2017-04-21 18:32:36 +10:00
										 |  |  | 		*min_out = 0; | 
					
						
							|  |  |  | 		*max_out = 0; | 
					
						
							|  |  |  | 		return 0; | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	uint min_value = values[0]; | 
					
						
							|  |  |  | 	uint max_value = values[0]; | 
					
						
							|  |  |  | 	for (uint i = 1; i < value_len; ++i) { | 
					
						
							|  |  |  | 		const uint value = values[i]; | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | 		if (value == GPU_PRIM_RESTART) | 
					
						
							| 
									
										
										
										
											2018-03-14 22:06:20 +01:00
										 |  |  | 			continue; | 
					
						
							|  |  |  | 		else if (value < min_value) | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 			min_value = value; | 
					
						
							|  |  |  | 		else if (value > max_value) | 
					
						
							|  |  |  | 			max_value = value; | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 	*min_out = min_value; | 
					
						
							|  |  |  | 	*max_out = max_value; | 
					
						
							|  |  |  | 	return max_value - min_value; | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 23:09:31 +10:00
										 |  |  | static void squeeze_indices_byte(GPUIndexBufBuilder *builder, GPUIndexBuf *elem) | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | { | 
					
						
							|  |  |  | 	const uint *values = builder->data; | 
					
						
							|  |  |  | 	const uint index_len = elem->index_len; | 
					
						
							| 
									
										
										
										
											2018-03-17 18:23:04 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | 	/* data will never be *larger* than builder->data...
 | 
					
						
							|  |  |  | 	 * converting in place to avoid extra allocation */ | 
					
						
							| 
									
										
										
										
											2018-03-17 18:23:04 +01:00
										 |  |  | 	GLubyte *data = (GLubyte *)builder->data; | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | 	if (elem->max_index > 0xFF) { | 
					
						
							|  |  |  | 		const uint base = elem->min_index; | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 		elem->base_index = base; | 
					
						
							|  |  |  | 		elem->min_index = 0; | 
					
						
							|  |  |  | 		elem->max_index -= base; | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | 		for (uint i = 0; i < index_len; ++i) { | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | 			data[i] = (values[i] == GPU_PRIM_RESTART) ? 0xFF : (GLubyte)(values[i] - base); | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 		elem->base_index = 0; | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | 		for (uint i = 0; i < index_len; ++i) { | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 			data[i] = (GLubyte)(values[i]); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 23:09:31 +10:00
										 |  |  | static void squeeze_indices_short(GPUIndexBufBuilder *builder, GPUIndexBuf *elem) | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | { | 
					
						
							|  |  |  | 	const uint *values = builder->data; | 
					
						
							|  |  |  | 	const uint index_len = elem->index_len; | 
					
						
							| 
									
										
										
										
											2018-03-17 18:23:04 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | 	/* data will never be *larger* than builder->data...
 | 
					
						
							|  |  |  | 	 * converting in place to avoid extra allocation */ | 
					
						
							| 
									
										
										
										
											2018-03-17 18:23:04 +01:00
										 |  |  | 	GLushort *data = (GLushort *)builder->data; | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | 	if (elem->max_index > 0xFFFF) { | 
					
						
							|  |  |  | 		const uint base = elem->min_index; | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 		elem->base_index = base; | 
					
						
							|  |  |  | 		elem->min_index = 0; | 
					
						
							|  |  |  | 		elem->max_index -= base; | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | 		for (uint i = 0; i < index_len; ++i) { | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | 			data[i] = (values[i] == GPU_PRIM_RESTART) ? 0xFFFF : (GLushort)(values[i] - base); | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 		elem->base_index = 0; | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | 		for (uint i = 0; i < index_len; ++i) { | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 			data[i] = (GLushort)(values[i]); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | #endif /* GPU_TRACK_INDEX_RANGE */
 | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 23:09:31 +10:00
										 |  |  | GPUIndexBuf *GPU_indexbuf_build(GPUIndexBufBuilder *builder) | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-07-31 16:54:58 +02:00
										 |  |  | 	GPUIndexBuf *elem = MEM_callocN(sizeof(GPUIndexBuf), "GPUIndexBuf"); | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | 	GPU_indexbuf_build_in_place(builder, elem); | 
					
						
							| 
									
										
										
										
											2016-10-23 23:16:54 -04:00
										 |  |  | 	return elem; | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-10-23 23:16:54 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 23:09:31 +10:00
										 |  |  | void GPU_indexbuf_build_in_place(GPUIndexBufBuilder *builder, GPUIndexBuf *elem) | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | #if TRUST_NO_ONE
 | 
					
						
							|  |  |  | 	assert(builder->data != NULL); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2018-07-08 13:05:41 +02:00
										 |  |  | 	elem->index_len = builder->index_len; | 
					
						
							| 
									
										
										
										
											2018-03-14 22:06:20 +01:00
										 |  |  | 	elem->use_prim_restart = builder->use_prim_restart; | 
					
						
							| 
									
										
										
										
											2018-12-08 18:15:57 +01:00
										 |  |  | 	elem->ibo_id = 0; /* Created at first use. */ | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | #if GPU_TRACK_INDEX_RANGE
 | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | 	uint range = index_range(builder->data, builder->index_len, &elem->min_index, &elem->max_index); | 
					
						
							| 
									
										
										
										
											2018-03-14 22:06:20 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | 	/* count the primitive restart index. */ | 
					
						
							|  |  |  | 	if (elem->use_prim_restart) { | 
					
						
							| 
									
										
										
										
											2018-03-14 22:06:20 +01:00
										 |  |  | 		range += 1; | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | 	if (range <= 0xFF) { | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | 		elem->index_type = GPU_INDEX_U8; | 
					
						
							| 
									
										
										
										
											2018-03-17 18:23:04 +01:00
										 |  |  | 		squeeze_indices_byte(builder, elem); | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else if (range <= 0xFFFF) { | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | 		elem->index_type = GPU_INDEX_U16; | 
					
						
							| 
									
										
										
										
											2018-03-17 18:23:04 +01:00
										 |  |  | 		squeeze_indices_short(builder, elem); | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | 		elem->index_type = GPU_INDEX_U32; | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 		elem->base_index = 0; | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-05-21 18:25:30 -04:00
										 |  |  | 	elem->gl_index_type = convert_index_type_to_gl(elem->index_type); | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-08 18:15:57 +01:00
										 |  |  | 	/* Transfer data ownership to GPUIndexBuf.
 | 
					
						
							|  |  |  | 	 * It will be uploaded upon first use. */ | 
					
						
							|  |  |  | 	elem->data = builder->data; | 
					
						
							| 
									
										
										
										
											2016-09-13 02:41:43 -04:00
										 |  |  | 	builder->data = NULL; | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | 	/* other fields are safe to leave */ | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2016-10-23 23:16:54 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-08 18:15:57 +01:00
										 |  |  | static void indexbuf_upload_data(GPUIndexBuf *elem) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* send data to GPU */ | 
					
						
							|  |  |  | 	glBufferData(GL_ELEMENT_ARRAY_BUFFER, GPU_indexbuf_size_get(elem), elem->data, GL_STATIC_DRAW); | 
					
						
							|  |  |  | 	/* No need to keep copy of data in system memory. */ | 
					
						
							|  |  |  | 	MEM_freeN(elem->data); | 
					
						
							|  |  |  | 	elem->data = NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 23:09:31 +10:00
										 |  |  | void GPU_indexbuf_use(GPUIndexBuf *elem) | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-12-08 18:15:57 +01:00
										 |  |  | 	if (elem->ibo_id == 0) { | 
					
						
							|  |  |  | 		elem->ibo_id = GPU_buf_alloc(); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-12-08 15:53:23 +01:00
										 |  |  | 	glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, elem->ibo_id); | 
					
						
							| 
									
										
										
										
											2018-12-08 18:15:57 +01:00
										 |  |  | 	if (elem->data != NULL) { | 
					
						
							|  |  |  | 		indexbuf_upload_data(elem); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-03-17 18:23:04 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 23:09:31 +10:00
										 |  |  | void GPU_indexbuf_discard(GPUIndexBuf *elem) | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-12-08 15:53:23 +01:00
										 |  |  | 	if (elem->ibo_id) { | 
					
						
							|  |  |  | 		GPU_buf_free(elem->ibo_id); | 
					
						
							| 
									
										
										
										
											2016-10-23 23:16:54 -04:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-12-08 18:15:57 +01:00
										 |  |  | 	if (elem->data) { | 
					
						
							|  |  |  | 		MEM_freeN(elem->data); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-07-31 16:54:58 +02:00
										 |  |  | 	MEM_freeN(elem); | 
					
						
							| 
									
										
										
										
											2018-07-17 14:46:44 +02:00
										 |  |  | } |