blob: 04cc7345567240a0a74d9494b925c3e7ac03d3c3 [file] [log] [blame]
/**
* 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 &center);
inline float3x2 make_float3x2_scale(const float2 &xyscale);
inline float3x2 make_float3x2_scale(const float2 &xyscale, const float2 &center);
inline float3x2 make_float3x2_scale(float scale);
inline float3x2 make_float3x2_scale(float scale, const float2 &center);
inline float3x2 make_float3x2_skew(float xrad, float yrad);
inline float3x2 make_float3x2_skew(float xrad, float yrad, const float2 &center);
inline float3x2 make_float3x2_rotation(float rad);
inline float3x2 make_float3x2_rotation(float rad, const float2 &center);
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 &center);
inline float4x4 make_float4x4_scale(const float3 &xyzscale);
inline float4x4 make_float4x4_scale(const float3 &xyzscale, const float3 &center);
inline float4x4 make_float4x4_scale(float scale);
inline float4x4 make_float4x4_scale(float scale, const float3 &center);
inline float4x4 make_float4x4_rotation_x(float rad);
inline float4x4 make_float4x4_rotation_x(float rad, const float3 &center);
inline float4x4 make_float4x4_rotation_y(float rad);
inline float4x4 make_float4x4_rotation_y(float rad, const float3 &center);
inline float4x4 make_float4x4_rotation_z(float rad);
inline float4x4 make_float4x4_rotation_z(float rad, const float3 &center);
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 &center);
inline float3x2 make_float3x2_scale(const float2 &xyscale);
inline float3x2 make_float3x2_scale(const float2 &xyscale, const float2 &center);
inline float3x2 make_float3x2_scale(float scale);
inline float3x2 make_float3x2_scale(float scale, const float2 &center);
inline float3x2 make_float3x2_skew(float xrad, float yrad);
inline float3x2 make_float3x2_skew(float xrad, float yrad, const float2 &center);
inline float3x2 make_float3x2_rotation(float rad);
inline float3x2 make_float3x2_rotation(float rad, const float2 &center);
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 &center);
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 &center);
inline float4x4 make_float4x4_scale(float scale);
inline float4x4 make_float4x4_scale(float scale, const float3 &center);
inline float4x4 make_float4x4_rotation_x(float rad);
inline float4x4 make_float4x4_rotation_x(float rad, const float3 &center);
inline float4x4 make_float4x4_rotation_y(float rad);
inline float4x4 make_float4x4_rotation_y(float rad, const float3 &center);
inline float4x4 make_float4x4_rotation_z(float rad);
inline float4x4 make_float4x4_rotation_z(float rad, const float3 &center);
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