-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathcamera_math_default.h
271 lines (238 loc) · 8.47 KB
/
camera_math_default.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
/*
* INFO:
*
* This file provides an interface to math functions for camera.h
* This is the default implementation, relying only on math.h
*
* In order to reduce the duplication of math-code you can create your own
* camera_math.h implementing the required functions with the math library of your choice.
* This also allowes you to pass and receive arguments without the need to convert them.
* See camera_math_bx.h as an example.
*
*
* LICENSE:
*
* MIT License
*
* Copyright (c) 2022 Crydsch Cube
*
* 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 and this permission notice 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 THE
* AUTHORS OR COPYRIGHT HOLDERS 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.
*
*
* Parts of this file use implementations from the project bx.
* => https://github.com/bkaradzic/bx
* Specifically from the master branch and the commit with the sha a2dff8e0a0ecf705b264062656b3ddc1c9bd4569.
* The copied and adapted code segments are marked as such.
*
* For these parts the following BSD 2-Clause "Simplified" License applies:
*
* Copyright 2010-2022 Branimir Karadzic
*
* Redistribution and use in source and binary forms, with or without modification,
* are permitted provided that the following conditions are met:
*
* 1. Redistributions of source code must retain the above copyright notice, this
* list of conditions and the following disclaimer.
*
* 2. Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
* IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
* OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE
* OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
* OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include <math.h>
struct vec3 {
float x, y, z;
};
typedef struct vec3 Vec3;
#define CameraVec3 Vec3
struct quat {
float x, y, z, w;
};
typedef struct quat Quat;
#define CameraQuat Quat
static inline CameraVec3 cm_init_vec3(float _x, float _y, float _z) {
return { _x, _y, _z };
}
static inline CameraQuat cm_init_quat(float _x, float _y, float _z, float _w) {
return { _x, _y, _z, _w};
}
static inline CameraQuat cm_invert(CameraQuat _a) {
_a.x = -_a.x;
_a.y = -_a.y;
_a.z = -_a.z;
return _a;
}
static inline CameraQuat cm_mulQuat(CameraQuat _a, CameraQuat _b) {
/// Adapted from bx/include/bx/inline/math.inl#L836 function mul
const float ax = _a.x;
const float ay = _a.y;
const float az = _a.z;
const float aw = _a.w;
const float bx = _b.x;
const float by = _b.y;
const float bz = _b.z;
const float bw = _b.w;
return cm_init_quat(
aw * bx + ax * bw + ay * bz - az * by,
aw * by - ax * bz + ay * bw + az * bx,
aw * bz + ax * by - ay * bx + az * bw,
aw * bw - ax * bx - ay * by - az * bz
);
/// Adaption end
}
static inline CameraVec3 cm_mul(CameraVec3 _v, CameraQuat _q) {
/// Adapted from bx/include/bx/inline/math.inl#L857 function mul
const CameraQuat tmp0 = cm_invert(_q);
const CameraQuat qv = cm_init_quat(_v.x, _v.y, _v.z, 0.0f);
const CameraQuat tmp1 = cm_mulQuat(tmp0, qv);
/// Adaption end
const CameraQuat result = cm_mulQuat(tmp1, _q);
return cm_init_vec3(result.x, result.y, result.z);
}
static inline CameraVec3 cm_add(CameraVec3 _a, CameraVec3 _b) {
_a.x += _b.x;
_a.y += _b.y;
_a.z += _b.z;
return _a;
}
static inline CameraVec3 cm_scale(CameraVec3 _a, float _b) {
_a.x *= _b;
_a.y *= _b;
_a.z *= _b;
return _a;
}
static inline CameraVec3 cm_cross(CameraVec3 _a, CameraVec3 _b) {
/// Adapted from bx/include/bx/inline/math.inl#L617 function cross
return cm_init_vec3(
_a.y * _b.z - _a.z * _b.y,
_a.z * _b.x - _a.x * _b.z,
_a.x * _b.y - _a.y * _b.x
);
/// Adaption end
}
static inline float cm_min(float _a, float _b) {
return (_a < _b) ? _a : _b;
}
static inline float cm_max(float _a, float _b) {
return (_a > _b) ? _a : _b;
}
static inline float cm_sqrt(float _a) {
return sqrtf(_a);
}
static inline CameraVec3 cm_toEuler(CameraQuat _a) {
/// Adapted from bx/include/bx/inline/math.inl#L927 function toEuler
const float xx = _a.x;
const float yy = _a.y;
const float zz = _a.z;
const float ww = _a.w;
const float xsq = xx * xx;
const float ysq = yy * yy;
const float zsq = zz * zz;
return cm_init_vec3(
atan2f(2.0f * (xx * ww - yy * zz), 1.0f - 2.0f * (xsq + zsq)),
atan2f(2.0f * (yy * ww + xx * zz), 1.0f - 2.0f * (ysq + zsq)),
asinf(2.0f * (xx * yy + zz * ww))
);
/// Adaption end
}
static inline CameraQuat cm_fromAxisAngle(CameraVec3 _axis, float _angle) {
/// Adapted from bx/include/bx/inline/math.inl#L996 function fromAxisAngle
const float ha = _angle * 0.5f;
const float sa = sinf(ha);
return cm_init_quat(
_axis.x * sa,
_axis.y * sa,
_axis.z * sa,
cosf(ha)
);
/// Adaption end
}
static inline CameraQuat cm_normalizeQuat(CameraQuat _a) {
/// Adapted from bx/include/bx/inline/math.inl#L877 function normalize
const float norm = _a.x * _a.x + _a.y * _a.y + _a.z * _a.z + _a.w * _a.w;
if (0.0f < norm)
{
const float invNorm = powf(norm, -0.5f);
_a.x *= invNorm;
_a.y *= invNorm;
_a.z *= invNorm;
_a.w *= invNorm;
return _a;
}
return cm_init_quat(0.0f, 0.0f, 0.0f, 1.0f);
/// Adaption end
}
static inline CameraVec3 cm_normalizeVec3(CameraVec3 _a) {
/// Adapted from bx/include/bx/inline/math.inl#L663 function normalize
const float invLen = 1.0f / sqrtf(_a.x * _a.x + _a.y * _a.y + _a.z * _a.z);
return cm_scale(_a, invLen);
/// Adaption end
}
static inline CameraVec3 cm_negate(CameraVec3 _a) {
_a.x = -_a.x;
_a.y = -_a.y;
_a.z = -_a.z;
return _a;
}
static inline void cm_matrixFromQuat(float* _result, CameraQuat _rotation) {
/// Adapted from bx/include/bx/inline/math.inl#L1142 function mtxFromQuaternion
const float qx = _rotation.x;
const float qy = _rotation.y;
const float qz = _rotation.z;
const float qw = _rotation.w;
const float x2 = qx + qx;
const float y2 = qy + qy;
const float z2 = qz + qz;
const float x2x = x2 * qx;
const float x2y = x2 * qy;
const float x2z = x2 * qz;
const float x2w = x2 * qw;
const float y2y = y2 * qy;
const float y2z = y2 * qz;
const float y2w = y2 * qw;
const float z2z = z2 * qz;
const float z2w = z2 * qw;
_result[0] = 1.0f - (y2y + z2z);
_result[1] = x2y - z2w;
_result[2] = x2z + y2w;
_result[3] = 0.0f;
_result[4] = x2y + z2w;
_result[5] = 1.0f - (x2x + z2z);
_result[6] = y2z - x2w;
_result[7] = 0.0f;
_result[8] = x2z - y2w;
_result[9] = y2z + x2w;
_result[10] = 1.0f - (x2x + y2y);
_result[11] = 0.0f;
_result[12] = 0.0f;
_result[13] = 0.0f;
_result[14] = 0.0f;
_result[15] = 1.0f;
/// Adaption end
}