|  | /** | 
|  | * This file has no copyright assigned and is placed in the Public Domain. | 
|  | * This file is part of the mingw-w64 runtime package. | 
|  | * No warranty is given; refer to the file DISCLAIMER.PD within this package. | 
|  | */ | 
|  |  | 
|  | /** | 
|  | * Normal users should include `windowsnumerics.h` instead of this header. | 
|  | * However, the cppwinrt headers set `_WINDOWS_NUMERICS_NAMESPACE_`, | 
|  | * `_WINDOWS_NUMERICS_BEGIN_NAMESPACE_` and `_WINDOWS_NUMERICS_END_NAMESPACE_` | 
|  | * to custom values and include `windowsnumerics.impl.h`. Therefore this shall | 
|  | * be considered a public header, and these macros are public API. | 
|  | */ | 
|  |  | 
|  |  | 
|  | #ifdef min | 
|  | #  pragma push_macro("min") | 
|  | #  undef min | 
|  | #  define _WINDOWS_NUMERICS_IMPL_PUSHED_MIN_ | 
|  | #endif | 
|  |  | 
|  | #ifdef max | 
|  | #  pragma push_macro("max") | 
|  | #  undef max | 
|  | #  define _WINDOWS_NUMERICS_IMPL_PUSHED_MAX_ | 
|  | #endif | 
|  |  | 
|  | #include <algorithm> | 
|  | #include <cmath> | 
|  |  | 
|  | #include "directxmath.h" | 
|  |  | 
|  |  | 
|  | // === Internal macros === | 
|  | #define _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(_ty1, _op, _ty2) \ | 
|  | inline _ty1 &operator _op ## =(_ty1 &val1, _ty2 val2) { \ | 
|  | val1 = operator _op (val1, val2); \ | 
|  | return val1; \ | 
|  | } | 
|  |  | 
|  |  | 
|  | // === Internal functions === | 
|  | _WINDOWS_NUMERICS_BEGIN_NAMESPACE_ { | 
|  | namespace _impl { | 
|  |  | 
|  | #if 0 && defined(__cpp_lib_clamp) | 
|  | using std::clamp; | 
|  | #else | 
|  | constexpr const float &clamp(const float &val, const float &min, const float &max) { | 
|  | return val < min ? min : (val > max ? max : val); | 
|  | } | 
|  | #endif | 
|  |  | 
|  | #if 0 && defined(__cpp_lib_interpolate) | 
|  | using std::lerp; | 
|  | #else | 
|  | constexpr float lerp(float val1, float val2, float amount) { | 
|  | // Don't do (val2 - val1) * amount + val1 as it has worse precision. | 
|  | return val2 * amount + val1 * (1.0f - amount); | 
|  | } | 
|  | #endif | 
|  |  | 
|  | } | 
|  | } _WINDOWS_NUMERICS_END_NAMESPACE_ | 
|  |  | 
|  |  | 
|  | // === Forward decls === | 
|  | _WINDOWS_NUMERICS_BEGIN_NAMESPACE_ { | 
|  |  | 
|  | struct float2; | 
|  | struct float3; | 
|  | struct float4; | 
|  | struct float3x2; | 
|  | struct float4x4; | 
|  | struct plane; | 
|  | struct quaternion; | 
|  |  | 
|  | } _WINDOWS_NUMERICS_END_NAMESPACE_ | 
|  |  | 
|  |  | 
|  | // === float2: Struct and function defs === | 
|  | _WINDOWS_NUMERICS_BEGIN_NAMESPACE_ { | 
|  |  | 
|  | struct float2 { | 
|  | float2() = default; | 
|  | constexpr float2(float x, float y) | 
|  | : x(x), y(y) | 
|  | {} | 
|  | constexpr explicit float2(float val) | 
|  | : x(val), y(val) | 
|  | {} | 
|  |  | 
|  | static constexpr float2 zero() { | 
|  | return float2(0.0f); | 
|  | } | 
|  | static constexpr float2 one() { | 
|  | return float2(1.0f); | 
|  | } | 
|  | static constexpr float2 unit_x() { | 
|  | return { 1.0f, 0.0f }; | 
|  | } | 
|  | static constexpr float2 unit_y() { | 
|  | return { 0.0f, 1.0f }; | 
|  | } | 
|  |  | 
|  | float x; | 
|  | float y; | 
|  | }; | 
|  |  | 
|  | // Forward decl functions | 
|  | inline float length(const float2 &val); | 
|  | inline float length_squared(const float2 &val); | 
|  | inline float distance(const float2 &val1, const float2 &val2); | 
|  | inline float distance_squared(const float2 &val1, const float2 &val2); | 
|  | inline float dot(const float2 &val1, const float2 &val2); | 
|  | inline float2 normalize(const float2 &val); | 
|  | inline float2 reflect(const float2 &vec, const float2 &norm); | 
|  | inline float2 min(const float2 &val1, const float2 &val2); | 
|  | inline float2 max(const float2 &val1, const float2 &val2); | 
|  | inline float2 clamp(const float2 &val, const float2 &min, const float2 &max); | 
|  | inline float2 lerp(const float2 &val1, const float2 &val2, float amount); | 
|  | inline float2 transform(const float2 &pos, const float3x2 &mat); | 
|  | inline float2 transform(const float2 &pos, const float4x4 &mat); | 
|  | inline float2 transform_normal(const float2 &norm, const float3x2 &mat); | 
|  | inline float2 transform_normal(const float2 &norm, const float4x4 &mat); | 
|  | inline float2 transform(const float2 &val, const quaternion &rot); | 
|  |  | 
|  | // Define operators | 
|  | #define _WINDOWS_NUMERICS_IMPL_BINARY_OP(_ty, _op) \ | 
|  | inline _ty operator _op(const _ty &val1, const _ty &val2) { \ | 
|  | return { val1.x _op val2.x, val1.y _op val2.y }; \ | 
|  | } | 
|  | _WINDOWS_NUMERICS_IMPL_BINARY_OP(float2, +) | 
|  | _WINDOWS_NUMERICS_IMPL_BINARY_OP(float2, -) | 
|  | _WINDOWS_NUMERICS_IMPL_BINARY_OP(float2, *) | 
|  | inline float2 operator*(const float2 &val1, float val2) { | 
|  | return { val1.x * val2, val1.y * val2 }; | 
|  | } | 
|  | inline float2 operator*(float val1, const float2 &val2) { | 
|  | return operator*(val2, val1); | 
|  | } | 
|  | _WINDOWS_NUMERICS_IMPL_BINARY_OP(float2, /) | 
|  | inline float2 operator/(const float2 &val1, float val2) { | 
|  | return operator*(val1, 1.0f / val2); | 
|  | } | 
|  | inline float2 operator-(const float2 &val) { | 
|  | return { -val.x, -val.y }; | 
|  | } | 
|  | #undef _WINDOWS_NUMERICS_IMPL_BINARY_OP | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(float2, +, const float2 &) | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(float2, -, const float2 &) | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(float2, *, const float2 &) | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(float2, *, float) | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(float2, /, const float2 &) | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(float2, /, float) | 
|  | inline bool operator==(const float2 &val1, const float2 &val2) { | 
|  | return val1.x == val2.x && val1.y == val2.y; | 
|  | } | 
|  | inline bool operator!=(const float2 &val1, const float2 &val2) { | 
|  | return !operator==(val1, val2); | 
|  | } | 
|  |  | 
|  | } _WINDOWS_NUMERICS_END_NAMESPACE_ | 
|  |  | 
|  |  | 
|  | // === float3: Struct and function defs === | 
|  | _WINDOWS_NUMERICS_BEGIN_NAMESPACE_ { | 
|  |  | 
|  | struct float3 { | 
|  | float3() = default; | 
|  | constexpr float3(float x, float y, float z) | 
|  | : x(x), y(y), z(z) | 
|  | {} | 
|  | constexpr float3(float2 val, float z) | 
|  | : x(val.x), y(val.y), z(z) | 
|  | {} | 
|  | constexpr explicit float3(float val) | 
|  | : x(val), y(val), z(val) | 
|  | {} | 
|  |  | 
|  | static constexpr float3 zero() { | 
|  | return float3(0.0); | 
|  | } | 
|  | static constexpr float3 one() { | 
|  | return float3(1.0); | 
|  | } | 
|  | static constexpr float3 unit_x() { | 
|  | return { 1.0f, 0.0f, 0.0f }; | 
|  | } | 
|  | static constexpr float3 unit_y() { | 
|  | return { 0.0f, 1.0f, 0.0f }; | 
|  | } | 
|  | static constexpr float3 unit_z() { | 
|  | return { 0.0f, 0.0f, 1.0f }; | 
|  | } | 
|  |  | 
|  | float x; | 
|  | float y; | 
|  | float z; | 
|  | }; | 
|  |  | 
|  | // Forward decl functions | 
|  | inline float length(const float3 &val); | 
|  | inline float length_squared(const float3 &val); | 
|  | inline float distance(const float3 &val1, const float3 &val2); | 
|  | inline float distance_squared(const float3 &val1, const float3 &val2); | 
|  | inline float dot(const float3 &val1, const float3 &val2); | 
|  | inline float3 normalize(const float3 &val); | 
|  | inline float3 cross(const float3 &val1, const float3 &val2); | 
|  | inline float3 reflect(const float3 &vec, const float3 &norm); | 
|  | inline float3 min(const float3 &val1, const float3 &val2); | 
|  | inline float3 max(const float3 &val1, const float3 &val2); | 
|  | inline float3 clamp(const float3 &val, const float3 &min, const float3 &max); | 
|  | inline float3 lerp(const float3 &val1, const float3 &val2, float amount); | 
|  | inline float3 transform(const float3 &pos, const float4x4 &mat); | 
|  | inline float3 transform_normal(const float3 &norm, const float4x4 &mat); | 
|  | inline float3 transform(const float3 &val, const quaternion &rot); | 
|  |  | 
|  | // Define operators | 
|  | #define _WINDOWS_NUMERICS_IMPL_BINARY_OP(_ty, _op) \ | 
|  | inline _ty operator _op(const _ty &val1, const _ty &val2) { \ | 
|  | return { val1.x _op val2.x, val1.y _op val2.y, val1.z _op val2.z }; \ | 
|  | } | 
|  | _WINDOWS_NUMERICS_IMPL_BINARY_OP(float3, +) | 
|  | _WINDOWS_NUMERICS_IMPL_BINARY_OP(float3, -) | 
|  | _WINDOWS_NUMERICS_IMPL_BINARY_OP(float3, *) | 
|  | inline float3 operator*(const float3 &val1, float val2) { | 
|  | return { val1.x * val2, val1.y * val2, val1.z * val2 }; | 
|  | } | 
|  | inline float3 operator*(float val1, const float3 &val2) { | 
|  | return operator*(val2, val1); | 
|  | } | 
|  | _WINDOWS_NUMERICS_IMPL_BINARY_OP(float3, /) | 
|  | inline float3 operator/(const float3 &val1, float val2) { | 
|  | return operator*(val1, 1.0f / val2); | 
|  | } | 
|  | inline float3 operator-(const float3 &val) { | 
|  | return { -val.x, -val.y, -val.z }; | 
|  | } | 
|  | #undef _WINDOWS_NUMERICS_IMPL_BINARY_OP | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(float3, +, const float3 &) | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(float3, -, const float3 &) | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(float3, *, const float3 &) | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(float3, *, float) | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(float3, /, const float3 &) | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(float3, /, float) | 
|  | inline bool operator==(const float3 &val1, const float3 &val2) { | 
|  | return val1.x == val2.x && val1.y == val2.y && val1.z == val2.z; | 
|  | } | 
|  | inline bool operator!=(const float3 &val1, const float3 &val2) { | 
|  | return !operator==(val1, val2); | 
|  | } | 
|  |  | 
|  | } _WINDOWS_NUMERICS_END_NAMESPACE_ | 
|  |  | 
|  |  | 
|  | // === float4: Struct and function defs === | 
|  | _WINDOWS_NUMERICS_BEGIN_NAMESPACE_ { | 
|  |  | 
|  | struct float4 { | 
|  | float4() = default; | 
|  | constexpr float4(float x, float y, float z, float w) | 
|  | : x(x), y(y), z(z), w(w) | 
|  | {} | 
|  | constexpr float4(float2 val, float z, float w) | 
|  | : x(val.x), y(val.y), z(z), w(w) | 
|  | {} | 
|  | constexpr float4(float3 val, float w) | 
|  | : x(val.x), y(val.y), z(val.z), w(w) | 
|  | {} | 
|  | constexpr explicit float4(float val) | 
|  | : x(val), y(val), z(val), w(val) | 
|  | {} | 
|  |  | 
|  | static constexpr float4 zero() { | 
|  | return float4(0.0); | 
|  | } | 
|  | static constexpr float4 one() { | 
|  | return float4(1.0); | 
|  | } | 
|  | static constexpr float4 unit_x() { | 
|  | return { 1.0f, 0.0f, 0.0f, 0.0f }; | 
|  | } | 
|  | static constexpr float4 unit_y() { | 
|  | return { 0.0f, 1.0f, 0.0f, 0.0f }; | 
|  | } | 
|  | static constexpr float4 unit_z() { | 
|  | return { 0.0f, 0.0f, 1.0f, 0.0f }; | 
|  | } | 
|  | static constexpr float4 unit_w() { | 
|  | return { 0.0f, 0.0f, 0.0f, 1.0f }; | 
|  | } | 
|  |  | 
|  | float x; | 
|  | float y; | 
|  | float z; | 
|  | float w; | 
|  | }; | 
|  |  | 
|  | // Forward decl functions | 
|  | inline float length(const float4 &val); | 
|  | inline float length_squared(const float4 &val); | 
|  | inline float distance(const float4 &val1, const float4 &val2); | 
|  | inline float distance_squared(const float4 &val1, const float4 &val2); | 
|  | inline float dot(const float4 &val1, const float4 &val2); | 
|  | inline float4 normalize(const float4 &val); | 
|  | inline float4 min(const float4 &val1, const float4 &val2); | 
|  | inline float4 max(const float4 &val1, const float4 &val2); | 
|  | inline float4 clamp(const float4 &val, const float4 &min, const float4 &max); | 
|  | inline float4 lerp(const float4 &val1, const float4 &val2, float amount); | 
|  | inline float4 transform(const float4 &pos, const float4x4 &mat); | 
|  | inline float4 transform4(const float3 &pos, const float4x4 &mat); | 
|  | inline float4 transform4(const float2 &pos, const float4x4 &mat); | 
|  | inline float4 transform(const float4 &val, const quaternion &rot); | 
|  | inline float4 transform4(const float3 &val, const quaternion &rot); | 
|  | inline float4 transform4(const float2 &val, const quaternion &rot); | 
|  |  | 
|  | // Define operators | 
|  | #define _WINDOWS_NUMERICS_IMPL_BINARY_OP(_ty, _op) \ | 
|  | inline _ty operator _op(const _ty &val1, const _ty &val2) { \ | 
|  | return { val1.x _op val2.x, val1.y _op val2.y, val1.z _op val2.z, val1.w _op val2.w }; \ | 
|  | } | 
|  | _WINDOWS_NUMERICS_IMPL_BINARY_OP(float4, +) | 
|  | _WINDOWS_NUMERICS_IMPL_BINARY_OP(float4, -) | 
|  | _WINDOWS_NUMERICS_IMPL_BINARY_OP(float4, *) | 
|  | inline float4 operator*(const float4 &val1, float val2) { | 
|  | return { val1.x * val2, val1.y * val2, val1.z * val2, val1.w * val2 }; | 
|  | } | 
|  | inline float4 operator*(float val1, const float4 &val2) { | 
|  | return operator*(val2, val1); | 
|  | } | 
|  | _WINDOWS_NUMERICS_IMPL_BINARY_OP(float4, /) | 
|  | inline float4 operator/(const float4 &val1, float val2) { | 
|  | return operator*(val1, 1.0f / val2); | 
|  | } | 
|  | inline float4 operator-(const float4 &val) { | 
|  | return { -val.x, -val.y, -val.z, -val.w }; | 
|  | } | 
|  | #undef _WINDOWS_NUMERICS_IMPL_BINARY_OP | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(float4, +, const float4 &) | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(float4, -, const float4 &) | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(float4, *, const float4 &) | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(float4, *, float) | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(float4, /, const float4 &) | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(float4, /, float) | 
|  | inline bool operator==(const float4 &val1, const float4 &val2) { | 
|  | return val1.x == val2.x && val1.y == val2.y && val1.z == val2.z && val2.w == val2.w; | 
|  | } | 
|  | inline bool operator!=(const float4 &val1, const float4 &val2) { | 
|  | return !operator==(val1, val2); | 
|  | } | 
|  |  | 
|  | } _WINDOWS_NUMERICS_END_NAMESPACE_ | 
|  |  | 
|  |  | 
|  | // === float3x2: Struct and function defs === | 
|  | _WINDOWS_NUMERICS_BEGIN_NAMESPACE_ { | 
|  |  | 
|  | struct float3x2 { | 
|  | float3x2() = default; | 
|  | constexpr float3x2( | 
|  | float m11, float m12, | 
|  | float m21, float m22, | 
|  | float m31, float m32 | 
|  | ) | 
|  | : m11(m11), m12(m12) | 
|  | , m21(m21), m22(m22) | 
|  | , m31(m31), m32(m32) | 
|  | {} | 
|  |  | 
|  | static constexpr float3x2 identity() { | 
|  | return { | 
|  | 1.0f, 0.0f, | 
|  | 0.0f, 1.0f, | 
|  | 0.0f, 0.0f, | 
|  | }; | 
|  | } | 
|  |  | 
|  | float m11; float m12; | 
|  | float m21; float m22; | 
|  | float m31; float m32; | 
|  | }; | 
|  |  | 
|  | // Forward decl functions | 
|  | inline float3x2 make_float3x2_translation(const float2 &pos); | 
|  | inline float3x2 make_float3x2_translation(float xpos, float ypos); | 
|  | inline float3x2 make_float3x2_scale(float xscale, float yscale); | 
|  | inline float3x2 make_float3x2_scale(float xscale, float yscale, const float2 ¢er); | 
|  | inline float3x2 make_float3x2_scale(const float2 &xyscale); | 
|  | inline float3x2 make_float3x2_scale(const float2 &xyscale, const float2 ¢er); | 
|  | inline float3x2 make_float3x2_scale(float scale); | 
|  | inline float3x2 make_float3x2_scale(float scale, const float2 ¢er); | 
|  | inline float3x2 make_float3x2_skew(float xrad, float yrad); | 
|  | inline float3x2 make_float3x2_skew(float xrad, float yrad, const float2 ¢er); | 
|  | inline float3x2 make_float3x2_rotation(float rad); | 
|  | inline float3x2 make_float3x2_rotation(float rad, const float2 ¢er); | 
|  | inline bool is_identity(const float3x2 &val); | 
|  | inline float determinant(const float3x2 &val); | 
|  | inline float2 translation(const float3x2 &val); | 
|  | inline bool invert(const float3x2 &val, float3x2 *out); | 
|  | inline float3x2 lerp(const float3x2 &mat1, const float3x2 &mat2, float amount); | 
|  |  | 
|  | // Define operators | 
|  | #define _WINDOWS_NUMERICS_IMPL_BINARY_OP(_ty, _op) \ | 
|  | inline _ty operator _op(const _ty &val1, const _ty &val2) { \ | 
|  | return { \ | 
|  | val1.m11 _op val2.m11, val1.m12 _op val2.m12, \ | 
|  | val1.m21 _op val2.m21, val1.m22 _op val2.m22, \ | 
|  | val1.m31 _op val2.m31, val1.m32 _op val2.m32, \ | 
|  | }; \ | 
|  | } | 
|  | _WINDOWS_NUMERICS_IMPL_BINARY_OP(float3x2, +) | 
|  | _WINDOWS_NUMERICS_IMPL_BINARY_OP(float3x2, -) | 
|  | inline float3x2 operator*(const float3x2 &val1, const float3x2 &val2) { | 
|  | // 2D transformation matrix has an implied 3rd column with (0, 0, 1) | 
|  | const float3 v1r1(val1.m11, val1.m12, 0.0f); | 
|  | const float3 v1r2(val1.m21, val1.m22, 0.0f); | 
|  | const float3 v1r3(val1.m31, val1.m32, 1.0f); | 
|  | const float3 v2c1(val2.m11, val2.m21, val2.m31); | 
|  | const float3 v2c2(val2.m12, val2.m22, val2.m32); | 
|  | // const float3 v2c3(0.0f, 0.0f, 1.0f); | 
|  | return { | 
|  | dot(v1r1, v2c1), dot(v1r1, v2c2), | 
|  | dot(v1r2, v2c1), dot(v1r2, v2c2), | 
|  | dot(v1r3, v2c1), dot(v1r3, v2c2), | 
|  | }; | 
|  | } | 
|  | inline float3x2 operator*(const float3x2 &val1, float val2) { | 
|  | return { | 
|  | val1.m11 * val2, val1.m12 * val2, | 
|  | val1.m21 * val2, val1.m22 * val2, | 
|  | val1.m31 * val2, val1.m32 * val2, | 
|  | }; | 
|  | } | 
|  | inline float3x2 operator-(const float3x2 &val) { | 
|  | return { | 
|  | -val.m11, -val.m12, | 
|  | -val.m21, -val.m22, | 
|  | -val.m31, -val.m32, | 
|  | }; | 
|  | } | 
|  | #undef _WINDOWS_NUMERICS_IMPL_BINARY_OP | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(float3x2, +, const float3x2 &) | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(float3x2, -, const float3x2 &) | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(float3x2, *, const float3x2 &) | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(float3x2, *, float) | 
|  | inline bool operator==(const float3x2 &val1, const float3x2 &val2) { | 
|  | return | 
|  | val1.m11 == val2.m11 && val1.m12 == val2.m12 && | 
|  | val1.m21 == val2.m21 && val1.m22 == val2.m22 && | 
|  | val1.m31 == val2.m31 && val1.m32 == val2.m32; | 
|  | } | 
|  | inline bool operator!=(const float3x2 &val1, const float3x2 &val2) { | 
|  | return !operator==(val1, val2); | 
|  | } | 
|  |  | 
|  | } _WINDOWS_NUMERICS_END_NAMESPACE_ | 
|  |  | 
|  |  | 
|  | // === float4x4: Struct and function defs === | 
|  | _WINDOWS_NUMERICS_BEGIN_NAMESPACE_ { | 
|  |  | 
|  | struct float4x4 { | 
|  | float4x4() = default; | 
|  | constexpr float4x4( | 
|  | float m11, float m12, float m13, float m14, | 
|  | float m21, float m22, float m23, float m24, | 
|  | float m31, float m32, float m33, float m34, | 
|  | float m41, float m42, float m43, float m44 | 
|  | ) | 
|  | : m11(m11), m12(m12), m13(m13), m14(m14) | 
|  | , m21(m21), m22(m22), m23(m23), m24(m24) | 
|  | , m31(m31), m32(m32), m33(m33), m34(m34) | 
|  | , m41(m41), m42(m42), m43(m43), m44(m44) | 
|  | {} | 
|  |  | 
|  | static constexpr float4x4 identity() { | 
|  | return { | 
|  | 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, | 
|  | }; | 
|  | } | 
|  |  | 
|  | float m11; float m12; float m13; float m14; | 
|  | float m21; float m22; float m23; float m24; | 
|  | float m31; float m32; float m33; float m34; | 
|  | float m41; float m42; float m43; float m44; | 
|  | }; | 
|  |  | 
|  | // Forward decl functions | 
|  | inline float4x4 make_float4x4_billboard(const float3 &objpos, const float3 &camerapos, const float3 &cameraup, const float3 &camerafwd); | 
|  | inline float4x4 make_float4x4_constrained_billboard(const float3 &objpos, const float3 &camerapos, const float3 &rotateaxis, const float3 &camerafwd, const float3 &objfwd); | 
|  | inline float4x4 make_float4x4_translation(const float3 &pos); | 
|  | inline float4x4 make_float4x4_translation(float xpos, float ypos, float zpos); | 
|  | inline float4x4 make_float4x4_scale(float xscale, float yscale, float zscale); | 
|  | inline float4x4 make_float4x4_scale(float xscale, float yscale, float zscale, const float3 ¢er); | 
|  | inline float4x4 make_float4x4_scale(const float3 &xyzscale); | 
|  | inline float4x4 make_float4x4_scale(const float3 &xyzscale, const float3 ¢er); | 
|  | inline float4x4 make_float4x4_scale(float scale); | 
|  | inline float4x4 make_float4x4_scale(float scale, const float3 ¢er); | 
|  | inline float4x4 make_float4x4_rotation_x(float rad); | 
|  | inline float4x4 make_float4x4_rotation_x(float rad, const float3 ¢er); | 
|  | inline float4x4 make_float4x4_rotation_y(float rad); | 
|  | inline float4x4 make_float4x4_rotation_y(float rad, const float3 ¢er); | 
|  | inline float4x4 make_float4x4_rotation_z(float rad); | 
|  | inline float4x4 make_float4x4_rotation_z(float rad, const float3 ¢er); | 
|  | inline float4x4 make_float4x4_from_axis_angle(const float3 &axis, float angle); | 
|  | inline float4x4 make_float4x4_perspective_field_of_view(float fov, float aspect, float nearplane, float farplane); | 
|  | inline float4x4 make_float4x4_perspective(float w, float h, float nearplane, float farplane); | 
|  | inline float4x4 make_float4x4_perspective_off_center(float left, float right, float bottom, float top, float nearplane, float farplane); | 
|  | inline float4x4 make_float4x4_orthographic(float w, float h, float znearplane, float zfarplane); | 
|  | inline float4x4 make_float4x4_orthographic_off_center(float left, float right, float bottom, float top, float znearplane, float zfarplane); | 
|  | inline float4x4 make_float4x4_look_at(const float3 &camerapos, const float3 &target, const float3 &cameraup); | 
|  | inline float4x4 make_float4x4_world(const float3 &pos, const float3 &fwd, const float3 &up); | 
|  | inline float4x4 make_float4x4_from_quaternion(const quaternion &quat); | 
|  | inline float4x4 make_float4x4_from_yaw_pitch_roll(float yaw, float pitch, float roll); | 
|  | inline float4x4 make_float4x4_shadow(const float3 &lightdir, const plane &plane); | 
|  | inline float4x4 make_float4x4_reflection(const plane &plane); | 
|  | inline bool is_identity(const float4x4 &val); | 
|  | inline float determinant(const float4x4 &val); | 
|  | inline float3 translation(const float4x4 &val); | 
|  | inline bool invert(const float4x4 &mat, float4x4 *out); | 
|  | inline bool decompose(const float4x4 &mat, float3 *out_scale, quaternion *out_rot, float3 *out_translate); | 
|  | inline float4x4 transform(const float4x4 &val, const quaternion &rot); | 
|  | inline float4x4 transpose(const float4x4 &val); | 
|  | inline float4x4 lerp(const float4x4 &val1, const float4x4 &val2, float amount); | 
|  |  | 
|  | // Define operators | 
|  | #define _WINDOWS_NUMERICS_IMPL_BINARY_OP(_ty, _op) \ | 
|  | inline _ty operator _op(const _ty &val1, const _ty &val2) { \ | 
|  | return { \ | 
|  | val1.m11 _op val2.m11, val1.m12 _op val2.m12, val1.m13 _op val2.m13, val1.m14 _op val2.m14, \ | 
|  | val1.m21 _op val2.m21, val1.m22 _op val2.m22, val1.m23 _op val2.m23, val1.m24 _op val2.m24, \ | 
|  | val1.m31 _op val2.m31, val1.m32 _op val2.m32, val1.m33 _op val2.m33, val1.m34 _op val2.m34, \ | 
|  | val1.m41 _op val2.m41, val1.m42 _op val2.m42, val1.m43 _op val2.m43, val1.m44 _op val2.m44, \ | 
|  | }; \ | 
|  | } | 
|  | _WINDOWS_NUMERICS_IMPL_BINARY_OP(float4x4, +) | 
|  | _WINDOWS_NUMERICS_IMPL_BINARY_OP(float4x4, -) | 
|  | inline float4x4 operator*(const float4x4 &val1, const float4x4 &val2) { | 
|  | const float4 v1r1(val1.m11, val1.m12, val1.m13, val1.m14); | 
|  | const float4 v1r2(val1.m21, val1.m22, val1.m23, val1.m24); | 
|  | const float4 v1r3(val1.m31, val1.m32, val1.m33, val1.m34); | 
|  | const float4 v1r4(val1.m41, val1.m42, val1.m43, val1.m44); | 
|  | const float4 v2c1(val2.m11, val2.m21, val2.m31, val2.m41); | 
|  | const float4 v2c2(val2.m12, val2.m22, val2.m32, val2.m42); | 
|  | const float4 v2c3(val2.m13, val2.m23, val2.m33, val2.m43); | 
|  | const float4 v2c4(val2.m14, val2.m24, val2.m34, val2.m44); | 
|  | return { | 
|  | dot(v1r1, v2c1), dot(v1r1, v2c2), dot(v1r1, v2c3), dot(v1r1, v2c4), | 
|  | dot(v1r2, v2c1), dot(v1r2, v2c2), dot(v1r2, v2c3), dot(v1r2, v2c4), | 
|  | dot(v1r3, v2c1), dot(v1r3, v2c2), dot(v1r3, v2c3), dot(v1r3, v2c4), | 
|  | dot(v1r4, v2c1), dot(v1r4, v2c2), dot(v1r4, v2c3), dot(v1r4, v2c4), | 
|  | }; | 
|  | } | 
|  | inline float4x4 operator*(const float4x4 &val1, float val2) { | 
|  | return { | 
|  | val1.m11 * val2, val1.m12 * val2, val1.m13 * val2, val1.m14 * val2, | 
|  | val1.m21 * val2, val1.m22 * val2, val1.m23 * val2, val1.m24 * val2, | 
|  | val1.m31 * val2, val1.m32 * val2, val1.m33 * val2, val1.m34 * val2, | 
|  | val1.m41 * val2, val1.m42 * val2, val1.m43 * val2, val1.m44 * val2, | 
|  | }; | 
|  | } | 
|  | inline float4x4 operator-(const float4x4 &val) { | 
|  | return { | 
|  | -val.m11, -val.m12, -val.m13, -val.m14, | 
|  | -val.m21, -val.m22, -val.m23, -val.m24, | 
|  | -val.m31, -val.m32, -val.m33, -val.m34, | 
|  | -val.m41, -val.m42, -val.m43, -val.m44, | 
|  | }; | 
|  | } | 
|  | #undef _WINDOWS_NUMERICS_IMPL_BINARY_OP | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(float4x4, +, const float4x4 &) | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(float4x4, -, const float4x4 &) | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(float4x4, *, const float4x4 &) | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(float4x4, *, float) | 
|  | inline bool operator==(const float4x4 &val1, const float4x4 &val2) { | 
|  | return | 
|  | val1.m11 == val2.m11 && val1.m12 == val2.m12 && val1.m13 == val2.m13 && val1.m14 == val2.m14 && | 
|  | val1.m21 == val2.m21 && val1.m22 == val2.m22 && val1.m23 == val2.m23 && val1.m24 == val2.m24 && | 
|  | val1.m31 == val2.m31 && val1.m32 == val2.m32 && val1.m33 == val2.m33 && val1.m34 == val2.m34 && | 
|  | val1.m41 == val2.m41 && val1.m42 == val2.m42 && val1.m43 == val2.m43 && val1.m44 == val2.m44; | 
|  | } | 
|  | inline bool operator!=(const float4x4 &val1, const float4x4 &val2) { | 
|  | return !operator==(val1, val2); | 
|  | } | 
|  |  | 
|  | } _WINDOWS_NUMERICS_END_NAMESPACE_ | 
|  |  | 
|  |  | 
|  | // === plane: Struct and function defs === | 
|  | _WINDOWS_NUMERICS_BEGIN_NAMESPACE_ { | 
|  |  | 
|  | struct plane { | 
|  | plane() = default; | 
|  | constexpr plane(float x, float y, float z, float d) | 
|  | : normal(float3(x, y, z)), d(d) | 
|  | {} | 
|  | constexpr plane(float3 normal, float d) | 
|  | : normal(normal), d(d) | 
|  | {} | 
|  | constexpr explicit plane(float4 val) | 
|  | : normal(float3(val.x, val.y, val.z)), d(val.w) | 
|  | {} | 
|  |  | 
|  | float3 normal; | 
|  | float d; | 
|  | }; | 
|  |  | 
|  | // Forward decl functions | 
|  | inline plane make_plane_from_vertices(const float3 &pt1, const float3 &pt2, const float3 &pt3); | 
|  | inline plane normalize(const plane &val); | 
|  | inline plane transform(const plane &plane, const float4x4 &mat); | 
|  | inline plane transform(const plane &plane, const quaternion &rot); | 
|  | inline float dot(const plane &plane, const float4 &val); | 
|  | inline float dot_coordinate(const plane &plane, const float3 &val); | 
|  | inline float dot_normal(const plane &plane, const float3 &val); | 
|  |  | 
|  | // Define operators | 
|  | inline bool operator==(const plane &val1, const plane &val2) { | 
|  | return val1.normal == val2.normal && val1.d == val2.d; | 
|  | } | 
|  | inline bool operator!=(const plane &val1, const plane &val2) { | 
|  | return !operator==(val1, val2); | 
|  | } | 
|  |  | 
|  | } _WINDOWS_NUMERICS_END_NAMESPACE_ | 
|  |  | 
|  |  | 
|  | // === quaternion: Struct and function defs === | 
|  | _WINDOWS_NUMERICS_BEGIN_NAMESPACE_ { | 
|  |  | 
|  | struct quaternion { | 
|  | quaternion() = default; | 
|  | constexpr quaternion(float x, float y, float z, float w) | 
|  | : x(x), y(y), z(z), w(w) | 
|  | {} | 
|  | constexpr quaternion(float3 vecPart, float scalarPart) | 
|  | : x(vecPart.x), y(vecPart.y), z(vecPart.z), w(scalarPart) | 
|  | {} | 
|  |  | 
|  | static constexpr quaternion identity() { | 
|  | return { 0.0f, 0.0f, 0.0f, 1.0f }; | 
|  | } | 
|  |  | 
|  | float x; | 
|  | float y; | 
|  | float z; | 
|  | float w; | 
|  | }; | 
|  |  | 
|  | // Forward decl functions | 
|  | inline quaternion make_quaternion_from_axis_angle(const float3 &axis, float angle); | 
|  | inline quaternion make_quaternion_from_yaw_pitch_roll(float yaw, float pitch, float roll); | 
|  | inline quaternion make_quaternion_from_rotation_matrix(const float4x4 &mat); | 
|  | inline bool is_identity(const quaternion &val); | 
|  | inline float length(const quaternion &val); | 
|  | inline float length_squared(const quaternion &val); | 
|  | inline float dot(const quaternion &val1, const quaternion &val2); | 
|  | inline quaternion normalize(const quaternion &val); | 
|  | inline quaternion conjugate(const quaternion &val); | 
|  | inline quaternion inverse(const quaternion &val); | 
|  | inline quaternion slerp(const quaternion &val1, const quaternion &val2, float amount); | 
|  | inline quaternion lerp(const quaternion &val1, const quaternion &val2, float amount); | 
|  | inline quaternion concatenate(const quaternion &val1, const quaternion &val2); | 
|  |  | 
|  | // Define operators | 
|  | #define _WINDOWS_NUMERICS_IMPL_BINARY_OP(_ty, _op) \ | 
|  | inline _ty operator _op(const _ty &val1, const _ty &val2) { \ | 
|  | return { val1.x _op val2.x, val1.y _op val2.y, val1.z _op val2.z, val1.w _op val2.w }; \ | 
|  | } | 
|  | _WINDOWS_NUMERICS_IMPL_BINARY_OP(quaternion, +) | 
|  | _WINDOWS_NUMERICS_IMPL_BINARY_OP(quaternion, -) | 
|  | inline quaternion operator*(const quaternion &val1, const quaternion &val2) { | 
|  | return { | 
|  | val1.w * val2.x + val1.x * val2.w + val1.y * val2.z - val1.z * val2.y, | 
|  | val1.w * val2.y - val1.x * val2.z + val1.y * val2.w + val1.z * val2.x, | 
|  | val1.w * val2.z + val1.x * val2.y - val1.y * val2.x + val1.z * val2.w, | 
|  | val1.w * val2.w - val1.x * val2.x - val1.y * val2.y - val1.z * val2.z, | 
|  | }; } | 
|  | inline quaternion operator*(const quaternion &val1, float val2) { | 
|  | return { val1.x * val2, val1.y * val2, val1.z * val2, val1.w * val2 }; | 
|  | } | 
|  | inline quaternion operator/(const quaternion &val1, const quaternion &val2) { | 
|  | return operator*(val1, inverse(val2)); | 
|  | } | 
|  | inline quaternion operator-(const quaternion &val) { | 
|  | return { -val.x, -val.y, -val.z, -val.w }; | 
|  | } | 
|  | #undef _WINDOWS_NUMERICS_IMPL_BINARY_OP | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(quaternion, +, const quaternion &) | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(quaternion, -, const quaternion &) | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(quaternion, *, const quaternion &) | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(quaternion, *, float) | 
|  | _WINDOWS_NUMERICS_IMPL_ASSIGN_OP(quaternion, /, const quaternion &) | 
|  | inline bool operator==(const quaternion &val1, const quaternion &val2) { | 
|  | return val1.x == val2.x && val1.y == val2.y && val1.z == val2.z && val2.w == val2.w; | 
|  | } | 
|  | inline bool operator!=(const quaternion &val1, const quaternion &val2) { | 
|  | return !operator==(val1, val2); | 
|  | } | 
|  |  | 
|  | } _WINDOWS_NUMERICS_END_NAMESPACE_ | 
|  |  | 
|  |  | 
|  | // === Function definitions === | 
|  |  | 
|  | // Define float2 functions | 
|  | _WINDOWS_NUMERICS_BEGIN_NAMESPACE_ { | 
|  |  | 
|  | inline float length(const float2 &val) { | 
|  | return ::std::sqrt(length_squared(val)); | 
|  | } | 
|  | inline float length_squared(const float2 &val) { | 
|  | return val.x * val.x + val.y * val.y; | 
|  | } | 
|  | inline float distance(const float2 &val1, const float2 &val2) { | 
|  | return length(val2 - val1); | 
|  | } | 
|  | inline float distance_squared(const float2 &val1, const float2 &val2) { | 
|  | return length_squared(val2 - val1); | 
|  | } | 
|  | inline float dot(const float2 &val1, const float2 &val2) { | 
|  | return val1.x * val2.x + val1.y * val2.y; | 
|  | } | 
|  | inline float2 normalize(const float2 &val) { | 
|  | return val / length(val); | 
|  | } | 
|  | inline float2 reflect(const float2 &vec, const float2 &norm) { | 
|  | // norm is assumed to be normalized. | 
|  | return vec - 2.0f * dot(vec, norm) * norm; | 
|  | } | 
|  | inline float2 min(const float2 &val1, const float2 &val2) { | 
|  | return { ::std::min(val1.x, val2.x), ::std::min(val1.y, val2.y) }; | 
|  | } | 
|  | inline float2 max(const float2 &val1, const float2 &val2) { | 
|  | return { ::std::max(val1.x, val2.x), ::std::max(val1.y, val2.y) }; | 
|  | } | 
|  | inline float2 clamp(const float2 &val, const float2 &min, const float2 &max) { | 
|  | return { _impl::clamp(val.x, min.x, max.x), _impl::clamp(val.y, min.y, max.y) }; | 
|  | } | 
|  | inline float2 lerp(const float2 &val1, const float2 &val2, float amount) { | 
|  | return { _impl::lerp(val1.x, val2.x, amount), _impl::lerp(val1.y, val2.y, amount) }; | 
|  | } | 
|  | // TODO: impl | 
|  | inline float2 transform(const float2 &pos, const float3x2 &mat); | 
|  | inline float2 transform(const float2 &pos, const float4x4 &mat); | 
|  | inline float2 transform_normal(const float2 &norm, const float3x2 &mat); | 
|  | inline float2 transform_normal(const float2 &norm, const float4x4 &mat); | 
|  | inline float2 transform(const float2 &val, const quaternion &rot) { | 
|  | // See comments in the float3 transform function. | 
|  | quaternion result = rot * quaternion(float3(val.x, val.y, 0.0f), 0.0f) * inverse(rot); | 
|  | return { result.x, result.y }; | 
|  | } | 
|  |  | 
|  | } _WINDOWS_NUMERICS_END_NAMESPACE_ | 
|  |  | 
|  |  | 
|  | // Define float3 functions | 
|  | _WINDOWS_NUMERICS_BEGIN_NAMESPACE_ { | 
|  |  | 
|  | inline float length(const float3 &val) { | 
|  | return ::std::sqrt(length_squared(val)); | 
|  | } | 
|  | inline float length_squared(const float3 &val) { | 
|  | return val.x * val.x + val.y * val.y + val.z * val.z; | 
|  | } | 
|  | inline float distance(const float3 &val1, const float3 &val2) { | 
|  | return length(val2 - val1); | 
|  | } | 
|  | inline float distance_squared(const float3 &val1, const float3 &val2) { | 
|  | return length_squared(val2 - val1); | 
|  | } | 
|  | inline float dot(const float3 &val1, const float3 &val2) { | 
|  | return val1.x * val2.x + val1.y * val2.y + val1.z * val2.z; | 
|  | } | 
|  | inline float3 normalize(const float3 &val) { | 
|  | return val / length(val); | 
|  | } | 
|  | inline float3 cross(const float3 &val1, const float3 &val2) { | 
|  | return { | 
|  | val1.y * val2.z - val2.y * val1.z, | 
|  | val1.z * val2.x - val2.z * val1.x, | 
|  | val1.x * val2.y - val2.x * val1.y, | 
|  | }; | 
|  | } | 
|  | inline float3 reflect(const float3 &vec, const float3 &norm) { | 
|  | // norm is assumed to be normalized. | 
|  | return vec - 2.0f * dot(vec, norm) * norm; | 
|  | } | 
|  | inline float3 min(const float3 &val1, const float3 &val2) { | 
|  | return { ::std::min(val1.x, val2.x), ::std::min(val1.y, val2.y), ::std::min(val1.z, val2.z) }; | 
|  | } | 
|  | inline float3 max(const float3 &val1, const float3 &val2) { | 
|  | return { ::std::max(val1.x, val2.x), ::std::max(val1.y, val2.y), ::std::max(val1.z, val2.z) }; | 
|  | } | 
|  | inline float3 clamp(const float3 &val, const float3 &min, const float3 &max) { | 
|  | return { _impl::clamp(val.x, min.x, max.x), _impl::clamp(val.y, min.y, max.y), _impl::clamp(val.z, min.z, max.z) }; | 
|  | } | 
|  | inline float3 lerp(const float3 &val1, const float3 &val2, float amount) { | 
|  | return { _impl::lerp(val1.x, val2.x, amount), _impl::lerp(val1.y, val2.y, amount), _impl::lerp(val1.z, val2.z, amount) }; | 
|  | } | 
|  | // TODO: impl | 
|  | inline float3 transform(const float3 &pos, const float4x4 &mat); | 
|  | inline float3 transform_normal(const float3 &norm, const float4x4 &mat); | 
|  | inline float3 transform(const float3 &val, const quaternion &rot) { | 
|  | // https://en.wikipedia.org/wiki/Quaternions_and_spatial_rotation#Using_quaternions_as_rotations | 
|  | // If assuming rot is a unit quaternion, this could use | 
|  | // conjugate() instead of inverse() too. | 
|  | // This can be expressed as a matrix operation too, with | 
|  | // https://en.wikipedia.org/wiki/Quaternions_and_spatial_rotation#Quaternion-derived_rotation_matrix | 
|  | // (see make_float4x4_from_quaternion). | 
|  | quaternion result = rot * quaternion(val, 0.0f) * inverse(rot); | 
|  | return { result.x, result.y, result.z }; | 
|  | } | 
|  |  | 
|  | } _WINDOWS_NUMERICS_END_NAMESPACE_ | 
|  |  | 
|  |  | 
|  | // Define float4 functions | 
|  | _WINDOWS_NUMERICS_BEGIN_NAMESPACE_ { | 
|  |  | 
|  | inline float length(const float4 &val) { | 
|  | return ::std::sqrt(length_squared(val)); | 
|  | } | 
|  | inline float length_squared(const float4 &val) { | 
|  | return val.x * val.x + val.y * val.y + val.z * val.z + val.w * val.w; | 
|  | } | 
|  | inline float distance(const float4 &val1, const float4 &val2) { | 
|  | return length(val2 - val1); | 
|  | } | 
|  | inline float distance_squared(const float4 &val1, const float4 &val2) { | 
|  | return length_squared(val2 - val1); | 
|  | } | 
|  | inline float dot(const float4 &val1, const float4 &val2) { | 
|  | return val1.x * val2.x + val1.y * val2.y + val1.z * val2.z + val1.w * val2.w; | 
|  | } | 
|  | inline float4 normalize(const float4 &val) { | 
|  | return val / length(val); | 
|  | } | 
|  | inline float4 min(const float4 &val1, const float4 &val2) { | 
|  | return { | 
|  | ::std::min(val1.x, val2.x), | 
|  | ::std::min(val1.y, val2.y), | 
|  | ::std::min(val1.z, val2.z), | 
|  | ::std::min(val1.w, val2.w), | 
|  | }; | 
|  | } | 
|  | inline float4 max(const float4 &val1, const float4 &val2) { | 
|  | return { | 
|  | ::std::max(val1.x, val2.x), | 
|  | ::std::max(val1.y, val2.y), | 
|  | ::std::max(val1.z, val2.z), | 
|  | ::std::max(val1.w, val2.w), | 
|  | }; | 
|  | } | 
|  | inline float4 clamp(const float4 &val, const float4 &min, const float4 &max) { | 
|  | return { | 
|  | _impl::clamp(val.x, min.x, max.x), | 
|  | _impl::clamp(val.y, min.y, max.y), | 
|  | _impl::clamp(val.z, min.z, max.z), | 
|  | _impl::clamp(val.w, min.w, max.w), | 
|  | }; | 
|  | } | 
|  | inline float4 lerp(const float4 &val1, const float4 &val2, float amount) { | 
|  | return { | 
|  | _impl::lerp(val1.x, val2.x, amount), | 
|  | _impl::lerp(val1.y, val2.y, amount), | 
|  | _impl::lerp(val1.z, val2.z, amount), | 
|  | _impl::lerp(val1.w, val2.w, amount), | 
|  | }; | 
|  | } | 
|  | // TODO: impl | 
|  | inline float4 transform(const float4 &pos, const float4x4 &mat); | 
|  | inline float4 transform4(const float3 &pos, const float4x4 &mat); | 
|  | inline float4 transform4(const float2 &pos, const float4x4 &mat); | 
|  | inline float4 transform(const float4 &val, const quaternion &rot) { | 
|  | // See comments in the float3 transform function. | 
|  | quaternion result = rot * quaternion(float3(val.x, val.y, val.z), 0.0f) * inverse(rot); | 
|  | return { result.x, result.y, result.z, val.w }; | 
|  | } | 
|  | inline float4 transform4(const float3 &val, const quaternion &rot) { | 
|  | quaternion result = rot * quaternion(val, 0.0f) * inverse(rot); | 
|  | return { result.x, result.y, result.z, 1.0f }; | 
|  | } | 
|  | inline float4 transform4(const float2 &val, const quaternion &rot) { | 
|  | quaternion result = rot * quaternion(float3(val.x, val.y, 0.0f), 0.0f) * inverse(rot); | 
|  | return { result.x, result.y, result.z, 1.0f }; | 
|  | } | 
|  |  | 
|  | } _WINDOWS_NUMERICS_END_NAMESPACE_ | 
|  |  | 
|  |  | 
|  | // Define float3x2 functions | 
|  | _WINDOWS_NUMERICS_BEGIN_NAMESPACE_ { | 
|  |  | 
|  | // TODO: impl | 
|  | inline float3x2 make_float3x2_translation(const float2 &pos); | 
|  | inline float3x2 make_float3x2_translation(float xpos, float ypos); | 
|  | inline float3x2 make_float3x2_scale(float xscale, float yscale); | 
|  | inline float3x2 make_float3x2_scale(float xscale, float yscale, const float2 ¢er); | 
|  | inline float3x2 make_float3x2_scale(const float2 &xyscale); | 
|  | inline float3x2 make_float3x2_scale(const float2 &xyscale, const float2 ¢er); | 
|  | inline float3x2 make_float3x2_scale(float scale); | 
|  | inline float3x2 make_float3x2_scale(float scale, const float2 ¢er); | 
|  | inline float3x2 make_float3x2_skew(float xrad, float yrad); | 
|  | inline float3x2 make_float3x2_skew(float xrad, float yrad, const float2 ¢er); | 
|  | inline float3x2 make_float3x2_rotation(float rad); | 
|  | inline float3x2 make_float3x2_rotation(float rad, const float2 ¢er); | 
|  | inline bool is_identity(const float3x2 &val) { | 
|  | return val == float3x2::identity(); | 
|  | } | 
|  | inline float determinant(const float3x2 &val) { | 
|  | // 2D transformation matrix has an implied 3rd column with (0, 0, 1) | 
|  | // det = m11 * m22 * m33 + m12 * m23 * m31 + m13 * m21 * m32 | 
|  | //     - m31 * m22 * m13 - m21 * m12 * m33 - m11 * m32 * m23; | 
|  | return val.m11 * val.m22 - val.m21 * val.m12; | 
|  | } | 
|  | inline float2 translation(const float3x2 &val) { | 
|  | return { val.m31, val.m32 }; | 
|  | } | 
|  | inline bool invert(const float3x2 &val, float3x2 *out); | 
|  | inline float3x2 lerp(const float3x2 &mat1, const float3x2 &mat2, float amount); | 
|  |  | 
|  | } _WINDOWS_NUMERICS_END_NAMESPACE_ | 
|  |  | 
|  |  | 
|  | // Define float4x4 functions | 
|  | _WINDOWS_NUMERICS_BEGIN_NAMESPACE_ { | 
|  |  | 
|  | // TODO: impl | 
|  | inline float4x4 make_float4x4_billboard(const float3 &objpos, const float3 &camerapos, const float3 &cameraup, const float3 &camerafwd); | 
|  | inline float4x4 make_float4x4_constrained_billboard(const float3 &objpos, const float3 &camerapos, const float3 &rotateaxis, const float3 &camerafwd, const float3 &objfwd); | 
|  | inline float4x4 make_float4x4_translation(const float3 &pos) { | 
|  | return { | 
|  | 1.0f,  0.0f,  0.0f,  0.0f, | 
|  | 0.0f,  1.0f,  0.0f,  0.0f, | 
|  | 0.0f,  0.0f,  1.0f,  0.0f, | 
|  | pos.x, pos.y, pos.z, 1.0f | 
|  | }; | 
|  | } | 
|  | inline float4x4 make_float4x4_translation(float xpos, float ypos, float zpos); | 
|  | inline float4x4 make_float4x4_scale(float xscale, float yscale, float zscale); | 
|  | inline float4x4 make_float4x4_scale(float xscale, float yscale, float zscale, const float3 ¢er); | 
|  | inline float4x4 make_float4x4_scale(const float3 &xyzscale) { | 
|  | return { | 
|  | xyzscale.x, 0.0f,       0.0f,       0.0f, | 
|  | 0.0f,       xyzscale.y, 0.0f,       0.0f, | 
|  | 0.0f,       0.0f,       xyzscale.z, 0.0f, | 
|  | 0.0f,       0.0f,       0.0f,       1.0f | 
|  | }; | 
|  | } | 
|  | inline float4x4 make_float4x4_scale(const float3 &xyzscale, const float3 ¢er); | 
|  | inline float4x4 make_float4x4_scale(float scale); | 
|  | inline float4x4 make_float4x4_scale(float scale, const float3 ¢er); | 
|  | inline float4x4 make_float4x4_rotation_x(float rad); | 
|  | inline float4x4 make_float4x4_rotation_x(float rad, const float3 ¢er); | 
|  | inline float4x4 make_float4x4_rotation_y(float rad); | 
|  | inline float4x4 make_float4x4_rotation_y(float rad, const float3 ¢er); | 
|  | inline float4x4 make_float4x4_rotation_z(float rad); | 
|  | inline float4x4 make_float4x4_rotation_z(float rad, const float3 ¢er); | 
|  | inline float4x4 make_float4x4_from_axis_angle(const float3 &axis, float angle); | 
|  | inline float4x4 make_float4x4_perspective_field_of_view(float fov, float aspect, float nearplane, float farplane); | 
|  | inline float4x4 make_float4x4_perspective(float w, float h, float nearplane, float farplane); | 
|  | inline float4x4 make_float4x4_perspective_off_center(float left, float right, float bottom, float top, float nearplane, float farplane); | 
|  | inline float4x4 make_float4x4_orthographic(float w, float h, float znearplane, float zfarplane); | 
|  | inline float4x4 make_float4x4_orthographic_off_center(float left, float right, float bottom, float top, float znearplane, float zfarplane); | 
|  | inline float4x4 make_float4x4_look_at(const float3 &camerapos, const float3 &target, const float3 &cameraup); | 
|  | inline float4x4 make_float4x4_world(const float3 &pos, const float3 &fwd, const float3 &up); | 
|  | inline float4x4 make_float4x4_from_quaternion(const quaternion &quat) { | 
|  | // https://en.wikipedia.org/wiki/Rotation_matrix#Quaternion | 
|  | float xx = quat.x * quat.x; | 
|  | float yy = quat.y * quat.y; | 
|  | float zz = quat.z * quat.z; | 
|  | float xy = quat.x * quat.y; | 
|  | float xz = quat.x * quat.z; | 
|  | float xw = quat.x * quat.w; | 
|  | float yz = quat.y * quat.z; | 
|  | float yw = quat.y * quat.w; | 
|  | float zw = quat.z * quat.w; | 
|  | return { | 
|  | 1.0f - 2.0f*yy - 2.0f*zz, 2.0f*xy + 2.0f*zw,        2.0f*xz - 2.0f*yw,        0.0f, | 
|  | 2.0f*xy - 2.0f*zw,        1.0f - 2.0f*xx - 2.0f*zz, 2.0f*yz + 2.0f*xw,        0.0f, | 
|  | 2.0f*xz + 2.0f*yw,        2.0f*yz - 2.0f*xw,        1.0f - 2.0f*xx - 2.0f*yy, 0.0f, | 
|  | 0.0f,                     0.0f,                     0.0f,                     1.0f | 
|  | }; | 
|  | } | 
|  | inline float4x4 make_float4x4_from_yaw_pitch_roll(float yaw, float pitch, float roll); | 
|  | inline float4x4 make_float4x4_shadow(const float3 &lightdir, const plane &plane); | 
|  | inline float4x4 make_float4x4_reflection(const plane &plane); | 
|  | inline bool is_identity(const float4x4 &val) { | 
|  | return val == float4x4::identity(); | 
|  | } | 
|  | inline float determinant(const float4x4 &val) { | 
|  | const float det_33_44 = (val.m33 * val.m44 - val.m34 * val.m43); | 
|  | const float det_32_44 = (val.m32 * val.m44 - val.m34 * val.m42); | 
|  | const float det_32_43 = (val.m32 * val.m43 - val.m33 * val.m42); | 
|  | const float det_31_44 = (val.m31 * val.m44 - val.m34 * val.m41); | 
|  | const float det_31_43 = (val.m31 * val.m43 - val.m33 * val.m41); | 
|  | const float det_31_42 = (val.m31 * val.m42 - val.m32 * val.m41); | 
|  | return val.m11 * (val.m22 * det_33_44 - val.m23 * det_32_44 + val.m24 * det_32_43) | 
|  | - val.m12 * (val.m21 * det_33_44 - val.m23 * det_31_44 + val.m24 * det_31_43) | 
|  | + val.m13 * (val.m21 * det_32_44 - val.m22 * det_31_44 + val.m24 * det_31_42) | 
|  | - val.m14 * (val.m21 * det_32_43 - val.m22 * det_31_43 + val.m23 * det_31_42); | 
|  | } | 
|  | inline float3 translation(const float4x4 &val) { | 
|  | return { val.m41, val.m42, val.m43 }; | 
|  | } | 
|  | inline bool invert(const float4x4 &mat, float4x4 *out); | 
|  | inline bool decompose(const float4x4 &mat, float3 *out_scale, quaternion *out_rot, float3 *out_translate); | 
|  | inline float4x4 transform(const float4x4 &val, const quaternion &rot) { | 
|  | return val * make_float4x4_from_quaternion(rot); | 
|  | } | 
|  | inline float4x4 transpose(const float4x4 &val) { | 
|  | return  { | 
|  | val.m11, val.m21, val.m31, val.m41, | 
|  | val.m12, val.m22, val.m32, val.m42, | 
|  | val.m13, val.m23, val.m33, val.m43, | 
|  | val.m14, val.m24, val.m34, val.m44, | 
|  | }; | 
|  | } | 
|  | inline float4x4 lerp(const float4x4 &val1, const float4x4 &val2, float amount); | 
|  |  | 
|  | } _WINDOWS_NUMERICS_END_NAMESPACE_ | 
|  |  | 
|  |  | 
|  | // Define plane functions | 
|  | _WINDOWS_NUMERICS_BEGIN_NAMESPACE_ { | 
|  |  | 
|  | // TODO: impl | 
|  | inline plane make_plane_from_vertices(const float3 &pt1, const float3 &pt2, const float3 &pt3); | 
|  | inline plane normalize(const plane &val) { | 
|  | const float invlen = 1.0f / length(val.normal); | 
|  | return { val.normal * invlen, val.d * invlen }; | 
|  | } | 
|  | inline plane transform(const plane &plane, const float4x4 &mat); | 
|  | inline plane transform(const plane &plane, const quaternion &rot) { | 
|  | quaternion result = rot * quaternion(plane.normal, 0.0f) * inverse(rot); | 
|  | return { float3(result.x, result.y, result.z), plane.d }; | 
|  | } | 
|  | inline float dot(const plane &plane, const float4 &val); | 
|  | inline float dot_coordinate(const plane &plane, const float3 &val); | 
|  | inline float dot_normal(const plane &plane, const float3 &val); | 
|  |  | 
|  | } _WINDOWS_NUMERICS_END_NAMESPACE_ | 
|  |  | 
|  |  | 
|  | // Define quaternion functions | 
|  | _WINDOWS_NUMERICS_BEGIN_NAMESPACE_ { | 
|  |  | 
|  | inline quaternion make_quaternion_from_axis_angle(const float3 &axis, float angle) { | 
|  | return quaternion(axis * ::std::sin(angle * 0.5f), ::std::cos(angle * 0.5f)); | 
|  | } | 
|  | inline quaternion make_quaternion_from_yaw_pitch_roll(float yaw, float pitch, float roll) { | 
|  | quaternion yq = make_quaternion_from_axis_angle(float3(0.0f, 1.0f, 0.0f), yaw); | 
|  | quaternion pq = make_quaternion_from_axis_angle(float3(1.0f, 0.0f, 0.0f), pitch); | 
|  | quaternion rq = make_quaternion_from_axis_angle(float3(0.0f, 0.0f, 1.0f), roll); | 
|  | return concatenate(concatenate(rq, pq), yq); | 
|  | } | 
|  | inline quaternion make_quaternion_from_rotation_matrix(const float4x4 &mat) { | 
|  | // https://en.wikipedia.org/wiki/Rotation_matrix#Quaternion | 
|  | float t = mat.m11 + mat.m22 + mat.m33; | 
|  | if (t > 0) { | 
|  | float r = ::std::sqrt(1.0f + t); | 
|  | float s = 1.0f / (2.0f * r); | 
|  | return { (mat.m23 - mat.m32) * s, (mat.m31 - mat.m13) * s, | 
|  | (mat.m12 - mat.m21) * s, r * 0.5f }; | 
|  | } else if (mat.m11 >= mat.m22 && mat.m11 >= mat.m33) { | 
|  | float r = ::std::sqrt(1.0f + mat.m11 - mat.m22 - mat.m33); | 
|  | float s = 1.0f / (2.0f * r); | 
|  | return { r * 0.5f, (mat.m21 + mat.m12) * s, | 
|  | (mat.m13 + mat.m31) * s, (mat.m23 - mat.m32) * s }; | 
|  | } else if (mat.m22 >= mat.m11 && mat.m22 >= mat.m33) { | 
|  | float r = ::std::sqrt(1.0f + mat.m22 - mat.m11 - mat.m33); | 
|  | float s = 1.0f / (2.0f * r); | 
|  | return { (mat.m21 + mat.m12) * s, r * 0.5f, | 
|  | (mat.m32 + mat.m23) * s, (mat.m31 - mat.m13) * s }; | 
|  | } else { | 
|  | float r = ::std::sqrt(1.0f + mat.m33 - mat.m11 - mat.m22); | 
|  | float s = 1.0f / (2.0f * r); | 
|  | return { (mat.m13 + mat.m31) * s, (mat.m32 + mat.m23) * s, | 
|  | r * 0.5f, (mat.m12 - mat.m21) * s }; | 
|  | } | 
|  | } | 
|  | inline bool is_identity(const quaternion &val) { | 
|  | return val == quaternion::identity(); | 
|  | } | 
|  | inline float length(const quaternion &val) { | 
|  | return ::std::sqrt(length_squared(val)); | 
|  | } | 
|  | inline float length_squared(const quaternion &val) { | 
|  | return dot(val, val); | 
|  | } | 
|  | inline float dot(const quaternion &val1, const quaternion &val2) { | 
|  | return val1.x * val2.x + val1.y * val2.y + val1.z * val2.z + val1.w * val2.w; | 
|  | } | 
|  | inline quaternion normalize(const quaternion &val) { | 
|  | return operator*(val, 1.0f / length(val)); | 
|  | } | 
|  | inline quaternion conjugate(const quaternion &val) { | 
|  | return { -val.x, -val.y, -val.z, val.w}; | 
|  | } | 
|  | inline quaternion inverse(const quaternion &val) { | 
|  | return operator*(conjugate(val), 1.0f / length_squared(val)); | 
|  | } | 
|  | inline quaternion slerp(const quaternion &val1, const quaternion &val2, float amount) { | 
|  | // https://en.wikipedia.org/wiki/Slerp#Geometric_Slerp | 
|  | float cosangle = dot(val1, val2); | 
|  | quaternion end = val2; | 
|  | if (cosangle < 0.0f) { | 
|  | end = -val2; | 
|  | cosangle = -cosangle; | 
|  | } | 
|  | float fact1, fact2; | 
|  | const float epsilon = 1.0e-6f; | 
|  | if (cosangle > 1.0f - epsilon) { | 
|  | // Very small rotation range, or non-normalized input quaternions. | 
|  | fact1 = (1.0f - amount); | 
|  | fact2 = amount; | 
|  | } else { | 
|  | float angle = ::std::acos(cosangle); | 
|  | float sinangle = ::std::sin(angle); | 
|  | fact1 = ::std::sin((1.0f - amount) * angle) / sinangle; | 
|  | fact2 = ::std::sin(amount * angle) / sinangle; | 
|  | } | 
|  | return val1 * fact1 + end * fact2; | 
|  | } | 
|  | inline quaternion lerp(const quaternion &val1, const quaternion &val2, float amount) { | 
|  | quaternion end = val2; | 
|  | if (dot(val1, val2) < 0.0f) | 
|  | end = -val2; | 
|  | return normalize(quaternion( | 
|  | _impl::lerp(val1.x, end.x, amount), _impl::lerp(val1.y, end.y, amount), | 
|  | _impl::lerp(val1.z, end.z, amount), _impl::lerp(val1.w, end.w, amount) | 
|  | )); | 
|  | } | 
|  | inline quaternion concatenate(const quaternion &val1, const quaternion &val2) { | 
|  | return val2 * val1; | 
|  | } | 
|  |  | 
|  | } _WINDOWS_NUMERICS_END_NAMESPACE_ | 
|  |  | 
|  |  | 
|  | /** | 
|  | * FIXME: Implement interop functions with DirectXMath. | 
|  | * This is where we are supposed to define the functions to convert between | 
|  | * Windows::Foundation::Numerics types and XMVECTOR / XMMATRIX. But our | 
|  | * directxmath.h does not contain the definitions for these types... | 
|  | */ | 
|  | #if 0 | 
|  | // === DirectXMath Interop === | 
|  | namespace DirectX { | 
|  |  | 
|  | // TODO: impl | 
|  | XMVECTOR XMLoadFloat2(const _WINDOWS_NUMERICS_NAMESPACE_::float2 *src); | 
|  | XMVECTOR XMLoadFloat3(const _WINDOWS_NUMERICS_NAMESPACE_::float3 *src); | 
|  | XMVECTOR XMLoadFloat4(const _WINDOWS_NUMERICS_NAMESPACE_::float4 *src); | 
|  | XMMATRIX XMLoadFloat3x2(const _WINDOWS_NUMERICS_NAMESPACE_::float3x2 *src); | 
|  | XMMATRIX XMLoadFloat4x4(const _WINDOWS_NUMERICS_NAMESPACE_::float4x4 *src); | 
|  | XMVECTOR XMLoadPlane(const _WINDOWS_NUMERICS_NAMESPACE_::plane *src); | 
|  | XMVECTOR XMLoadQuaternion(const _WINDOWS_NUMERICS_NAMESPACE_::quaternion *src); | 
|  | void XMStoreFloat2(_WINDOWS_NUMERICS_NAMESPACE_::float2 *out, FXMVECTOR in); | 
|  | void XMStoreFloat3(_WINDOWS_NUMERICS_NAMESPACE_::float3 *out, FXMVECTOR in); | 
|  | void XMStoreFloat4(_WINDOWS_NUMERICS_NAMESPACE_::float4 *out, FXMVECTOR in); | 
|  | void XMStoreFloat3x2(_WINDOWS_NUMERICS_NAMESPACE_::float3x2 *out, FXMMATRIX in); | 
|  | void XMStoreFloat4x4(_WINDOWS_NUMERICS_NAMESPACE_::float4x4 *out, FXMMATRIX in); | 
|  | void XMStorePlane(_WINDOWS_NUMERICS_NAMESPACE_::plane *out, FXMVECTOR in); | 
|  | void XMStoreQuaternion(_WINDOWS_NUMERICS_NAMESPACE_::quaternion *out, FXMVECTOR in); | 
|  |  | 
|  | } /* namespace DirectX */ | 
|  | #endif | 
|  |  | 
|  |  | 
|  | #undef _WINDOWS_NUMERICS_IMPL_ASSIGN_OP | 
|  |  | 
|  | #ifdef _WINDOWS_NUMERICS_IMPL_PUSHED_MIN_ | 
|  | #  undef _WINDOWS_NUMERICS_IMPL_PUSHED_MIN_ | 
|  | #  pragma pop_macro("min") | 
|  | #endif | 
|  |  | 
|  | #ifdef _WINDOWS_NUMERICS_IMPL_PUSHED_MAX_ | 
|  | #  undef _WINDOWS_NUMERICS_IMPL_PUSHED_MAX_ | 
|  | #  pragma pop_macro("max") | 
|  | #endif |