| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * ***** BEGIN GPL LICENSE BLOCK ***** | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU General Public License | 
					
						
							|  |  |  |  * as published by the Free Software Foundation; either version 2 | 
					
						
							|  |  |  |  * of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the ipmlied 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) 2012 Blender Foundation. | 
					
						
							|  |  |  |  * All rights reserved. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The Original Code is: all of this file. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  |  * Contributor(s): Alexandr Kuznetsov, Jason Wilkins, Mike Erwin | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  |  * | 
					
						
							|  |  |  |  * ***** END GPL LICENSE BLOCK ***** | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \file source/blender/gpu/intern/gpu_matrix.c
 | 
					
						
							|  |  |  |  *  \ingroup gpu | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-21 17:27:17 -04:00
										 |  |  | #define SUPPRESS_GENERIC_MATRIX_API
 | 
					
						
							| 
									
										
										
										
											2017-08-20 17:23:49 +10: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"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-11 14:44:12 -04:00
										 |  |  | #define DEBUG_MATRIX_BIND 0
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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 { | 
					
						
							|  |  |  | 	Mat4 stack[MATRIX_STACK_DEPTH]; | 
					
						
							|  |  |  | 	unsigned int top; | 
					
						
							|  |  |  | } MatrixStack; | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-19 10:13:10 +02:00
										 |  |  | typedef struct { | 
					
						
							|  |  |  | 	MatrixStack model_view_stack; | 
					
						
							|  |  |  | 	MatrixStack projection_stack; | 
					
						
							| 
									
										
										
										
											2016-10-20 14:17:54 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	bool dirty; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +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 | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | } MatrixState; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-14 14:13:38 +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}} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static MatrixState state = { | 
					
						
							| 
									
										
										
										
											2017-04-19 10:13:10 +02:00
										 |  |  | 	.model_view_stack = {{MATRIX_4X4_IDENTITY}, 0}, | 
					
						
							|  |  |  | 	.projection_stack = {{MATRIX_4X4_IDENTITY}, 0}, | 
					
						
							| 
									
										
										
										
											2017-04-15 01:29:25 -04:00
										 |  |  | 	.dirty = true | 
					
						
							| 
									
										
										
										
											2017-04-14 14:13:38 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #undef MATRIX_4X4_IDENTITY
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-19 10:13:10 +02:00
										 |  |  | #define ModelViewStack state.model_view_stack
 | 
					
						
							|  |  |  | #define ModelView ModelViewStack.stack[ModelViewStack.top]
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define ProjectionStack state.projection_stack
 | 
					
						
							|  |  |  | #define Projection ProjectionStack.stack[ProjectionStack.top]
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 01:29:25 -04:00
										 |  |  | void gpuMatrixReset(void) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-04-19 10:13:10 +02:00
										 |  |  | 	state.model_view_stack.top = 0; | 
					
						
							|  |  |  | 	state.projection_stack.top = 0; | 
					
						
							| 
									
										
										
										
											2017-04-15 01:29:25 -04:00
										 |  |  | 	unit_m4(ModelView); | 
					
						
							|  |  |  | 	unit_m4(Projection); | 
					
						
							|  |  |  | 	state.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
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-04-13 04:00:19 -04:00
										 |  |  | 	const int n = 16; | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 	for (int i = 0; i < n; i++) { | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | #if _MSC_VER
 | 
					
						
							|  |  |  | 		BLI_assert(_finite(m[i])); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 		BLI_assert(!isinf(m[i])); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | #define CHECKMAT(m) checkmat((const float*)m)
 | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define CHECKMAT(m)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-09 02:28:55 +11:00
										 |  |  | void gpuPushMatrix(void) | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-04-19 10:13:10 +02:00
										 |  |  | 	BLI_assert(ModelViewStack.top < MATRIX_STACK_DEPTH); | 
					
						
							|  |  |  | 	ModelViewStack.top++; | 
					
						
							|  |  |  | 	copy_m4_m4(ModelView, ModelViewStack.stack[ModelViewStack.top - 1]); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-09 02:28:55 +11:00
										 |  |  | void gpuPopMatrix(void) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-04-19 10:13:10 +02:00
										 |  |  | 	BLI_assert(ModelViewStack.top > 0); | 
					
						
							|  |  |  | 	ModelViewStack.top--; | 
					
						
							|  |  |  | 	state.dirty = true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void gpuPushProjectionMatrix(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	BLI_assert(ProjectionStack.top < MATRIX_STACK_DEPTH); | 
					
						
							|  |  |  | 	ProjectionStack.top++; | 
					
						
							|  |  |  | 	copy_m4_m4(Projection, ProjectionStack.stack[ProjectionStack.top - 1]); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void gpuPopProjectionMatrix(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	BLI_assert(ProjectionStack.top > 0); | 
					
						
							|  |  |  | 	ProjectionStack.top--; | 
					
						
							| 
									
										
										
										
											2016-10-20 14:17:54 -04:00
										 |  |  | 	state.dirty = true; | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 01:29:25 -04:00
										 |  |  | void gpuLoadMatrix(const float m[4][4]) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-04-14 18:06:56 +02:00
										 |  |  | 	copy_m4_m4(ModelView, m); | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 	CHECKMAT(ModelView3D); | 
					
						
							| 
									
										
										
										
											2016-10-20 14:17:54 -04:00
										 |  |  | 	state.dirty = true; | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-19 10:46:05 +02:00
										 |  |  | void gpuLoadIdentityProjectionMatrix(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	unit_m4(Projection); | 
					
						
							|  |  |  | 	CHECKMAT(Projection3D); | 
					
						
							|  |  |  | 	state.dirty = true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 01:29:25 -04:00
										 |  |  | void gpuLoadProjectionMatrix(const float m[4][4]) | 
					
						
							| 
									
										
										
										
											2017-03-21 18:10:20 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-04-14 18:06:56 +02:00
										 |  |  | 	copy_m4_m4(Projection, m); | 
					
						
							| 
									
										
										
										
											2017-03-21 18:10:20 -04:00
										 |  |  | 	CHECKMAT(Projection3D); | 
					
						
							|  |  |  | 	state.dirty = true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-09 02:28:55 +11:00
										 |  |  | void gpuLoadIdentity(void) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-04-14 18:06:56 +02:00
										 |  |  | 	unit_m4(ModelView); | 
					
						
							| 
									
										
										
										
											2016-10-20 14:17:54 -04:00
										 |  |  | 	state.dirty = true; | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | void gpuTranslate2f(float x, float y) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2017-04-13 04:00:19 -04:00
										 |  |  | 	Mat4 m; | 
					
						
							|  |  |  | 	unit_m4(m); | 
					
						
							|  |  |  | 	m[3][0] = x; | 
					
						
							|  |  |  | 	m[3][1] = y; | 
					
						
							| 
									
										
										
										
											2017-04-15 01:29:25 -04:00
										 |  |  | 	gpuMultMatrix(m); | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | void gpuTranslate2fv(const float vec[2]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	gpuTranslate2f(vec[0], vec[1]); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | void gpuTranslate3f(float x, float y, float z) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-10-20 14:17:54 -04:00
										 |  |  | #if 1
 | 
					
						
							| 
									
										
										
										
											2017-04-14 18:06:56 +02:00
										 |  |  | 	translate_m4(ModelView, x, y, z); | 
					
						
							| 
									
										
										
										
											2017-04-15 01:29:25 -04:00
										 |  |  | 	CHECKMAT(ModelView); | 
					
						
							| 
									
										
										
										
											2016-10-11 21:13:13 -04:00
										 |  |  | #else /* above works well in early testing, below is generic version */
 | 
					
						
							|  |  |  | 	Mat4 m; | 
					
						
							|  |  |  | 	unit_m4(m); | 
					
						
							|  |  |  | 	m[3][0] = x; | 
					
						
							|  |  |  | 	m[3][1] = y; | 
					
						
							|  |  |  | 	m[3][2] = z; | 
					
						
							| 
									
										
										
										
											2017-04-15 01:29:25 -04:00
										 |  |  | 	gpuMultMatrix(m); | 
					
						
							| 
									
										
										
										
											2016-10-11 21:13:13 -04:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-10-20 14:17:54 -04:00
										 |  |  | 	state.dirty = true; | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | void gpuTranslate3fv(const float vec[3]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	gpuTranslate3f(vec[0], vec[1], vec[2]); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | void gpuScaleUniform(float factor) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2017-04-14 14:13:38 +02:00
										 |  |  | 	Mat4 m; | 
					
						
							|  |  |  | 	scale_m4_fl(m, factor); | 
					
						
							| 
									
										
										
										
											2017-04-15 01:29:25 -04:00
										 |  |  | 	gpuMultMatrix(m); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | void gpuScale2f(float x, float y) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-04-13 04:00:19 -04:00
										 |  |  | 	Mat4 m = {{0.0f}}; | 
					
						
							|  |  |  | 	m[0][0] = x; | 
					
						
							|  |  |  | 	m[1][1] = y; | 
					
						
							|  |  |  | 	m[2][2] = 1.0f; | 
					
						
							|  |  |  | 	m[3][3] = 1.0f; | 
					
						
							| 
									
										
										
										
											2017-04-15 01:29:25 -04:00
										 |  |  | 	gpuMultMatrix(m); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | void gpuScale2fv(const float vec[2]) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 	gpuScale2f(vec[0], vec[1]); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | void gpuScale3f(float x, float y, float z) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 	Mat4 m = {{0.0f}}; | 
					
						
							|  |  |  | 	m[0][0] = x; | 
					
						
							|  |  |  | 	m[1][1] = y; | 
					
						
							|  |  |  | 	m[2][2] = z; | 
					
						
							|  |  |  | 	m[3][3] = 1.0f; | 
					
						
							| 
									
										
										
										
											2017-04-15 01:29:25 -04:00
										 |  |  | 	gpuMultMatrix(m); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | void gpuScale3fv(const float vec[3]) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 	gpuScale3f(vec[0], vec[1], vec[2]); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 01:29:25 -04:00
										 |  |  | void gpuMultMatrix(const float m[4][4]) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-04-14 18:06:56 +02:00
										 |  |  | 	mul_m4_m4_post(ModelView, m); | 
					
						
							| 
									
										
										
										
											2017-04-15 01:29:25 -04:00
										 |  |  | 	CHECKMAT(ModelView); | 
					
						
							| 
									
										
										
										
											2016-10-20 14:17:54 -04:00
										 |  |  | 	state.dirty = true; | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-21 00:08:09 -04:00
										 |  |  | void gpuRotate2D(float deg) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2017-04-13 04:00:19 -04:00
										 |  |  | 	/* essentially RotateAxis('Z')
 | 
					
						
							|  |  |  | 	 * TODO: simpler math for 2D case | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2017-04-14 18:06:56 +02:00
										 |  |  | 	rotate_m4(ModelView, 'Z', DEG2RADF(deg)); | 
					
						
							| 
									
										
										
										
											2017-03-21 00:08:09 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-12 19:23:08 +01:00
										 |  |  | void gpuRotate3f(float deg, float x, float y, float z) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	const float axis[3] = {x, y, z}; | 
					
						
							|  |  |  | 	gpuRotate3fv(deg, axis); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | void gpuRotate3fv(float deg, const float axis[3]) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 	Mat4 m; | 
					
						
							|  |  |  | 	axis_angle_to_mat4(m, axis, DEG2RADF(deg)); | 
					
						
							| 
									
										
										
										
											2017-04-15 01:29:25 -04:00
										 |  |  | 	gpuMultMatrix(m); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | void gpuRotateAxis(float deg, char axis) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-04-13 04:00:19 -04:00
										 |  |  | 	/* rotate_m4 works in place */ | 
					
						
							| 
									
										
										
										
											2017-04-14 18:06:56 +02:00
										 |  |  | 	rotate_m4(ModelView, axis, DEG2RADF(deg)); | 
					
						
							| 
									
										
										
										
											2017-04-15 01:29:25 -04:00
										 |  |  | 	CHECKMAT(ModelView); | 
					
						
							| 
									
										
										
										
											2016-10-20 14:17:54 -04:00
										 |  |  | 	state.dirty = true; | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | static void mat4_ortho_set(float m[4][4], float left, float right, float bottom, float top, float near, float far) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	m[0][0] = 2.0f / (right - left); | 
					
						
							|  |  |  | 	m[1][0] = 0.0f; | 
					
						
							|  |  |  | 	m[2][0] = 0.0f; | 
					
						
							|  |  |  | 	m[3][0] = -(right + left) / (right - left); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	m[0][1] = 0.0f; | 
					
						
							|  |  |  | 	m[1][1] = 2.0f / (top - bottom); | 
					
						
							|  |  |  | 	m[2][1] = 0.0f; | 
					
						
							|  |  |  | 	m[3][1] = -(top + bottom) / (top - bottom); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	m[0][2] = 0.0f; | 
					
						
							|  |  |  | 	m[1][2] = 0.0f; | 
					
						
							|  |  |  | 	m[2][2] = -2.0f / (far - near); | 
					
						
							|  |  |  | 	m[3][2] = -(far + near) / (far - near); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	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
										 |  |  | 
 | 
					
						
							|  |  |  | 	state.dirty = true; | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-26 21:12:39 -04: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
										 |  |  | { | 
					
						
							|  |  |  | 	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; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	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; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	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); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	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
										 |  |  | 
 | 
					
						
							|  |  |  | 	state.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
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | /* This function is loosely based on Mesa implementation.
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 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. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 	float side[3]; | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 	normalize_v3(lookdir); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 	cross_v3_v3v3(side, lookdir, camup); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 	normalize_v3(side); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 	cross_v3_v3v3(camup, side, lookdir); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +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
										 |  |  | 
 | 
					
						
							|  |  |  | 	state.dirty = true; | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | void gpuOrtho(float left, float right, float bottom, float top, float near, float far) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-04-14 18:06:56 +02:00
										 |  |  | 	mat4_ortho_set(Projection, left, right, bottom, top, near, far); | 
					
						
							| 
									
										
										
										
											2017-04-15 01:29:25 -04:00
										 |  |  | 	CHECKMAT(Projection); | 
					
						
							| 
									
										
										
										
											2016-10-20 14:17:54 -04:00
										 |  |  | 	state.dirty = true; | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | void gpuOrtho2D(float left, float right, float bottom, float top) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 	Mat4 m; | 
					
						
							|  |  |  | 	mat4_ortho_set(m, left, right, bottom, top, -1.0f, 1.0f); | 
					
						
							|  |  |  | 	CHECKMAT(Projection2D); | 
					
						
							| 
									
										
										
										
											2016-10-20 14:17:54 -04:00
										 |  |  | 	state.dirty = true; | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | void gpuFrustum(float left, float right, float bottom, float top, float near, float far) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-04-14 18:06:56 +02:00
										 |  |  | 	mat4_frustum_set(Projection, left, right, bottom, top, near, far); | 
					
						
							| 
									
										
										
										
											2017-04-15 01:29:25 -04:00
										 |  |  | 	CHECKMAT(Projection); | 
					
						
							| 
									
										
										
										
											2016-10-20 14:17:54 -04:00
										 |  |  | 	state.dirty = true; | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | void gpuPerspective(float fovy, float aspect, float near, float far) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	float half_height = tanf(fovy * (float)(M_PI / 360.0)) * near; | 
					
						
							|  |  |  | 	float half_width = half_height * aspect; | 
					
						
							|  |  |  | 	gpuFrustum(-half_width, +half_width, -half_height, +half_height, near, far); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | void gpuLookAt(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
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 	Mat4 cm; | 
					
						
							|  |  |  | 	float lookdir[3]; | 
					
						
							|  |  |  | 	float camup[3] = {upX, upY, upZ}; | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	lookdir[0] = centerX - eyeX; | 
					
						
							|  |  |  | 	lookdir[1] = centerY - eyeY; | 
					
						
							|  |  |  | 	lookdir[2] = centerZ - eyeZ; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	mat4_look_from_origin(cm, lookdir, camup); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 01:29:25 -04:00
										 |  |  | 	gpuMultMatrix(cm); | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 	gpuTranslate3f(-eyeX, -eyeY, -eyeZ); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-25 21:58:23 +01:00
										 |  |  | void gpuProject(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
										 |  |  | { | 
					
						
							|  |  |  | 	float v[4]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-25 21:58:23 +01:00
										 |  |  | 	mul_v4_m4v3(v, model, world); | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 	mul_m4_v4(proj, v); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-25 21:58:23 +01:00
										 |  |  | 	if (v[3] != 0.0f) { | 
					
						
							|  |  |  | 		mul_v3_fl(v, 1.0f / v[3]); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +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; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-25 21:58:23 +01:00
										 |  |  | bool gpuUnProject(const float win[3], const float model[4][4], const float proj[4][4], const int view[4], float world[3]) | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 	float pm[4][4]; | 
					
						
							|  |  |  | 	float in[4]; | 
					
						
							|  |  |  | 	float out[4]; | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 	mul_m4_m4m4(pm, proj, model); | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (!invert_m4(pm)) { | 
					
						
							| 
									
										
										
										
											2017-02-25 21:58:23 +01:00
										 |  |  | 		zero_v3(world); | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 		return false; | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	in[0] = win[0]; | 
					
						
							|  |  |  | 	in[1] = win[1]; | 
					
						
							|  |  |  | 	in[2] = win[2]; | 
					
						
							|  |  |  | 	in[3] = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Map x and y from window coordinates */ | 
					
						
							|  |  |  | 	in[0] = (in[0] - view[0]) / view[2]; | 
					
						
							|  |  |  | 	in[1] = (in[1] - view[1]) / view[3]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 	/* Map to range -1 to +1 */ | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 	in[0] = 2 * in[0] - 1; | 
					
						
							|  |  |  | 	in[1] = 2 * in[1] - 1; | 
					
						
							|  |  |  | 	in[2] = 2 * in[2] - 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	mul_v4_m4v3(out, pm, in); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (out[3] == 0.0f) { | 
					
						
							| 
									
										
										
										
											2017-02-25 21:58:23 +01:00
										 |  |  | 		copy_v3_v3(world, out); | 
					
						
							| 
									
										
										
										
											2016-09-25 19:29:45 +02:00
										 |  |  | 		return false; | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-25 21:58:23 +01:00
										 |  |  | 	mul_v3_v3fl(world, out, 1.0f / out[3]); | 
					
						
							|  |  |  | 	return true; | 
					
						
							| 
									
										
										
										
											2016-09-22 12:11:40 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-10-09 23:03:35 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 17:43:48 +10:00
										 |  |  | const float (*gpuGetModelViewMatrix(float m[4][4]))[4] | 
					
						
							| 
									
										
										
										
											2016-10-09 23:03:35 -04:00
										 |  |  | { | 
					
						
							|  |  |  | 	if (m) { | 
					
						
							| 
									
										
										
										
											2017-04-14 18:06:56 +02:00
										 |  |  | 		copy_m4_m4(m, ModelView); | 
					
						
							| 
									
										
										
										
											2017-04-15 17:43:48 +10:00
										 |  |  | 		return m; | 
					
						
							| 
									
										
										
										
											2016-10-09 23:03:35 -04:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2017-04-15 17:43:48 +10:00
										 |  |  | 		return ModelView; | 
					
						
							| 
									
										
										
										
											2016-10-09 23:03:35 -04:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 17:43:48 +10:00
										 |  |  | const float (*gpuGetProjectionMatrix(float m[4][4]))[4] | 
					
						
							| 
									
										
										
										
											2016-10-09 23:03:35 -04:00
										 |  |  | { | 
					
						
							|  |  |  | 	if (m) { | 
					
						
							| 
									
										
										
										
											2017-04-14 18:06:56 +02:00
										 |  |  | 		copy_m4_m4(m, Projection); | 
					
						
							| 
									
										
										
										
											2017-04-15 17:43:48 +10:00
										 |  |  | 		return m; | 
					
						
							| 
									
										
										
										
											2016-10-09 23:03:35 -04:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2017-04-15 17:43:48 +10:00
										 |  |  | 		return Projection; | 
					
						
							| 
									
										
										
										
											2016-10-09 23:03:35 -04:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 17:43:48 +10:00
										 |  |  | const float (*gpuGetModelViewProjectionMatrix(float m[4][4]))[4] | 
					
						
							| 
									
										
										
										
											2016-10-09 23:03:35 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-02-18 19:53:55 -05:00
										 |  |  | 	if (m == NULL) { | 
					
						
							|  |  |  | 		static Mat4 temp; | 
					
						
							|  |  |  | 		m = temp; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-14 18:06:56 +02:00
										 |  |  | 	mul_m4_m4m4(m, Projection, ModelView); | 
					
						
							| 
									
										
										
										
											2017-04-15 17:43:48 +10:00
										 |  |  | 	return m; | 
					
						
							| 
									
										
										
										
											2016-10-09 23:03:35 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 17:43:48 +10:00
										 |  |  | const float (*gpuGetNormalMatrix(float m[3][3]))[3] | 
					
						
							| 
									
										
										
										
											2016-10-23 23:37:53 -04:00
										 |  |  | { | 
					
						
							|  |  |  | 	if (m == NULL) { | 
					
						
							|  |  |  | 		static Mat3 temp3; | 
					
						
							|  |  |  | 		m = temp3; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 01:29:25 -04:00
										 |  |  | 	copy_m3_m4(m, (const float (*)[4])gpuGetModelViewMatrix(NULL)); | 
					
						
							| 
									
										
										
										
											2016-10-23 23:37:53 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	invert_m3(m); | 
					
						
							|  |  |  | 	transpose_m3(m); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 17:43:48 +10:00
										 |  |  | 	return m; | 
					
						
							| 
									
										
										
										
											2016-10-23 23:37:53 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 17:43:48 +10:00
										 |  |  | const float (*gpuGetNormalMatrixInverse(float m[3][3]))[3] | 
					
						
							| 
									
										
										
										
											2016-10-23 23:37:53 -04:00
										 |  |  | { | 
					
						
							|  |  |  | 	if (m == NULL) { | 
					
						
							|  |  |  | 		static Mat3 temp3; | 
					
						
							|  |  |  | 		m = temp3; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	gpuGetNormalMatrix(m); | 
					
						
							|  |  |  | 	invert_m3(m); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 17:43:48 +10:00
										 |  |  | 	return m; | 
					
						
							| 
									
										
										
										
											2016-10-23 23:37:53 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-19 20:18:04 +10:00
										 |  |  | void gpuBindMatrices(const Gwn_ShaderInterface* shaderface) | 
					
						
							| 
									
										
										
										
											2016-10-09 23:03:35 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-04-12 18:55:32 -04: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
										 |  |  | 	 */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-19 20:18:04 +10:00
										 |  |  | 	const Gwn_ShaderInput *MV = GWN_shaderinterface_uniform_builtin(shaderface, GWN_UNIFORM_MODELVIEW); | 
					
						
							|  |  |  | 	const Gwn_ShaderInput *P = GWN_shaderinterface_uniform_builtin(shaderface, GWN_UNIFORM_PROJECTION); | 
					
						
							|  |  |  | 	const Gwn_ShaderInput *MVP = GWN_shaderinterface_uniform_builtin(shaderface, GWN_UNIFORM_MVP); | 
					
						
							| 
									
										
										
										
											2016-10-11 14:44:12 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-19 20:18:04 +10:00
										 |  |  | 	const Gwn_ShaderInput *N = GWN_shaderinterface_uniform_builtin(shaderface, GWN_UNIFORM_NORMAL); | 
					
						
							|  |  |  | 	const Gwn_ShaderInput *MV_inv = GWN_shaderinterface_uniform_builtin(shaderface, GWN_UNIFORM_MODELVIEW_INV); | 
					
						
							|  |  |  | 	const Gwn_ShaderInput *P_inv = GWN_shaderinterface_uniform_builtin(shaderface, GWN_UNIFORM_PROJECTION_INV); | 
					
						
							| 
									
										
										
										
											2016-10-09 23:03:35 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-14 14:13:38 +02:00
										 |  |  | 	if (MV) { | 
					
						
							|  |  |  | 		#if DEBUG_MATRIX_BIND
 | 
					
						
							| 
									
										
										
										
											2017-04-15 01:29:25 -04:00
										 |  |  | 		puts("setting MV matrix"); | 
					
						
							| 
									
										
										
										
											2017-04-14 14:13:38 +02:00
										 |  |  | 		#endif
 | 
					
						
							| 
									
										
										
										
											2016-10-11 14:44:12 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 17:43:48 +10:00
										 |  |  | 		glUniformMatrix4fv(MV->location, 1, GL_FALSE, (const float *)gpuGetModelViewMatrix(NULL)); | 
					
						
							| 
									
										
										
										
											2016-10-09 23:03:35 -04:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-10-23 23:37:53 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-14 14:13:38 +02:00
										 |  |  | 	if (P) { | 
					
						
							|  |  |  | 		#if DEBUG_MATRIX_BIND
 | 
					
						
							| 
									
										
										
										
											2017-04-15 01:29:25 -04:00
										 |  |  | 		puts("setting P matrix"); | 
					
						
							| 
									
										
										
										
											2017-04-14 14:13:38 +02:00
										 |  |  | 		#endif
 | 
					
						
							| 
									
										
										
										
											2017-03-26 21:12:39 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 17:43:48 +10:00
										 |  |  | 		glUniformMatrix4fv(P->location, 1, GL_FALSE, (const float *)gpuGetProjectionMatrix(NULL)); | 
					
						
							| 
									
										
										
										
											2017-04-14 14:13:38 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-04-13 04:00:19 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-14 14:13:38 +02:00
										 |  |  | 	if (MVP) { | 
					
						
							|  |  |  | 		#if DEBUG_MATRIX_BIND
 | 
					
						
							| 
									
										
										
										
											2017-04-15 01:29:25 -04:00
										 |  |  | 		puts("setting MVP matrix"); | 
					
						
							| 
									
										
										
										
											2017-04-14 14:13:38 +02:00
										 |  |  | 		#endif
 | 
					
						
							| 
									
										
										
										
											2017-04-13 04:00:19 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 17:43:48 +10:00
										 |  |  | 		glUniformMatrix4fv(MVP->location, 1, GL_FALSE, (const float *)gpuGetModelViewProjectionMatrix(NULL)); | 
					
						
							| 
									
										
										
										
											2017-04-14 14:13:38 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-04-13 04:00:19 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-14 14:13:38 +02:00
										 |  |  | 	if (N) { | 
					
						
							|  |  |  | 		#if DEBUG_MATRIX_BIND
 | 
					
						
							| 
									
										
										
										
											2017-04-15 01:29:25 -04:00
										 |  |  | 		puts("setting normal matrix"); | 
					
						
							| 
									
										
										
										
											2017-04-14 14:13:38 +02:00
										 |  |  | 		#endif
 | 
					
						
							| 
									
										
										
										
											2017-04-13 04:00:19 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 17:43:48 +10:00
										 |  |  | 		glUniformMatrix3fv(N->location, 1, GL_FALSE, (const float *)gpuGetNormalMatrix(NULL)); | 
					
						
							| 
									
										
										
										
											2017-04-14 14:13:38 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-04-13 04:00:19 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-14 14:13:38 +02:00
										 |  |  | 	if (MV_inv) { | 
					
						
							|  |  |  | 		Mat4 m; | 
					
						
							| 
									
										
										
										
											2017-04-15 01:29:25 -04:00
										 |  |  | 		gpuGetModelViewMatrix(m); | 
					
						
							| 
									
										
										
										
											2017-04-14 14:13:38 +02:00
										 |  |  | 		invert_m4(m); | 
					
						
							|  |  |  | 		glUniformMatrix4fv(MV_inv->location, 1, GL_FALSE, (const float*) m); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-03-26 21:12:39 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-14 14:13:38 +02:00
										 |  |  | 	if (P_inv) { | 
					
						
							|  |  |  | 		Mat4 m; | 
					
						
							| 
									
										
										
										
											2017-04-15 01:29:25 -04:00
										 |  |  | 		gpuGetProjectionMatrix(m); | 
					
						
							| 
									
										
										
										
											2017-04-14 14:13:38 +02:00
										 |  |  | 		invert_m4(m); | 
					
						
							|  |  |  | 		glUniformMatrix4fv(P_inv->location, 1, GL_FALSE, (const float*) m); | 
					
						
							| 
									
										
										
										
											2017-03-26 21:12:39 -04:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-20 14:17:54 -04:00
										 |  |  | 	state.dirty = false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-09 02:28:55 +11:00
										 |  |  | bool gpuMatricesDirty(void) | 
					
						
							| 
									
										
										
										
											2016-10-20 14:17:54 -04:00
										 |  |  | { | 
					
						
							|  |  |  | 	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) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return (int)state.model_view_stack.top; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int GPU_matrix_stack_level_get_projection(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return (int)state.projection_stack.top; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \} */ |