| 
									
										
										
										
											2016-09-22 12:11:40 +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, | 
					
						
							| 
									
										
										
										
											2019-07-07 15:38:41 +10:00
										 |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  |  * 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) 2012 Blender Foundation. | 
					
						
							|  |  |  |  * All rights reserved. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-18 08:08:12 +11:00
										 |  |  | /** \file
 | 
					
						
							|  |  |  |  * \ingroup gpu | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-17 21:11:23 +02:00
										 |  |  | #include "GPU_shader_interface.h"
 | 
					
						
							| 
									
										
										
										
											2017-08-21 01:39:03 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-14 15:27:10 +02:00
										 |  |  | #include "gpu_context_private.h"
 | 
					
						
							|  |  |  | #include "gpu_matrix_private.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-21 17:27:17 -04:00
										 |  |  | #define SUPPRESS_GENERIC_MATRIX_API
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | #define USE_GPU_PY_MATRIX_API /* only so values are declared */
 | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | #include "GPU_matrix.h"
 | 
					
						
							| 
									
										
										
										
											2017-08-20 17:23:49 +10:00
										 |  |  | #undef USE_GPU_PY_MATRIX_API
 | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "BLI_math_matrix.h"
 | 
					
						
							|  |  |  | #include "BLI_math_rotation.h"
 | 
					
						
							|  |  |  | #include "BLI_math_vector.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-14 15:27:10 +02:00
										 |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | #define MATRIX_STACK_DEPTH 32
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef float Mat4[4][4]; | 
					
						
							|  |  |  | typedef float Mat3[3][3]; | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-19 10:13:10 +02:00
										 |  |  | typedef struct MatrixStack { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   Mat4 stack[MATRIX_STACK_DEPTH]; | 
					
						
							|  |  |  |   uint top; | 
					
						
							| 
									
										
										
										
											2017-04-19 10:13:10 +02:00
										 |  |  | } MatrixStack; | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-14 15:27:10 +02:00
										 |  |  | typedef struct GPUMatrixState { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   MatrixStack model_view_stack; | 
					
						
							|  |  |  |   MatrixStack projection_stack; | 
					
						
							| 
									
										
										
										
											2016-10-20 14:17:54 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   bool dirty; | 
					
						
							| 
									
										
										
										
											2016-10-20 14:17:54 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* TODO: cache of derived matrices (Normal, MVP, inverse MVP, etc)
 | 
					
						
							|  |  |  |    * generate as needed for shaders, invalidate when original matrices change | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * TODO: separate Model from View transform? Batches/objects have model, | 
					
						
							|  |  |  |    * camera/eye has view & projection | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2019-08-14 15:27:10 +02:00
										 |  |  | } GPUMatrixState; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define ModelViewStack gpu_context_active_matrix_state_get()->model_view_stack
 | 
					
						
							|  |  |  | #define ModelView ModelViewStack.stack[ModelViewStack.top]
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-14 15:27:10 +02:00
										 |  |  | #define ProjectionStack gpu_context_active_matrix_state_get()->projection_stack
 | 
					
						
							|  |  |  | #define Projection ProjectionStack.stack[ProjectionStack.top]
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GPUMatrixState *GPU_matrix_state_create(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | #define MATRIX_4X4_IDENTITY \
 | 
					
						
							|  |  |  |   { \ | 
					
						
							|  |  |  |     {1.0f, 0.0f, 0.0f, 0.0f}, {0.0f, 1.0f, 0.0f, 0.0f}, {0.0f, 0.0f, 1.0f, 0.0f}, \ | 
					
						
							|  |  |  |     { \ | 
					
						
							|  |  |  |       0.0f, 0.0f, 0.0f, 1.0f \ | 
					
						
							|  |  |  |     } \ | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-04-14 14:13:38 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-30 23:50:43 +02:00
										 |  |  |   GPUMatrixState *state = (GPUMatrixState *)MEM_mallocN(sizeof(*state), __func__); | 
					
						
							| 
									
										
										
										
											2019-08-14 15:27:10 +02:00
										 |  |  |   const MatrixStack identity_stack = {{MATRIX_4X4_IDENTITY}, 0}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   state->model_view_stack = state->projection_stack = identity_stack; | 
					
						
							|  |  |  |   state->dirty = true; | 
					
						
							| 
									
										
										
										
											2017-04-14 14:13:38 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | #undef MATRIX_4X4_IDENTITY
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-14 15:27:10 +02:00
										 |  |  |   return state; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2017-04-19 10:13:10 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-14 15:27:10 +02:00
										 |  |  | void GPU_matrix_state_discard(GPUMatrixState *state) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   MEM_freeN(state); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void gpu_matrix_state_active_set_dirty(bool value) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   GPUMatrixState *state = gpu_context_active_matrix_state_get(); | 
					
						
							|  |  |  |   state->dirty = value; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | void GPU_matrix_reset(void) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-08-14 15:27:10 +02:00
										 |  |  |   GPUMatrixState *state = gpu_context_active_matrix_state_get(); | 
					
						
							|  |  |  |   state->model_view_stack.top = 0; | 
					
						
							|  |  |  |   state->projection_stack.top = 0; | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   unit_m4(ModelView); | 
					
						
							|  |  |  |   unit_m4(Projection); | 
					
						
							| 
									
										
										
										
											2019-08-14 15:27:10 +02:00
										 |  |  |   gpu_matrix_state_active_set_dirty(true); | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef WITH_GPU_SAFETY
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | /* Check if matrix is numerically good */ | 
					
						
							|  |  |  | static void checkmat(cosnt float *m) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const int n = 16; | 
					
						
							|  |  |  |   for (int i = 0; i < n; i++) { | 
					
						
							|  |  |  | #  if _MSC_VER
 | 
					
						
							|  |  |  |     BLI_assert(_finite(m[i])); | 
					
						
							|  |  |  | #  else
 | 
					
						
							|  |  |  |     BLI_assert(!isinf(m[i])); | 
					
						
							|  |  |  | #  endif
 | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | #  define CHECKMAT(m) checkmat((const float *)m)
 | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | #  define CHECKMAT(m)
 | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | void GPU_matrix_push(void) | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BLI_assert(ModelViewStack.top + 1 < MATRIX_STACK_DEPTH); | 
					
						
							|  |  |  |   ModelViewStack.top++; | 
					
						
							|  |  |  |   copy_m4_m4(ModelView, ModelViewStack.stack[ModelViewStack.top - 1]); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | void GPU_matrix_pop(void) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BLI_assert(ModelViewStack.top > 0); | 
					
						
							|  |  |  |   ModelViewStack.top--; | 
					
						
							| 
									
										
										
										
											2019-08-14 15:27:10 +02:00
										 |  |  |   gpu_matrix_state_active_set_dirty(true); | 
					
						
							| 
									
										
										
										
											2017-04-19 10:13:10 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | void GPU_matrix_push_projection(void) | 
					
						
							| 
									
										
										
										
											2017-04-19 10:13:10 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BLI_assert(ProjectionStack.top + 1 < MATRIX_STACK_DEPTH); | 
					
						
							|  |  |  |   ProjectionStack.top++; | 
					
						
							|  |  |  |   copy_m4_m4(Projection, ProjectionStack.stack[ProjectionStack.top - 1]); | 
					
						
							| 
									
										
										
										
											2017-04-19 10:13:10 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | void GPU_matrix_pop_projection(void) | 
					
						
							| 
									
										
										
										
											2017-04-19 10:13:10 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   BLI_assert(ProjectionStack.top > 0); | 
					
						
							|  |  |  |   ProjectionStack.top--; | 
					
						
							| 
									
										
										
										
											2019-08-14 15:27:10 +02:00
										 |  |  |   gpu_matrix_state_active_set_dirty(true); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | void GPU_matrix_set(const float m[4][4]) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   copy_m4_m4(ModelView, m); | 
					
						
							|  |  |  |   CHECKMAT(ModelView3D); | 
					
						
							| 
									
										
										
										
											2019-08-14 15:27:10 +02:00
										 |  |  |   gpu_matrix_state_active_set_dirty(true); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | void GPU_matrix_identity_projection_set(void) | 
					
						
							| 
									
										
										
										
											2017-04-19 10:46:05 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   unit_m4(Projection); | 
					
						
							|  |  |  |   CHECKMAT(Projection3D); | 
					
						
							| 
									
										
										
										
											2019-08-14 15:27:10 +02:00
										 |  |  |   gpu_matrix_state_active_set_dirty(true); | 
					
						
							| 
									
										
										
										
											2017-04-19 10:46:05 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | void GPU_matrix_projection_set(const float m[4][4]) | 
					
						
							| 
									
										
										
										
											2017-03-21 18:10:20 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   copy_m4_m4(Projection, m); | 
					
						
							|  |  |  |   CHECKMAT(Projection3D); | 
					
						
							| 
									
										
										
										
											2019-08-14 15:27:10 +02:00
										 |  |  |   gpu_matrix_state_active_set_dirty(true); | 
					
						
							| 
									
										
										
										
											2017-03-21 18:10:20 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | void GPU_matrix_identity_set(void) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   unit_m4(ModelView); | 
					
						
							| 
									
										
										
										
											2019-08-14 15:27:10 +02:00
										 |  |  |   gpu_matrix_state_active_set_dirty(true); | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | void GPU_matrix_translate_2f(float x, float y) | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   Mat4 m; | 
					
						
							|  |  |  |   unit_m4(m); | 
					
						
							|  |  |  |   m[3][0] = x; | 
					
						
							|  |  |  |   m[3][1] = y; | 
					
						
							|  |  |  |   GPU_matrix_mul(m); | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | void GPU_matrix_translate_2fv(const float vec[2]) | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   GPU_matrix_translate_2f(vec[0], vec[1]); | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | void GPU_matrix_translate_3f(float x, float y, float z) | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-10-20 14:17:54 -04:00
										 |  |  | #if 1
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   translate_m4(ModelView, x, y, z); | 
					
						
							|  |  |  |   CHECKMAT(ModelView); | 
					
						
							| 
									
										
										
										
											2016-10-11 21:13:13 -04:00
										 |  |  | #else /* above works well in early testing, below is generic version */
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   Mat4 m; | 
					
						
							|  |  |  |   unit_m4(m); | 
					
						
							|  |  |  |   m[3][0] = x; | 
					
						
							|  |  |  |   m[3][1] = y; | 
					
						
							|  |  |  |   m[3][2] = z; | 
					
						
							|  |  |  |   GPU_matrix_mul(m); | 
					
						
							| 
									
										
										
										
											2016-10-11 21:13:13 -04:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2019-08-14 15:27:10 +02:00
										 |  |  |   gpu_matrix_state_active_set_dirty(true); | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | void GPU_matrix_translate_3fv(const float vec[3]) | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   GPU_matrix_translate_3f(vec[0], vec[1], vec[2]); | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | void GPU_matrix_scale_1f(float factor) | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   Mat4 m; | 
					
						
							|  |  |  |   scale_m4_fl(m, factor); | 
					
						
							|  |  |  |   GPU_matrix_mul(m); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | void GPU_matrix_scale_2f(float x, float y) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   Mat4 m = {{0.0f}}; | 
					
						
							|  |  |  |   m[0][0] = x; | 
					
						
							|  |  |  |   m[1][1] = y; | 
					
						
							|  |  |  |   m[2][2] = 1.0f; | 
					
						
							|  |  |  |   m[3][3] = 1.0f; | 
					
						
							|  |  |  |   GPU_matrix_mul(m); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | void GPU_matrix_scale_2fv(const float vec[2]) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   GPU_matrix_scale_2f(vec[0], vec[1]); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | void GPU_matrix_scale_3f(float x, float y, float z) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   Mat4 m = {{0.0f}}; | 
					
						
							|  |  |  |   m[0][0] = x; | 
					
						
							|  |  |  |   m[1][1] = y; | 
					
						
							|  |  |  |   m[2][2] = z; | 
					
						
							|  |  |  |   m[3][3] = 1.0f; | 
					
						
							|  |  |  |   GPU_matrix_mul(m); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | void GPU_matrix_scale_3fv(const float vec[3]) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   GPU_matrix_scale_3f(vec[0], vec[1], vec[2]); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | void GPU_matrix_mul(const float m[4][4]) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   mul_m4_m4_post(ModelView, m); | 
					
						
							|  |  |  |   CHECKMAT(ModelView); | 
					
						
							| 
									
										
										
										
											2019-08-14 15:27:10 +02:00
										 |  |  |   gpu_matrix_state_active_set_dirty(true); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | void GPU_matrix_rotate_2d(float deg) | 
					
						
							| 
									
										
										
										
											2017-03-21 00:08:09 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* essentially RotateAxis('Z')
 | 
					
						
							|  |  |  |    * TODO: simpler math for 2D case | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   rotate_m4(ModelView, 'Z', DEG2RADF(deg)); | 
					
						
							| 
									
										
										
										
											2017-03-21 00:08:09 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | void GPU_matrix_rotate_3f(float deg, float x, float y, float z) | 
					
						
							| 
									
										
										
										
											2017-02-12 19:23:08 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   const float axis[3] = {x, y, z}; | 
					
						
							|  |  |  |   GPU_matrix_rotate_3fv(deg, axis); | 
					
						
							| 
									
										
										
										
											2017-02-12 19:23:08 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | void GPU_matrix_rotate_3fv(float deg, const float axis[3]) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   Mat4 m; | 
					
						
							|  |  |  |   axis_angle_to_mat4(m, axis, DEG2RADF(deg)); | 
					
						
							|  |  |  |   GPU_matrix_mul(m); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | void GPU_matrix_rotate_axis(float deg, char axis) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* rotate_m4 works in place */ | 
					
						
							|  |  |  |   rotate_m4(ModelView, axis, DEG2RADF(deg)); | 
					
						
							|  |  |  |   CHECKMAT(ModelView); | 
					
						
							| 
									
										
										
										
											2019-08-14 15:27:10 +02:00
										 |  |  |   gpu_matrix_state_active_set_dirty(true); | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | static void mat4_ortho_set( | 
					
						
							|  |  |  |     float m[4][4], float left, float right, float bottom, float top, float near, float far) | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   m[0][0] = 2.0f / (right - left); | 
					
						
							|  |  |  |   m[1][0] = 0.0f; | 
					
						
							|  |  |  |   m[2][0] = 0.0f; | 
					
						
							|  |  |  |   m[3][0] = -(right + left) / (right - left); | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   m[0][1] = 0.0f; | 
					
						
							|  |  |  |   m[1][1] = 2.0f / (top - bottom); | 
					
						
							|  |  |  |   m[2][1] = 0.0f; | 
					
						
							|  |  |  |   m[3][1] = -(top + bottom) / (top - bottom); | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   m[0][2] = 0.0f; | 
					
						
							|  |  |  |   m[1][2] = 0.0f; | 
					
						
							|  |  |  |   m[2][2] = -2.0f / (far - near); | 
					
						
							|  |  |  |   m[3][2] = -(far + near) / (far - near); | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   m[0][3] = 0.0f; | 
					
						
							|  |  |  |   m[1][3] = 0.0f; | 
					
						
							|  |  |  |   m[2][3] = 0.0f; | 
					
						
							|  |  |  |   m[3][3] = 1.0f; | 
					
						
							| 
									
										
										
										
											2016-10-20 14:17:54 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-14 15:27:10 +02:00
										 |  |  |   gpu_matrix_state_active_set_dirty(true); | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | static void mat4_frustum_set( | 
					
						
							|  |  |  |     float m[4][4], float left, float right, float bottom, float top, float near, float far) | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   m[0][0] = 2.0f * near / (right - left); | 
					
						
							|  |  |  |   m[1][0] = 0.0f; | 
					
						
							|  |  |  |   m[2][0] = (right + left) / (right - left); | 
					
						
							|  |  |  |   m[3][0] = 0.0f; | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   m[0][1] = 0.0f; | 
					
						
							|  |  |  |   m[1][1] = 2.0f * near / (top - bottom); | 
					
						
							|  |  |  |   m[2][1] = (top + bottom) / (top - bottom); | 
					
						
							|  |  |  |   m[3][1] = 0.0f; | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   m[0][2] = 0.0f; | 
					
						
							|  |  |  |   m[1][2] = 0.0f; | 
					
						
							|  |  |  |   m[2][2] = -(far + near) / (far - near); | 
					
						
							|  |  |  |   m[3][2] = -2.0f * far * near / (far - near); | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   m[0][3] = 0.0f; | 
					
						
							|  |  |  |   m[1][3] = 0.0f; | 
					
						
							|  |  |  |   m[2][3] = -1.0f; | 
					
						
							|  |  |  |   m[3][3] = 0.0f; | 
					
						
							| 
									
										
										
										
											2016-10-20 14:17:54 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-14 15:27:10 +02:00
										 |  |  |   gpu_matrix_state_active_set_dirty(true); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | static void mat4_look_from_origin(float m[4][4], float lookdir[3], float camup[3]) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* This function is loosely based on Mesa implementation.
 | 
					
						
							| 
									
										
										
										
											2019-04-18 07:21:26 +02:00
										 |  |  |    * | 
					
						
							|  |  |  |    * SGI FREE SOFTWARE LICENSE B (Version 2.0, Sept. 18, 2008) | 
					
						
							|  |  |  |    * Copyright (C) 1991-2000 Silicon Graphics, Inc. All Rights Reserved. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * Permission is hereby granted, free of charge, to any person obtaining a | 
					
						
							|  |  |  |    * copy of this software and associated documentation files (the "Software"), | 
					
						
							|  |  |  |    * to deal in the Software without restriction, including without limitation | 
					
						
							|  |  |  |    * the rights to use, copy, modify, merge, publish, distribute, sublicense, | 
					
						
							|  |  |  |    * and/or sell copies of the Software, and to permit persons to whom the | 
					
						
							|  |  |  |    * Software is furnished to do so, subject to the following conditions: | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * The above copyright notice including the dates of first publication and | 
					
						
							|  |  |  |    * either this permission notice or a reference to | 
					
						
							|  |  |  |    * http://oss.sgi.com/projects/FreeB/
 | 
					
						
							|  |  |  |    * shall be included in all copies or substantial portions of the Software. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS | 
					
						
							|  |  |  |    * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | 
					
						
							|  |  |  |    * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL | 
					
						
							|  |  |  |    * SILICON GRAPHICS, INC. BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, | 
					
						
							|  |  |  |    * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF | 
					
						
							|  |  |  |    * OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | 
					
						
							|  |  |  |    * SOFTWARE. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * Except as contained in this notice, the name of Silicon Graphics, Inc. | 
					
						
							|  |  |  |    * shall not be used in advertising or otherwise to promote the sale, use or | 
					
						
							|  |  |  |    * other dealings in this Software without prior written authorization from | 
					
						
							|  |  |  |    * Silicon Graphics, Inc. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   float side[3]; | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   normalize_v3(lookdir); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   cross_v3_v3v3(side, lookdir, camup); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   normalize_v3(side); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   cross_v3_v3v3(camup, side, lookdir); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   m[0][0] = side[0]; | 
					
						
							|  |  |  |   m[1][0] = side[1]; | 
					
						
							|  |  |  |   m[2][0] = side[2]; | 
					
						
							|  |  |  |   m[3][0] = 0.0f; | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   m[0][1] = camup[0]; | 
					
						
							|  |  |  |   m[1][1] = camup[1]; | 
					
						
							|  |  |  |   m[2][1] = camup[2]; | 
					
						
							|  |  |  |   m[3][1] = 0.0f; | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   m[0][2] = -lookdir[0]; | 
					
						
							|  |  |  |   m[1][2] = -lookdir[1]; | 
					
						
							|  |  |  |   m[2][2] = -lookdir[2]; | 
					
						
							|  |  |  |   m[3][2] = 0.0f; | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   m[0][3] = 0.0f; | 
					
						
							|  |  |  |   m[1][3] = 0.0f; | 
					
						
							|  |  |  |   m[2][3] = 0.0f; | 
					
						
							|  |  |  |   m[3][3] = 1.0f; | 
					
						
							| 
									
										
										
										
											2016-10-20 14:17:54 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-14 15:27:10 +02:00
										 |  |  |   gpu_matrix_state_active_set_dirty(true); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | void GPU_matrix_ortho_set(float left, float right, float bottom, float top, float near, float far) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   mat4_ortho_set(Projection, left, right, bottom, top, near, far); | 
					
						
							|  |  |  |   CHECKMAT(Projection); | 
					
						
							| 
									
										
										
										
											2019-08-14 15:27:10 +02:00
										 |  |  |   gpu_matrix_state_active_set_dirty(true); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-04 21:55:57 +11:00
										 |  |  | void GPU_matrix_ortho_set_z(float near, float far) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   CHECKMAT(Projection); | 
					
						
							|  |  |  |   Projection[2][2] = -2.0f / (far - near); | 
					
						
							|  |  |  |   Projection[3][2] = -(far + near) / (far - near); | 
					
						
							|  |  |  |   gpu_matrix_state_active_set_dirty(true); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | void GPU_matrix_ortho_2d_set(float left, float right, float bottom, float top) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   Mat4 m; | 
					
						
							|  |  |  |   mat4_ortho_set(m, left, right, bottom, top, -1.0f, 1.0f); | 
					
						
							|  |  |  |   CHECKMAT(Projection2D); | 
					
						
							| 
									
										
										
										
											2019-08-14 15:27:10 +02:00
										 |  |  |   gpu_matrix_state_active_set_dirty(true); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | void GPU_matrix_frustum_set( | 
					
						
							|  |  |  |     float left, float right, float bottom, float top, float near, float far) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   mat4_frustum_set(Projection, left, right, bottom, top, near, far); | 
					
						
							|  |  |  |   CHECKMAT(Projection); | 
					
						
							| 
									
										
										
										
											2019-08-14 15:27:10 +02:00
										 |  |  |   gpu_matrix_state_active_set_dirty(true); | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | void GPU_matrix_perspective_set(float fovy, float aspect, float near, float far) | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   float half_height = tanf(fovy * (float)(M_PI / 360.0)) * near; | 
					
						
							|  |  |  |   float half_width = half_height * aspect; | 
					
						
							|  |  |  |   GPU_matrix_frustum_set(-half_width, +half_width, -half_height, +half_height, near, far); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | void GPU_matrix_look_at(float eyeX, | 
					
						
							|  |  |  |                         float eyeY, | 
					
						
							|  |  |  |                         float eyeZ, | 
					
						
							|  |  |  |                         float centerX, | 
					
						
							|  |  |  |                         float centerY, | 
					
						
							|  |  |  |                         float centerZ, | 
					
						
							|  |  |  |                         float upX, | 
					
						
							|  |  |  |                         float upY, | 
					
						
							|  |  |  |                         float upZ) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   Mat4 cm; | 
					
						
							|  |  |  |   float lookdir[3]; | 
					
						
							|  |  |  |   float camup[3] = {upX, upY, upZ}; | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   lookdir[0] = centerX - eyeX; | 
					
						
							|  |  |  |   lookdir[1] = centerY - eyeY; | 
					
						
							|  |  |  |   lookdir[2] = centerZ - eyeZ; | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   mat4_look_from_origin(cm, lookdir, camup); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   GPU_matrix_mul(cm); | 
					
						
							|  |  |  |   GPU_matrix_translate_3f(-eyeX, -eyeY, -eyeZ); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | void GPU_matrix_project(const float world[3], | 
					
						
							|  |  |  |                         const float model[4][4], | 
					
						
							|  |  |  |                         const float proj[4][4], | 
					
						
							|  |  |  |                         const int view[4], | 
					
						
							|  |  |  |                         float win[3]) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   float v[4]; | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   mul_v4_m4v3(v, model, world); | 
					
						
							|  |  |  |   mul_m4_v4(proj, v); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (v[3] != 0.0f) { | 
					
						
							|  |  |  |     mul_v3_fl(v, 1.0f / v[3]); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-02-25 21:58:23 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   win[0] = view[0] + (view[2] * (v[0] + 1)) * 0.5f; | 
					
						
							|  |  |  |   win[1] = view[1] + (view[3] * (v[1] + 1)) * 0.5f; | 
					
						
							|  |  |  |   win[2] = (v[2] + 1) * 0.5f; | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-22 10:32:57 -03:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * The same result could be obtained as follows: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \code{.c} | 
					
						
							|  |  |  |  * float projinv[4][4]; | 
					
						
							|  |  |  |  * invert_m4_m4(projinv, projmat); | 
					
						
							|  |  |  |  * co[0] = 2 * co[0] - 1; | 
					
						
							|  |  |  |  * co[1] = 2 * co[1] - 1; | 
					
						
							|  |  |  |  * co[2] = 2 * co[2] - 1; | 
					
						
							|  |  |  |  * mul_project_m4_v3(projinv, co); | 
					
						
							|  |  |  |  * \endcode | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * But that solution loses much precision. | 
					
						
							|  |  |  |  * Therefore, get the same result without inverting the matrix. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2019-06-22 13:19:11 +10:00
										 |  |  | static void gpu_mul_invert_projmat_m4_unmapped_v3_with_precalc( | 
					
						
							|  |  |  |     const struct GPUMatrixUnproject_Precalc *precalc, float co[3]) | 
					
						
							| 
									
										
										
										
											2019-03-22 10:32:57 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-06-22 13:19:11 +10:00
										 |  |  |   /* 'precalc->dims' is the result of 'projmat_dimensions(proj, ...)'. */ | 
					
						
							|  |  |  |   co[0] = precalc->dims.xmin + co[0] * (precalc->dims.xmax - precalc->dims.xmin); | 
					
						
							|  |  |  |   co[1] = precalc->dims.ymin + co[1] * (precalc->dims.ymax - precalc->dims.ymin); | 
					
						
							| 
									
										
										
										
											2019-03-22 10:32:57 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-22 13:19:11 +10:00
										 |  |  |   if (precalc->is_persp) { | 
					
						
							|  |  |  |     co[2] = precalc->dims.zmax * precalc->dims.zmin / | 
					
						
							|  |  |  |             (precalc->dims.zmax + co[2] * (precalc->dims.zmin - precalc->dims.zmax)); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     co[0] *= co[2]; | 
					
						
							|  |  |  |     co[1] *= co[2]; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							| 
									
										
										
										
											2019-06-22 13:19:11 +10:00
										 |  |  |     co[2] = precalc->dims.zmin + co[2] * (precalc->dims.zmax - precalc->dims.zmin); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							|  |  |  |   co[2] *= -1; | 
					
						
							| 
									
										
										
										
											2019-03-22 10:32:57 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-22 13:19:11 +10:00
										 |  |  | bool GPU_matrix_unproject_precalc(struct GPUMatrixUnproject_Precalc *precalc, | 
					
						
							|  |  |  |                                   const float model[4][4], | 
					
						
							|  |  |  |                                   const float proj[4][4], | 
					
						
							|  |  |  |                                   const int view[4]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   precalc->is_persp = proj[3][3] == 0.0f; | 
					
						
							| 
									
										
										
										
											2020-07-19 21:27:13 +10:00
										 |  |  |   projmat_dimensions_db(proj, | 
					
						
							|  |  |  |                         &precalc->dims.xmin, | 
					
						
							|  |  |  |                         &precalc->dims.xmax, | 
					
						
							|  |  |  |                         &precalc->dims.ymin, | 
					
						
							|  |  |  |                         &precalc->dims.ymax, | 
					
						
							|  |  |  |                         &precalc->dims.zmin, | 
					
						
							|  |  |  |                         &precalc->dims.zmax); | 
					
						
							| 
									
										
										
										
											2019-12-19 11:44:50 -03:00
										 |  |  |   if (isinf(precalc->dims.zmax)) { | 
					
						
							|  |  |  |     /* We cannot retrieve the actual value of the clip_end.
 | 
					
						
							|  |  |  |      * Use `FLT_MAX` to avoid nans. */ | 
					
						
							|  |  |  |     precalc->dims.zmax = FLT_MAX; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-06-22 13:19:11 +10:00
										 |  |  |   for (int i = 0; i < 4; i++) { | 
					
						
							|  |  |  |     precalc->view[i] = (float)view[i]; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (!invert_m4_m4(precalc->model_inverted, model)) { | 
					
						
							|  |  |  |     unit_m4(precalc->model_inverted); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-22 13:19:11 +10:00
										 |  |  | void GPU_matrix_unproject_with_precalc(const struct GPUMatrixUnproject_Precalc *precalc, | 
					
						
							|  |  |  |                                        const float win[3], | 
					
						
							|  |  |  |                                        float r_world[3]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   float in[3] = { | 
					
						
							|  |  |  |       (win[0] - precalc->view[0]) / precalc->view[2], | 
					
						
							|  |  |  |       (win[1] - precalc->view[1]) / precalc->view[3], | 
					
						
							|  |  |  |       win[2], | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  |   gpu_mul_invert_projmat_m4_unmapped_v3_with_precalc(precalc, in); | 
					
						
							|  |  |  |   mul_v3_m4v3(r_world, precalc->model_inverted, in); | 
					
						
							| 
									
										
										
										
											2019-05-15 14:16:35 +10:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-03-22 10:32:57 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-15 14:16:35 +10:00
										 |  |  | bool GPU_matrix_unproject(const float win[3], | 
					
						
							|  |  |  |                           const float model[4][4], | 
					
						
							|  |  |  |                           const float proj[4][4], | 
					
						
							|  |  |  |                           const int view[4], | 
					
						
							| 
									
										
										
										
											2019-06-22 13:19:11 +10:00
										 |  |  |                           float r_world[3]) | 
					
						
							| 
									
										
										
										
											2019-05-15 14:16:35 +10:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-06-22 13:19:11 +10:00
										 |  |  |   struct GPUMatrixUnproject_Precalc precalc; | 
					
						
							|  |  |  |   if (!GPU_matrix_unproject_precalc(&precalc, model, proj, view)) { | 
					
						
							|  |  |  |     zero_v3(r_world); | 
					
						
							| 
									
										
										
										
											2019-05-15 14:16:35 +10:00
										 |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-06-22 13:19:11 +10:00
										 |  |  |   GPU_matrix_unproject_with_precalc(&precalc, win, r_world); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-10-09 23:03:35 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | const float (*GPU_matrix_model_view_get(float m[4][4]))[4] | 
					
						
							| 
									
										
										
										
											2016-10-09 23:03:35 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (m) { | 
					
						
							|  |  |  |     copy_m4_m4(m, ModelView); | 
					
						
							|  |  |  |     return m; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     return ModelView; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-10-09 23:03:35 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | const float (*GPU_matrix_projection_get(float m[4][4]))[4] | 
					
						
							| 
									
										
										
										
											2016-10-09 23:03:35 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (m) { | 
					
						
							|  |  |  |     copy_m4_m4(m, Projection); | 
					
						
							|  |  |  |     return m; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     return Projection; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-10-09 23:03:35 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | const float (*GPU_matrix_model_view_projection_get(float m[4][4]))[4] | 
					
						
							| 
									
										
										
										
											2016-10-09 23:03:35 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (m == NULL) { | 
					
						
							|  |  |  |     static Mat4 temp; | 
					
						
							|  |  |  |     m = temp; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-02-18 19:53:55 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   mul_m4_m4m4(m, Projection, ModelView); | 
					
						
							|  |  |  |   return m; | 
					
						
							| 
									
										
										
										
											2016-10-09 23:03:35 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | const float (*GPU_matrix_normal_get(float m[3][3]))[3] | 
					
						
							| 
									
										
										
										
											2016-10-23 23:37:53 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (m == NULL) { | 
					
						
							|  |  |  |     static Mat3 temp3; | 
					
						
							|  |  |  |     m = temp3; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-10-23 23:37:53 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   copy_m3_m4(m, (const float(*)[4])GPU_matrix_model_view_get(NULL)); | 
					
						
							| 
									
										
										
										
											2016-10-23 23:37:53 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   invert_m3(m); | 
					
						
							|  |  |  |   transpose_m3(m); | 
					
						
							| 
									
										
										
										
											2016-10-23 23:37:53 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return m; | 
					
						
							| 
									
										
										
										
											2016-10-23 23:37:53 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | const float (*GPU_matrix_normal_inverse_get(float m[3][3]))[3] | 
					
						
							| 
									
										
										
										
											2016-10-23 23:37:53 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   if (m == NULL) { | 
					
						
							|  |  |  |     static Mat3 temp3; | 
					
						
							|  |  |  |     m = temp3; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-10-23 23:37:53 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   GPU_matrix_normal_get(m); | 
					
						
							|  |  |  |   invert_m3(m); | 
					
						
							| 
									
										
										
										
											2016-10-23 23:37:53 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   return m; | 
					
						
							| 
									
										
										
										
											2016-10-23 23:37:53 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | void GPU_matrix_bind(const GPUShaderInterface *shaderface) | 
					
						
							| 
									
										
										
										
											2016-10-09 23:03:35 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   /* set uniform values to matrix stack values
 | 
					
						
							|  |  |  |    * call this before a draw call if desired matrices are dirty | 
					
						
							|  |  |  |    * call glUseProgram before this, as glUniform expects program to be bound | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2016-10-09 23:03:35 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-04 14:15:25 +02:00
										 |  |  |   int32_t MV = GPU_shaderinterface_uniform_builtin(shaderface, GPU_UNIFORM_MODELVIEW); | 
					
						
							|  |  |  |   int32_t P = GPU_shaderinterface_uniform_builtin(shaderface, GPU_UNIFORM_PROJECTION); | 
					
						
							|  |  |  |   int32_t MVP = GPU_shaderinterface_uniform_builtin(shaderface, GPU_UNIFORM_MVP); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   int32_t N = GPU_shaderinterface_uniform_builtin(shaderface, GPU_UNIFORM_NORMAL); | 
					
						
							|  |  |  |   int32_t MV_inv = GPU_shaderinterface_uniform_builtin(shaderface, GPU_UNIFORM_MODELVIEW_INV); | 
					
						
							|  |  |  |   int32_t P_inv = GPU_shaderinterface_uniform_builtin(shaderface, GPU_UNIFORM_PROJECTION_INV); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-30 23:50:43 +02:00
										 |  |  |   /* XXX(fclem) this works but this assumes shader is unused inside GPU_shader_uniform_vector. */ | 
					
						
							|  |  |  |   GPUShader *sh = NULL; | 
					
						
							| 
									
										
										
										
											2020-06-04 14:15:25 +02:00
										 |  |  |   if (MV != -1) { | 
					
						
							| 
									
										
										
										
											2020-07-30 23:50:43 +02:00
										 |  |  |     GPU_shader_uniform_vector(sh, MV, 16, 1, (const float *)GPU_matrix_model_view_get(NULL)); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-06-04 14:15:25 +02:00
										 |  |  |   if (P != -1) { | 
					
						
							| 
									
										
										
										
											2020-07-30 23:50:43 +02:00
										 |  |  |     GPU_shader_uniform_vector(sh, P, 16, 1, (const float *)GPU_matrix_projection_get(NULL)); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-06-04 14:15:25 +02:00
										 |  |  |   if (MVP != -1) { | 
					
						
							| 
									
										
										
										
											2020-07-30 23:50:43 +02:00
										 |  |  |     GPU_shader_uniform_vector( | 
					
						
							|  |  |  |         sh, MVP, 16, 1, (const float *)GPU_matrix_model_view_projection_get(NULL)); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-06-04 14:15:25 +02:00
										 |  |  |   if (N != -1) { | 
					
						
							| 
									
										
										
										
											2020-07-30 23:50:43 +02:00
										 |  |  |     GPU_shader_uniform_vector(sh, N, 9, 1, (const float *)GPU_matrix_normal_get(NULL)); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-06-04 14:15:25 +02:00
										 |  |  |   if (MV_inv != -1) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     Mat4 m; | 
					
						
							|  |  |  |     GPU_matrix_model_view_get(m); | 
					
						
							|  |  |  |     invert_m4(m); | 
					
						
							| 
									
										
										
										
											2020-07-30 23:50:43 +02:00
										 |  |  |     GPU_shader_uniform_vector(sh, MV_inv, 16, 1, (const float *)m); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-06-04 14:15:25 +02:00
										 |  |  |   if (P_inv != -1) { | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     Mat4 m; | 
					
						
							|  |  |  |     GPU_matrix_projection_get(m); | 
					
						
							|  |  |  |     invert_m4(m); | 
					
						
							| 
									
										
										
										
											2020-07-30 23:50:43 +02:00
										 |  |  |     GPU_shader_uniform_vector(sh, P_inv, 16, 1, (const float *)m); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-03-26 21:12:39 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-14 15:27:10 +02:00
										 |  |  |   gpu_matrix_state_active_set_dirty(false); | 
					
						
							| 
									
										
										
										
											2016-10-20 14:17:54 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-15 15:27:15 +02:00
										 |  |  | bool GPU_matrix_dirty_get(void) | 
					
						
							| 
									
										
										
										
											2016-10-20 14:17:54 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-08-14 15:27:10 +02:00
										 |  |  |   GPUMatrixState *state = gpu_context_active_matrix_state_get(); | 
					
						
							|  |  |  |   return state->dirty; | 
					
						
							| 
									
										
										
										
											2016-10-09 23:03:35 -04:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2017-08-20 17:23:49 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Python API Helpers
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | BLI_STATIC_ASSERT(GPU_PY_MATRIX_STACK_LEN + 1 == MATRIX_STACK_DEPTH, "define mismatch"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Return int since caller is may subtract. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int GPU_matrix_stack_level_get_model_view(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-08-14 15:27:10 +02:00
										 |  |  |   GPUMatrixState *state = gpu_context_active_matrix_state_get(); | 
					
						
							|  |  |  |   return (int)state->model_view_stack.top; | 
					
						
							| 
									
										
										
										
											2017-08-20 17:23:49 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int GPU_matrix_stack_level_get_projection(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-08-14 15:27:10 +02:00
										 |  |  |   GPUMatrixState *state = gpu_context_active_matrix_state_get(); | 
					
						
							|  |  |  |   return (int)state->projection_stack.top; | 
					
						
							| 
									
										
										
										
											2017-08-20 17:23:49 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \} */ | 
					
						
							| 
									
										
										
										
											2020-07-16 20:24:05 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Polygon Offset Hack
 | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-07-18 14:27:53 +10:00
										 |  |  |  * Workaround the fact that polygon-offset is implementation dependent. | 
					
						
							|  |  |  |  * We modify the projection matrix \a winmat in order to change the final depth a tiny amount. | 
					
						
							| 
									
										
										
										
											2020-07-16 20:24:05 +02:00
										 |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | float GPU_polygon_offset_calc(const float (*winmat)[4], float viewdist, float dist) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   /* Seems like we have a factor of 2 more offset than 2.79 for some reason. Correct for this. */ | 
					
						
							|  |  |  |   dist *= 0.5f; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (winmat[3][3] > 0.5f) { | 
					
						
							|  |  |  | #if 1
 | 
					
						
							|  |  |  |     return 0.00001f * dist * viewdist;  // ortho tweaking
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     static float depth_fac = 0.0f; | 
					
						
							|  |  |  |     if (depth_fac == 0.0f) { | 
					
						
							|  |  |  |       int depthbits; | 
					
						
							|  |  |  |       glGetIntegerv(GL_DEPTH_BITS, &depthbits); | 
					
						
							|  |  |  |       depth_fac = 1.0f / (float)((1 << depthbits) - 1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     offs = (-1.0 / winmat[2][2]) * dist * depth_fac; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     UNUSED_VARS(viewdist); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* This adjustment effectively results in reducing the Z value by 0.25%.
 | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * winmat[4][3] actually evaluates to `-2 * far * near / (far - near)`, | 
					
						
							|  |  |  |    * is very close to -0.2 with default clip range, | 
					
						
							|  |  |  |    * and is used as the coefficient multiplied by `w / z`, | 
					
						
							|  |  |  |    * thus controlling the z dependent part of the depth value. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   return winmat[3][2] * -0.0025f * dist; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * \note \a viewdist is only for ortho at the moment. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void GPU_polygon_offset(float viewdist, float dist) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   static float winmat[4][4], offset = 0.0f; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (dist != 0.0f) { | 
					
						
							|  |  |  |     /* hack below is to mimic polygon offset */ | 
					
						
							|  |  |  |     GPU_matrix_projection_get(winmat); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* dist is from camera to center point */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     float offs = GPU_polygon_offset_calc(winmat, viewdist, dist); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     winmat[3][2] -= offs; | 
					
						
							|  |  |  |     offset += offs; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     winmat[3][2] += offset; | 
					
						
							|  |  |  |     offset = 0.0; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   GPU_matrix_projection_set(winmat); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \} */ |